Как разбить строковый литерал на несколько строк в C / Objective-C? - PullRequest
304 голосов
/ 28 апреля 2009

У меня довольно длинный sqlite-запрос:

const char *sql_query = "SELECT statuses.word_id FROM lang1_words, statuses WHERE statuses.word_id = lang1_words.word_id ORDER BY lang1_words.word ASC";

Как мне разбить его на несколько строк, чтобы было легче читать? Если я сделаю следующее:

const char *sql_query = "SELECT word_id
                        FROM table1, table2
                        WHERE table2.word_id = table1.word_id
                        ORDER BY table1.word ASC";

Я получаю ошибку.

Есть ли способ писать запросы в несколько строк?

Ответы [ 8 ]

539 голосов
/ 28 апреля 2009

Есть два способа разбить строки на несколько строк:

Использование \

Все строки в C можно разбить на несколько строк, используя \.

Обычный C:

char *my_string = "Line 1 \
                   Line 2";

Objective-C:

NSString *my_string = @"Line1 \
                        Line2";

Лучший подход

Есть лучший подход, который работает только для строк.

Обычный C:

char *my_string = "Line 1 "
                  "Line 2";

Objective-C:

NSString *my_string = @"Line1 "
                       "Line2";    // the second @ is optional

Второй подход лучше, потому что там не так много пробелов. Однако для SQL-запроса возможны оба варианта.

ПРИМЕЧАНИЕ. С помощью #define вам нужно добавить дополнительный '\' для объединения двух строк:

Обычный C:

#define kMyString "Line 1"\
                  "Line 2"
101 голосов
/ 01 августа 2013

Есть хитрость, которую вы можете сделать с препроцессором.
У него есть потенциальные отрицательные стороны, что он разрушит пробелы, и это может сбить с толку людей, читающих код.
Но есть и обратная сторона: вам не нужно экранировать кавычки внутри него.

#define QUOTE(...) #__VA_ARGS__
const char *sql_query = QUOTE(
    SELECT word_id
    FROM table1, table2
    WHERE table2.word_id = table1.word_id
    ORDER BY table1.word ASC
);

препроцессор превращает это в:

const char *sql_query = "SELECT word_id FROM table1, table2 WHERE table2.word_id = table1.word_id ORDER BY table1.word ASC";

Я использовал этот трюк, когда писал некоторые модульные тесты, в которых были большие литеральные строки, содержащие JSON. Это означало, что мне не нужно было избегать каждого символа кавычки \ ".

24 голосов
/ 16 июня 2010

Вы также можете зайти в XCode -> «Настройки», выбрать вкладку «Отступ» и включить перенос строк.

Таким образом, вам не нужно будет ничего вводить, и это сработает для того, что вы уже написали. : -)

Одна неприятная вещь, хотя ...

if (you're long on indentation
    && short on windows) {
            then your code will
                end up squished
                     against th
                         e side
                             li
                              k
                              e

                              t
                              h
                              i
                              s
}
20 голосов
/ 27 июня 2012

У меня постоянно возникает эта проблема, поэтому я создал крошечный инструмент для преобразования текста в экранированную многострочную строку Objective-C:

http://multilineobjc.herokuapp.com/

Надеюсь, это сэкономит вам время.

18 голосов
/ 30 апреля 2014

Расширение идеи Цитата для Objective-C:

#define NSStringMultiline(...) [[NSString alloc] initWithCString:#__VA_ARGS__ encoding:NSUTF8StringEncoding]

NSString *sql = NSStringMultiline(
    SELECT name, age
    FROM users
    WHERE loggedin = true
);
5 голосов
/ 02 июня 2017

Еще одно решение для кучи, измените ваш файл .m на .mm, чтобы он стал Objective-C ++, и использовали литералы C ++, например:

const char *sql_query = R"(SELECT word_id
                           FROM table1, table2
                           WHERE table2.word_id = table1.word_id
                           ORDER BY table1.word ASC)";

Необработанные литералы игнорируют все до последовательности завершения, которая в случае по умолчанию является кавычкой.

Если в строке где-то должна присутствовать последовательность в скобках, вы также можете легко указать собственный разделитель, например:

const char *sql_query = R"T3RM!N8(
                                  SELECT word_id
                                  FROM table1, table2
                                  WHERE table2.word_id = table1.word_id
                                  ORDER BY table1.word ASC
                         )T3RM!N8";
3 голосов
/ 09 июня 2010

Вы также можете сделать:

NSString * query = @"SELECT * FROM foo "
                   @"WHERE "
                     @"bar = 42 "
                     @"AND baz = datetime() "
                   @"ORDER BY fizbit ASC";
0 голосов

GCC добавляет C ++ многострочные необработанные строковые литералы как расширение C

C ++ 11 имеет необработанные строковые литералы, как указано в: https://stackoverflow.com/a/44337236/895245

Однако GCC также добавляет их как расширение C, вам просто нужно использовать -std=gnu99 вместо -std=c99. E.g.:

main.c

#include <assert.h>
#include <string.h>

int main(void) {
    assert(strcmp(R"(
a
b
)", "\na\nb\n") == 0);
}

Скомпилируйте и запустите:

gcc -o main -pedantic -std=gnu99 -Wall -Wextra main.c
./main

Это можно использовать, например, для вставки многострочной встроенной сборки в код C: Как написать код многострочной встроенной сборки в GCC C ++?

Теперь вам нужно просто откинуться на спинку кресла и дождаться его стандартизации на C20XY.

C ++ спросили по адресу: C ++ многострочный строковый литерал

Протестировано на Ubuntu 16.04, GCC 6.4.0, binutils 2.26.1.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...