我有clang ++ 4.2
Apple LLVM version 4.2 (clang-425.0.28) (based on LLVM 3.2svn)
Target: x86_64-apple-darwin11.4.2
Thread model: posix
当我尝试编译这个c ++ 11代码时:
class ContextSummary {
int id;
int hops;
std::map<std::string, int> db {};
std::time_t timestamp;
ContextSummary(int id, const std::map<std::string, int>& db = {}, int hops = 3, std::time_t timestamp = 0)
{
this->id = id;
this->db = db;
this->hops = hops;
this->timestamp = timestamp;
}
我收到此错误消息。该代码适用于g ++ 4.8
error:
chosen constructor is explicit in copy-initialization
...id, const std::map<std::string, int>& db = {}, int hops = 3, std::time_t...
^ ~~
这是clang ++ bug吗?如何绕过此错误?
答案 0 :(得分:8)
复制我在评论中所说的内容
这是http://cplusplus.github.io/LWG/lwg-active.html#2193。我不确定“提议的解决方案”或类似的东西是否会成为C ++ 14。在复制初始化上下文中使用显式默认构造函数进行值初始化是否良好的事实本身也是核心语言DR http://open-std.org/jtc1/sc22/wg21/docs/cwg_active.html#1518,这可能解释了交叉编译器之间的差异。
如果您的实现实现http://www.open-std.org/jtc1/sc22/wg21/docs/cwg_defects.html#1494,并且以下是默认参数的有效语法,您可以完成
const std::map<std::string, int>& db{}
不幸的是,这是不允许的(我认为我明确地传递一个参数的原因是,你也不能直接初始化参数,所以为什么允许它用于默认参数?)。因此,在我看来,唯一的方法是明确创建它
const std::map<std::string, int>& db = std::map<std::string, int>{}
决定是否要以可能更多的代码为代价来逃避冗余。一些替代方案
const std::map<std::string, int>& db = decltype(ContextSummary::db){}
const std::map<std::string, int>& db_ = decltype(db){}
const std::map<std::string, int>& db = std::decay<decltype(db)>::type{}