如何编写多行字符串文字

发布于 2024-07-29 02:00:42 字数 96 浏览 11 评论 0原文

有没有办法在 C++ 中使用 Perl 那样的多行纯文本、常量文字? 也许有一些解析文件的技巧?

我知道你可以用 C++11 中的原始字符串来做到这一点。

Is there any way to have multi-line plain-text, constant literals in C++, à la Perl?
Maybe some parsing trick with #includeing a file?

I know you can do it with raw strings in C++11.

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(11

安静被遗忘 2024-08-05 02:00:42

嗯……有点。 最简单的方法是仅利用相邻字符串文字由编译器连接的事实:

const char *text =
  "This text is pretty long, but will be "
  "concatenated into just a single string. "
  "The disadvantage is that you have to quote "
  "each part, and newlines must be literal as "
  "usual.";

缩进并不重要,因为它不在引号内。

您也可以这样做,只要注意转义嵌入的换行符即可。 如果不这样做,就像我的第一个答案一样,将无法编译:

const char *text2 =
  "Here, on the other hand, I've gone crazy \
and really let the literal span several lines, \
without bothering with quoting each line's \
content. This works, but you can't indent.";

再次注意每行末尾的反斜杠,它们必须紧接在行结束之前,它们在源代码中转义换行符,以便一切都充当如果换行符不存在。 在字符串中有反斜杠的位置不会出现换行符。 使用这种形式,您显然无法缩进文本,因为缩进将成为字符串的一部分,并用随机空格混淆它。

Well ... Sort of. The easiest is to just use the fact that adjacent string literals are concatenated by the compiler:

const char *text =
  "This text is pretty long, but will be "
  "concatenated into just a single string. "
  "The disadvantage is that you have to quote "
  "each part, and newlines must be literal as "
  "usual.";

The indentation doesn't matter, since it's not inside the quotes.

You can also do this, as long as you take care to escape the embedded newline. Failure to do so, like my first answer did, will not compile:

const char *text2 =
  "Here, on the other hand, I've gone crazy \
and really let the literal span several lines, \
without bothering with quoting each line's \
content. This works, but you can't indent.";

Again, note those backslashes at the end of each line, they must be immediately before the line ends, they are escaping the newline in the source, so that everything acts as if the newline wasn't there. You don't get newlines in the string at the locations where you had backslashes. With this form, you obviously can't indent the text since the indentation would then become part of the string, garbling it with random spaces.

゛时过境迁 2024-08-05 02:00:42

在 C++11 中,您拥有原始字符串文字。 有点像 shell 和脚本语言(如 Python、Perl 和 Ruby)中的此处文本。

const char * vogon_poem = R"V0G0N(
             O freddled gruntbuggly thy micturations are to me
                 As plured gabbleblochits on a lurgid bee.
              Groop, I implore thee my foonting turlingdromes.   
           And hooptiously drangle me with crinkly bindlewurdles,
Or I will rend thee in the gobberwarts with my blurlecruncheon, see if I don't.

                (by Prostetnic Vogon Jeltz; see p. 56/57)
)V0G0N";

字符串中的所有空格、缩进和换行符都会被保留。

这些也可以是 utf-8|16|32 或 wchar_t(带有常用前缀)。

我应该指出,这里实际上并不需要转义序列 V0G0N。 它的存在将允许将 )" 放入字符串中。换句话说,我可以放入

                "(by Prostetnic Vogon Jeltz; see p. 56/57)"

(注意额外的引号)并且上面的字符串仍然是正确的。否则我也可以使用

const char * vogon_poem = R"( ... )";

引号内的括号仍然需要。

In C++11 you have raw string literals. Sort of like here-text in shells and script languages like Python and Perl and Ruby.

const char * vogon_poem = R"V0G0N(
             O freddled gruntbuggly thy micturations are to me
                 As plured gabbleblochits on a lurgid bee.
              Groop, I implore thee my foonting turlingdromes.   
           And hooptiously drangle me with crinkly bindlewurdles,
Or I will rend thee in the gobberwarts with my blurlecruncheon, see if I don't.

                (by Prostetnic Vogon Jeltz; see p. 56/57)
)V0G0N";

All the spaces and indentation and the newlines in the string are preserved.

These can also be utf-8|16|32 or wchar_t (with the usual prefixes).

I should point out that the escape sequence, V0G0N, is not actually needed here. Its presence would allow putting )" inside the string. In other words, I could have put

                "(by Prostetnic Vogon Jeltz; see p. 56/57)"

(note extra quotes) and the string above would still be correct. Otherwise I could just as well have used

const char * vogon_poem = R"( ... )";

The parens just inside the quotes are still needed.

清风挽心 2024-08-05 02:00:42

您还可以这样做:

const char *longString = R""""(
This is 
a very 
long 
string
)"""";

You can also do this:

const char *longString = R""""(
This is 
a very 
long 
string
)"""";
宛菡 2024-08-05 02:00:42

#define MULTILINE(...) #__VA_ARGS__
消耗括号之间的所有内容。
将任意数量的连续空白字符替换为单个空格。

#define MULTILINE(...) #__VA_ARGS__
Consumes everything between the parentheses.
Replaces any number of consecutive whitespace characters by a single space.

剪不断理还乱 2024-08-05 02:00:42

输入多行字符串的一种可能方便的方法是使用宏。 仅当引号和括号平衡并且不包含“顶级”逗号时,这才有效:

#define MULTI_LINE_STRING(a) #a
const char *text = MULTI_LINE_STRING(
  Using this trick(,) you don't need to use quotes.
  Though newlines and     multiple     white   spaces
  will be replaced by a single whitespace.
);
printf("[[%s]]\n",text);

使用 gcc 4.6 或 g++ 4.6 编译,这会产生: [[使用此技巧(,),您不需要使用引号。 尽管换行符和多个空格将被单个空格替换。]]

请注意,, 不能出现在字符串中,除非它包含在括号或引号内。 单引号是可能的,但会产生编译器警告。

编辑:如评论中所述,#define MULTI_LINE_STRING(...) #__VA_ARGS__ 允许使用 ,

A probably convenient way to enter multi-line strings is by using macro's. This only works if quotes and parentheses are balanced and it does not contain 'top level' comma's:

#define MULTI_LINE_STRING(a) #a
const char *text = MULTI_LINE_STRING(
  Using this trick(,) you don't need to use quotes.
  Though newlines and     multiple     white   spaces
  will be replaced by a single whitespace.
);
printf("[[%s]]\n",text);

Compiled with gcc 4.6 or g++ 4.6, this produces: [[Using this trick(,) you don't need to use quotes. Though newlines and multiple white spaces will be replaced by a single whitespace.]]

Note that the , cannot be in the string, unless it is contained within parenthesis or quotes. Single quotes is possible, but creates compiler warnings.

Edit: As mentioned in the comments, #define MULTI_LINE_STRING(...) #__VA_ARGS__ allows the use of ,.

独享拥抱 2024-08-05 02:00:42

你可以这样做:

const char *text = "This is my string it is "
     "very long";

You can just do this:

const char *text = "This is my string it is "
     "very long";
别低头,皇冠会掉 2024-08-05 02:00:42

只是为了澄清一下 @unwind 的答案中 @emsr 的评论,如果一个人没有足够的运气拥有 C++11 编译器(例如 GCC 4.2.1),并且想要在字符串中嵌入换行符(char *或类字符串),可以写这样的东西:

const char *text =
  "This text is pretty long, but will be\n"
  "concatenated into just a single string.\n"
  "The disadvantage is that you have to quote\n"
  "each part, and newlines must be literal as\n"
  "usual.";

非常明显,正确,但是当我第一次读到这篇文章时,@emsr 的简短评论并没有引起我的注意,所以我必须自己发现这一点。 希望我已经为其他人节省了几分钟。

Just to elucidate a bit on @emsr's comment in @unwind's answer, if one is not fortunate enough to have a C++11 compiler (say GCC 4.2.1), and one wants to embed the newlines in the string (either char * or class string), one can write something like this:

const char *text =
  "This text is pretty long, but will be\n"
  "concatenated into just a single string.\n"
  "The disadvantage is that you have to quote\n"
  "each part, and newlines must be literal as\n"
  "usual.";

Very obvious, true, but @emsr's short comment didn't jump out at me when I read this the first time, so I had to discover this for myself. Hopefully, I've saved someone else a few minutes.

难以启齿的温柔 2024-08-05 02:00:42

由于一盎司的经验胜过大量的理论,因此我尝试了一个 MULTILINE 的小测试程序:

#define MULTILINE(...) #__VA_ARGS__

const char *mstr[] =
{
    MULTILINE(1, 2, 3),       // "1, 2, 3"
    MULTILINE(1,2,3),         // "1,2,3"
    MULTILINE(1 , 2 , 3),     // "1 , 2 , 3"
    MULTILINE( 1 , 2 , 3 ),   // "1 , 2 , 3"
    MULTILINE((1,  2,  3)),   // "(1,  2,  3)"
    MULTILINE(1
              2
              3),             // "1 2 3"
    MULTILINE(1\n2\n3\n),     // "1\n2\n3\n"
    MULTILINE(1\n
              2\n
              3\n),           // "1\n 2\n 3\n"
    MULTILINE(1, "2" \3)      // "1, \"2\" \3"
};

使用 cpp -P -std=c++11 filename 编译此片段重现。

#__VA_ARGS__ 背后的技巧是 __VA_ARGS__ 不处理逗号分隔符。 因此您可以将其传递给字符串化运算符。 前导和尾随空格被修剪,然后单词之间的空格(包括换行符)被压缩为单个空格。 括号需要平衡。 我认为这些缺点解释了为什么 C++11 的设计者尽管有 #__VA_ARGS__ 仍然看到了对原始字符串文字的需求。

Since an ounce of experience is worth a ton of theory, I tried a little test program for MULTILINE:

#define MULTILINE(...) #__VA_ARGS__

const char *mstr[] =
{
    MULTILINE(1, 2, 3),       // "1, 2, 3"
    MULTILINE(1,2,3),         // "1,2,3"
    MULTILINE(1 , 2 , 3),     // "1 , 2 , 3"
    MULTILINE( 1 , 2 , 3 ),   // "1 , 2 , 3"
    MULTILINE((1,  2,  3)),   // "(1,  2,  3)"
    MULTILINE(1
              2
              3),             // "1 2 3"
    MULTILINE(1\n2\n3\n),     // "1\n2\n3\n"
    MULTILINE(1\n
              2\n
              3\n),           // "1\n 2\n 3\n"
    MULTILINE(1, "2" \3)      // "1, \"2\" \3"
};

Compile this fragment with cpp -P -std=c++11 filename to reproduce.

The trick behind #__VA_ARGS__ is that __VA_ARGS__ does not process the comma separator. So you can pass it to the stringizing operator. Leading and trailing spaces are trimmed, and spaces (including newlines) between words are compressed to a single space then. Parentheses need to be balanced. I think these shortcomings explain why the designers of C++11, despite #__VA_ARGS__, saw the need for raw string literals.

稳稳的幸福 2024-08-05 02:00:42
// C++11. 
std::string index_html=R"html(
<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8">
    <title>VIPSDK MONITOR</title>
    <meta http-equiv="refresh" content="10">
  </head>
  <style type="text/css">
  </style>
</html>
)html";
// C++11. 
std::string index_html=R"html(
<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8">
    <title>VIPSDK MONITOR</title>
    <meta http-equiv="refresh" content="10">
  </head>
  <style type="text/css">
  </style>
</html>
)html";
谜兔 2024-08-05 02:00:42
char* multiline_string = "line1\n"
"line2\n";
char* multiline_string = "line1\n"
"line2\n";
川水往事 2024-08-05 02:00:42

选项 1. 使用 Boost 库,您可以如下声明字符串:

const boost::string_view helpText = "This is very long help text.\n"
      "Also more text is here\n"
      "And here\n"

// Pass help text here
setHelpText(helpText);

选项 2. 如果您的项目中没有 Boost,您可以在现代 C++ 中使用 std::string_view()

Option 1. Using Boost library, you can declare the string as below:

const boost::string_view helpText = "This is very long help text.\n"
      "Also more text is here\n"
      "And here\n"

// Pass help text here
setHelpText(helpText);

Option 2. If Boost is not available in your project, you can use std::string_view() in modern C++.

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文