下面的代码在 Visual Studio 2013、gcc 4.8、clang 3.4 和 clang 3.5(Apple LLVM 6.0)中编译,但不能在 clang 3.6 中编译(通过 Apple LLVM 6.1)
该代码是我们代码库中一个复杂类的简化版本,这是展示问题的最低要求。
问题的症结在于TYPED_VALUE
,在 3.6 中,STRING
由于存在接受STRING
; 这会导致std::is_constructible
被评估,这导致它需要定义STRING
(我们无法在此处提供 - 会导致完整代码中的循环依赖)。
class STRING;
class TYPED_VALUE
{
public:
TYPED_VALUE( const TYPED_VALUE& ) = default; // explicit or implicit doesn't make a difference
TYPED_VALUE( const STRING & ) {}
template< typename TYPE, typename std::enable_if<!std::is_pointer< TYPE >::value && !std::is_constructible< TYPE, const STRING& >::value && !std::is_constructible< TYPE, bool >::value, int >::type = 0 >
operator TYPE( void ) const = delete;
};
class TYPED_STORAGE
{
public:
TYPED_STORAGE( const TYPED_VALUE &v ) : value( v ) {}
TYPED_VALUE value;
};
错误信息是
/type_traits:2329:38: error: incomplete type 'SICORE::STRING' used in type trait expression
: public integral_constant<bool, __is_constructible(_Tp, _Args...)>
^
/main.cpp:348:99: note: in instantiation of template class 'std::__1::is_constructible<SICORE::STRING, const SICORE::STRING &>' requested here
template< typename TYPE, typename std::enable_if<!std::is_pointer< TYPE >::value && !std::is_constructible< TYPE, const STRING& >::value && !std::is_constructible< TYPE, bool >::value, int >::type = 0 >
^
/main.cpp:349:9: note: while substituting prior template arguments into non-type template parameter [with TYPE = SICORE::STRING]
operator TYPE( void ) const = delete;
^~~~~~~~~~~~~~~~~~~~~~~~~~~
/main.cpp:355:56: note: while substituting deduced template arguments into function template 'operator type-parameter-0-0' [with TYPE = SICORE::STRING, $1 = (no value)]
TYPED_STORAGE( const TYPED_VALUE &v ) : value( v ) {}
^
/main.cpp:340:11: note: forward declaration of 'SICORE::STRING'
class STRING;
^
对我来说,这似乎是 3.6 中的一个错误,在以前的版本中,重载决议确定复制构造函数是最合适的,而无需评估模板参数 - 我试图理解标准中的重载决议注释,但我认为这只是混淆我更多;)
(这可以通过明确构造构造函数或转换运算符来解决,但这不是我们想要的行为)
那里有任何标准专家知道答案吗?