【发布时间】:2021-01-01 14:50:34
【问题描述】:
一些 C++ hack 使用转换运算符来获取有关构造函数的一些信息。
我想知道,在模板化强制转换运算符的解析中为T 选择具体类型的过程是什么。
#include <iostream>
#include <type_traits>
using std::cout;
using std::endl;
struct A {
A(int) { cout << "int" << endl; }
A() { cout << "def" << endl; }
A(const A&) { cout << "copy" << endl; }
A(A&&) { cout << "move" << endl; }
};
struct B {
template<typename T> operator T()
{ return {}; }
};
template<typename Except>
struct C {
template<typename T,
std::enable_if_t<!std::is_same_v<T, Except>>* = nullptr> operator T()
{ return {}; }
};
template<typename T>
void f(A a = { T() }) {}
int main() {
f<B>();
f<C<A>>();
return 0;
}
这段代码打印这个:
def
int
不是这个:
int
int
为什么要禁用转换以获取我想要的构造函数(int 版本)? C++ 标准说返回类型不参与寻找有效的模板重载,那么为什么它选择这个版本而不抱怨多种可能的解决方案呢?
生成文件:
EXE = C++Tuple
CXX = g++
CXXFLAGS = -std=c++17
run: $(EXE)
./$(EXE)
.PHONY: run
$(EXE): main.cpp
$(CXX) $(CXXFLAGS) -o $@ $<
平台:
$ gcc -v
Using built-in specs.
COLLECT_GCC=gcc
COLLECT_LTO_WRAPPER=/usr/lib/gcc/x86_64-linux-gnu/7/lto-wrapper
OFFLOAD_TARGET_NAMES=nvptx-none
OFFLOAD_TARGET_DEFAULT=1
Target: x86_64-linux-gnu
Configured with: ../src/configure -v --with-pkgversion='Ubuntu 7.5.0-3ubuntu1~18.04' --with-bugurl=file:///usr/share/doc/gcc-7/README.Bugs --enable-languages=c,ada,c++,go,brig,d,fortran,objc,obj-c++ --prefix=/usr --with-gcc-major-version-only --program-suffix=-7 --program-prefix=x86_64-linux-gnu- --enable-shared --enable-linker-build-id --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --libdir=/usr/lib --enable-nls --enable-bootstrap --enable-clocale=gnu --enable-libstdcxx-debug --enable-libstdcxx-time=yes --with-default-libstdcxx-abi=new --enable-gnu-unique-object --disable-vtable-verify --enable-libmpx --enable-plugin --enable-default-pie --with-system-zlib --with-target-system-zlib --enable-objc-gc=auto --enable-multiarch --disable-werror --with-arch-32=i686 --with-abi=m64 --with-multilib-list=m32,m64,mx32 --enable-multilib --with-tune=generic --enable-offload-targets=nvptx-none --without-cuda-driver --enable-checking=release --build=x86_64-linux-gnu --host=x86_64-linux-gnu --target=x86_64-linux-gnu
Thread model: posix
gcc version 7.5.0 (Ubuntu 7.5.0-3ubuntu1~18.04)
$ uname -r
4.4.0-19041-Microsoft
【问题讨论】:
-
有趣的是,GCC 和 MSVC 拒绝编译这个(你似乎使用 Clang)。
-
它是在 WSL 上使用 gcc 在 C++17 上编译的。
gcc -v给gcc version 7.5.0。问题是当在这种特定情况下有多个候选者时,为强制转换运算符找到有效重载时的全局优先级是什么,那么为什么在第一种情况下调用 A() 而不是 A(int) -
我添加了Makefile以获取详细信息,但验证后似乎仍然可以编译...