【发布时间】:2026-01-20 11:15:01
【问题描述】:
这是有效的,因为constexpr 表达式允许采用“一个字面量类型的左值,它引用一个用 constexpr 定义的非易失性对象,或者引用这样一个对象的子对象"(§5.19/2):
constexpr char str[] = "hello, world";
constexpr char e = str[1];
但是,字符串文字似乎不符合此描述:
constexpr char e = "hello, world"[1]; // error: literal is not constexpr
2.14.5/8 描述了字符串字面量的类型:
普通字符串文字和 UTF-8 字符串文字也称为窄字符串文字。窄字符串文字的类型为“n const char 数组”,其中 n 是字符串的大小,定义如下,并且具有静态存储持续时间。
似乎这种类型的对象可以被索引,只要它是临时的而不是静态存储持续时间(5.19/2,就在上面的 sn-p 之后):
[
constexpr允许左值到右值转换] ... 一个字面量类型的左值,它引用一个非易失性临时对象,其生命周期尚未结束,用常量表达式初始化
这特别奇怪,因为获取临时对象的左值通常是“作弊”。我想这条规则适用于引用类型的函数参数,例如在
constexpr char get_1( char const (&str)[ 6 ] )
{ return str[ 1 ]; }
constexpr char i = get_1( { 'y', 'i', 'k', 'e', 's', '\0' } ); // OK
constexpr char e = get_1( "hello" ); // error: string literal not temporary
不管怎样,GCC 4.7 接受 get_1( "hello" ),但拒绝 "hello"[1],因为“'._0' 的值在常量表达式中不可用”......然而"hello"[1] 是可接受作为案例标签或数组绑定。
我在这里分裂了一些标准的头发......分析是否正确,这个功能是否有一些设计意图?
编辑:哦……这有一些动机。似乎这种表达式是在预处理器中使用查找表的唯一方法。例如,这会引入一个代码块,除非SOME_INTEGER_FLAG 为 1 或 5,否则该代码块将被忽略,如果大于 6,则会导致诊断:
#if "\0\1\0\0\0\1"[ SOME_INTEGER_FLAG ]
这个结构对于 C++11 来说是新的。
【问题讨论】:
标签: c++ c++11 string-literals constexpr