统一初始化中的尾随逗号

时间:2017-09-06 08:54:25

标签: c++ c++14 language-lawyer c++17 uniform-initialization

在统一初始化期间使用尾随逗号时是否存在任何潜在的语义差异?

std::vector< std::size_t > v1{5, }; // allowed syntax
std::vector< std::size_t > v2{10};

我可以使用尾随逗号使编译器选择std::vector::vector(std::initializer_list< std::size_t >)构造函数而不是std::vector::vector(std::size_t, const std::size_t &),还是有其他提及语法的技巧?

我可以用它来检测是否有std::initializer_list - 构造函数重载?

考虑以下代码,必须选择哪个构造函数?

struct A { A(int) { ; } A(double, int = 3) { ; } };
A a{1};
A b{2, };

在这两种情况下都会选择此代码is accepted by gcc 8A(int)

2 个答案:

答案 0 :(得分:12)

首先,C ++语法规则使 braced-init-list 的尾随,可选。引用dcl.init/1

  

声明者可以指定标识符的初始值   声明。标识符指定要初始化的变量。该   [dcl.init]其余部分描述的初始化过程   也适用于其他语法上下文指定的初始化,   比如初始化函数参数([expr.call])或者   初始化返回值([stmt.return])。

initializer:
  brace-or-equal-initializer
  ( expression-list )
brace-or-equal-initializer:
  = initializer-clause
  braced-init-list
initializer-clause:
  assignment-expression
  braced-init-list
braced-init-list:
  { initializer-list ,opt }
  { designated-initializer-list ,opt }
  { }

其次,你几乎不能覆盖重载决策系统。如果您使用此类语法并且此类std::initializer_list构造函数可用,它将始终使用std::initializer_list构造函数。

dcl.init.list/2

  

构造函数是初始化列表构造函数(如果是第一个)   参数的类型为std :: initializer_list或引用   对于某些类型E,可能是cv-qualified std :: initializer_list   要么没有其他参数,要么所有其他参数都有   默认参数。    [注意:初始化列表构造函数优于列表初始化中的其他构造函数([over.match.list])....

以下程序打印Using InitList

#include <iostream>
#include <initializer_list>

struct X{
    X(std::initializer_list<double>){ std::cout << "Using InitList\n"; }
    X(int){ std::cout << "Using Single Arg ctor\n"; }
};

int main(){
    X x{5};
}

尽管5int类型的文字,但选择单个参数构造函数应该是有意义的,因为它是完美的匹配;并且std::initializer_list<double>构造函数需要double的列表。但是,规则支持std::initializer_list<double>,因为它是初始化列表构造函数

因此,即使下面的程序因转换范围缩小而失败:

#include <iostream>
#include <initializer_list>

struct Y{
    Y(std::initializer_list<char>){ std::cout << "Y Using InitList\n"; }
    Y(int, int=4){ std::cout << "Y Using Double Arg ctor\n"; }
};

int main(){
    Y y1{4777};
    Y y2{577,};
    Y y3{57,7777};
}

在回复下面的评论时,“如果没有使用std :: initializer_list进行重载,或者它不是第一个构造函数的参数?” - 那么重载决策不会选择​​它。演示:

#include <iostream>
#include <initializer_list>

struct Y{
    Y(int, std::initializer_list<double>){ std::cout << "Y Using InitList\n"; }
    Y(int, int=4){ std::cout << "Y Using Double Arg ctor\n"; }
};

int main(){
    Y y1{4};
    Y y2{5,};
    Y y3{5,7};
}

打印:

Y Using Double Arg ctor
Y Using Double Arg ctor
Y Using Double Arg ctor

如果没有初始化列表构造函数可用,那么{initializer-list...,}初始化程序几乎可以根据dcl.init/16回退到直接初始化,其语义由dcl.init/16

的前一段所涵盖

答案 1 :(得分:2)

没有。该逗号是使预处理器宏技巧在没有编译错误的情况下工作的让步。它对您的数据类型或其大小没有任何意义。