条件默认构造函数在 GCC 中编译但在 MSVC 中不编译
Conditional Default Constructor compiles in GCC but not MSVC
我正在创建一个模板 class,其中有以下限制:
- 如果class的
type
是generic
,它应该有正常的构造函数
- 如果 class 的
type
不是 generic
,它应该有明确的构造函数。
我想出的解决方案是这样的:
#include<type_traits>
#include<vector>
enum class point_type {
generic, center, corner
};
template<point_type type, typename T>
struct point2_base {
T x, y;
template<point_type _type = type, typename std::enable_if_t<_type == point_type::generic, int> = 0>
point2_base() :
x(0), y(0) {
}
template<point_type _type = type, typename std::enable_if_t<_type != point_type::generic, int> = 0>
explicit point2_base() :
x(0), y(0) {
}
};
using point2f = point2_base<point_type::generic, float>;
using corner2f = point2_base<point_type::corner, float>;
using center2f = point2_base<point_type::center, float>;
这似乎工作得很好(我可以用 point2f point;
或 corner2f corner;
等实例化类型),但是我写的一些深奥的代码无法在 MSVC 中编译:
struct line {
point2f baseline;
std::vector<int> characters; //The type of the vector doesn't seem to matter
};
int main() {
std::vector<line> lines;
lines.emplace_back();
}
Godbolt reports these errors 在 MSVC 中:
example.cpp
/opt/compiler-explorer/windows/19.10.25017/lib/native/include/vector(927): warning C4530: C++ exception handler used, but unwind semantics are not enabled. Specify /EHsc
33 : <source>(33): note: see reference to function template instantiation 'void std::vector<line,std::allocator<_Ty>>::emplace_back<>(void)' being compiled
with
[
_Ty=line
]
33 : <source>(33): note: see reference to function template instantiation 'void std::vector<line,std::allocator<_Ty>>::emplace_back<>(void)' being compiled
with
[
_Ty=line
]
/opt/compiler-explorer/windows/19.10.25017/lib/native/include/xmemory0(840): error C2783: 'point2_base<point_type::corner,float>::point2_base(void)': could not deduce template argument for '__formal'
13 : <source>(13): note: see declaration of 'point2_base<point_type::corner,float>::point2_base'
/opt/compiler-explorer/windows/19.10.25017/lib/native/include/xmemory0(959): note: see reference to function template instantiation 'void std::allocator<_Ty>::construct<_Objty,>(_Objty *)' being compiled
with
[
_Ty=line,
_Objty=line
]
/opt/compiler-explorer/windows/19.10.25017/lib/native/include/xmemory0(959): note: see reference to function template instantiation 'void std::allocator<_Ty>::construct<_Objty,>(_Objty *)' being compiled
with
[
_Ty=line,
_Objty=line
]
/opt/compiler-explorer/windows/19.10.25017/lib/native/include/xmemory0(1097): note: see reference to function template instantiation 'void std::allocator_traits<_Alloc>::construct<_Ty,>(std::allocator<_Ty> &,_Objty *)' being compiled
with
[
_Alloc=std::allocator<line>,
_Ty=line,
_Objty=line
]
/opt/compiler-explorer/windows/19.10.25017/lib/native/include/xmemory0(1096): note: see reference to function template instantiation 'void std::allocator_traits<_Alloc>::construct<_Ty,>(std::allocator<_Ty> &,_Objty *)' being compiled
with
[
_Alloc=std::allocator<line>,
_Ty=line,
_Objty=line
]
/opt/compiler-explorer/windows/19.10.25017/lib/native/include/vector(928): note: see reference to function template instantiation 'void std::_Wrap_alloc<std::allocator<_Ty>>::construct<_Ty,>(_Ty *)' being compiled
with
[
_Ty=line
]
/opt/compiler-explorer/windows/19.10.25017/lib/native/include/vector(928): note: see reference to function template instantiation 'void std::_Wrap_alloc<std::allocator<_Ty>>::construct<_Ty,>(_Ty *)' being compiled
with
[
_Ty=line
]
33 : <source>(33): note: see reference to function template instantiation 'void std::vector<line,std::allocator<_Ty>>::emplace_back<>(void)' being compiled
with
[
_Ty=line
]
33 : <source>(33): note: see reference to function template instantiation 'void std::vector<line,std::allocator<_Ty>>::emplace_back<>(void)' being compiled
with
[
_Ty=line
]
Microsoft (R) C/C++ Optimizing Compiler Version 19.10.25017 for x64
Copyright (C) Microsoft Corporation. All rights reserved.
Compiler exited with result code 2
奇怪的是 GCC 编译这段代码没有问题。
我还发现有一些允许 MSVC 编译的东西:
struct line {
point2f baseline;
//If I remove the vector, the code compiles
};
//////////
struct line {
point2f baseline;
std::vector<int> characters; //The type of the vector doesn't seem to matter
line() {} //Adding a trivial, explicitly declared constructor allows the code to compile
};
以下代码无法在 MSVC 中编译:
struct line {
point2f baseline;
std::vector<int> characters; //The type of the vector doesn't seem to matter
line() = default; //Same errors as before.
};
这是我为 point2_base
class 编写构造函数的方式的缺陷吗?我应该使用其他机制来有条件地设置这些构造函数 explicit
吗?这只是 MSVC 中的一个错误吗?
这似乎是 Visual Studio 编译器 which is fixed as of version 19.14 中的错误。我不清楚哪个版本的 Visual Studio 修复了这个错误。
我正在创建一个模板 class,其中有以下限制:
- 如果class的
type
是generic
,它应该有正常的构造函数 - 如果 class 的
type
不是generic
,它应该有明确的构造函数。
我想出的解决方案是这样的:
#include<type_traits>
#include<vector>
enum class point_type {
generic, center, corner
};
template<point_type type, typename T>
struct point2_base {
T x, y;
template<point_type _type = type, typename std::enable_if_t<_type == point_type::generic, int> = 0>
point2_base() :
x(0), y(0) {
}
template<point_type _type = type, typename std::enable_if_t<_type != point_type::generic, int> = 0>
explicit point2_base() :
x(0), y(0) {
}
};
using point2f = point2_base<point_type::generic, float>;
using corner2f = point2_base<point_type::corner, float>;
using center2f = point2_base<point_type::center, float>;
这似乎工作得很好(我可以用 point2f point;
或 corner2f corner;
等实例化类型),但是我写的一些深奥的代码无法在 MSVC 中编译:
struct line {
point2f baseline;
std::vector<int> characters; //The type of the vector doesn't seem to matter
};
int main() {
std::vector<line> lines;
lines.emplace_back();
}
Godbolt reports these errors 在 MSVC 中:
example.cpp
/opt/compiler-explorer/windows/19.10.25017/lib/native/include/vector(927): warning C4530: C++ exception handler used, but unwind semantics are not enabled. Specify /EHsc
33 : <source>(33): note: see reference to function template instantiation 'void std::vector<line,std::allocator<_Ty>>::emplace_back<>(void)' being compiled
with
[
_Ty=line
]
33 : <source>(33): note: see reference to function template instantiation 'void std::vector<line,std::allocator<_Ty>>::emplace_back<>(void)' being compiled
with
[
_Ty=line
]
/opt/compiler-explorer/windows/19.10.25017/lib/native/include/xmemory0(840): error C2783: 'point2_base<point_type::corner,float>::point2_base(void)': could not deduce template argument for '__formal'
13 : <source>(13): note: see declaration of 'point2_base<point_type::corner,float>::point2_base'
/opt/compiler-explorer/windows/19.10.25017/lib/native/include/xmemory0(959): note: see reference to function template instantiation 'void std::allocator<_Ty>::construct<_Objty,>(_Objty *)' being compiled
with
[
_Ty=line,
_Objty=line
]
/opt/compiler-explorer/windows/19.10.25017/lib/native/include/xmemory0(959): note: see reference to function template instantiation 'void std::allocator<_Ty>::construct<_Objty,>(_Objty *)' being compiled
with
[
_Ty=line,
_Objty=line
]
/opt/compiler-explorer/windows/19.10.25017/lib/native/include/xmemory0(1097): note: see reference to function template instantiation 'void std::allocator_traits<_Alloc>::construct<_Ty,>(std::allocator<_Ty> &,_Objty *)' being compiled
with
[
_Alloc=std::allocator<line>,
_Ty=line,
_Objty=line
]
/opt/compiler-explorer/windows/19.10.25017/lib/native/include/xmemory0(1096): note: see reference to function template instantiation 'void std::allocator_traits<_Alloc>::construct<_Ty,>(std::allocator<_Ty> &,_Objty *)' being compiled
with
[
_Alloc=std::allocator<line>,
_Ty=line,
_Objty=line
]
/opt/compiler-explorer/windows/19.10.25017/lib/native/include/vector(928): note: see reference to function template instantiation 'void std::_Wrap_alloc<std::allocator<_Ty>>::construct<_Ty,>(_Ty *)' being compiled
with
[
_Ty=line
]
/opt/compiler-explorer/windows/19.10.25017/lib/native/include/vector(928): note: see reference to function template instantiation 'void std::_Wrap_alloc<std::allocator<_Ty>>::construct<_Ty,>(_Ty *)' being compiled
with
[
_Ty=line
]
33 : <source>(33): note: see reference to function template instantiation 'void std::vector<line,std::allocator<_Ty>>::emplace_back<>(void)' being compiled
with
[
_Ty=line
]
33 : <source>(33): note: see reference to function template instantiation 'void std::vector<line,std::allocator<_Ty>>::emplace_back<>(void)' being compiled
with
[
_Ty=line
]
Microsoft (R) C/C++ Optimizing Compiler Version 19.10.25017 for x64
Copyright (C) Microsoft Corporation. All rights reserved.
Compiler exited with result code 2
奇怪的是 GCC 编译这段代码没有问题。
我还发现有一些允许 MSVC 编译的东西:
struct line {
point2f baseline;
//If I remove the vector, the code compiles
};
//////////
struct line {
point2f baseline;
std::vector<int> characters; //The type of the vector doesn't seem to matter
line() {} //Adding a trivial, explicitly declared constructor allows the code to compile
};
以下代码无法在 MSVC 中编译:
struct line {
point2f baseline;
std::vector<int> characters; //The type of the vector doesn't seem to matter
line() = default; //Same errors as before.
};
这是我为 point2_base
class 编写构造函数的方式的缺陷吗?我应该使用其他机制来有条件地设置这些构造函数 explicit
吗?这只是 MSVC 中的一个错误吗?
这似乎是 Visual Studio 编译器 which is fixed as of version 19.14 中的错误。我不清楚哪个版本的 Visual Studio 修复了这个错误。