- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
作为对我的 last question 的回答建议尽可能使用 std::common_type<X,Y>::type
在自动返回类型的声明中而不是我原来的 decltype()
.但是,这样做我遇到了问题(使用 gcc 4.7.0)。考虑以下简单代码
template<typename> class A;
template<typename X> class A {
X a[3];
template <typename> friend class A;
public:
A(X a0, X a1, X a2) { a[0]=a0; a[1]=a1; a[2]=a2; }
X operator[](int i) const { return a[i]; }
X operator*(A const&y) const // multiplication 0: dot product with self
{ return a[0]*y[0] + a[1]*y[1] + a[2]*y[2]; }
template<typename Y>
auto operator*(A<Y> const&y) const -> // multiplication 1: dot product with A<Y>
#ifdef USE_DECLTYPE
decltype((*this)[0]*y[0])
#else
typename std::common_type<X,Y>::type
#endif
{ return a[0]*y[0] + a[1]*y[1] + a[2]*y[2]; }
template<typename Y>
auto operator*(Y s) const -> // multiplication 2: with scalar
#ifdef USE_DECLTYPE
A<decltype((*this)[0]*s)>
#else
A<typename std::common_type<X,Y>::type>
#endif
{ return A<decltype((*this)[0]*s)>(s*a[0],s*a[1],s*a[2]); }
};
int main()
{
A<double> x(1.2,2.0,-0.4), y(0.2,4.4,5.0);
A<double> z = x*4;
auto dot = x*y; // <--
std::cout<<" x*4="<<z[0]<<' '<<z[1]<<' '<<z[2]<<'\n'
<<" x*y="<<dot<<'\n';
}
当USE_DECLTYPE
是#defined
, 代码在 gcc 4.7.0 下编译和运行良好。但除此之外,main()
中指示的行调用 multiplaction 2,如果没有错误的话,这看起来很奇怪。这可能是使用 std::common_type
的结果/副作用吗?还是 gcc 的错误?
我一直认为返回类型与选择众多合适的模板函数中的哪一个无关......
最佳答案
建议使用common_type
是假的。
使用decltype
的问题你在你的另一个问题中只是一个GCC bug .
你这个问题在使用common_type
时遇到的问题是因为std::common_type<X, Y>::type
告诉您将从表达式中获得的类型:
condition ? std::declval<X>() : std::declval<Y>()
即什么类型 X
和一个 Y
都可以转化为。
一般来说,这与 x * y
的结果完全无关。 , 如果 X
和 Y
重载了 operator*
返回一个完全不同的类型。
在您的特定情况下,您有表达式 x*y
其中两个变量的类型都是 A<double>
.重载解析尝试检查每个重载 operator*
看看它是否有效。作为重载决议的一部分,它实例化了这个成员函数模板:
template<typename Y>
auto operator*(Y s) const ->
A<typename std::common_type<X,Y>::type>;
与 A<double>
替换模板参数 Y
.尝试实例化 common_type<double, A<double>>
这是无效的,因为表达式
condition ? std::declval<double>() : std::declval< A<double> >()
无效,因为您无法转换 A<double>
至 double
反之亦然,或任何其他常见类型。
错误不会发生,因为它重载了 operator*
被调用,这是因为必须实例化模板才能决定应调用哪个运算符,而实例化它的行为会导致错误。编译器永远不会决定调用哪个运算符,错误会在它到达那一步之前停止它。
所以,正如我所说,建议使用 common_type
是假的,它阻止 SFINAE 禁用与参数类型不匹配的成员函数模板(形式上,SFINAE 在这里不起作用,因为替换错误发生在模板的“直接上下文”之外,即它发生在定义内部common_type
不在函数签名中,SFINAE 适用。)
允许专门化std::common_type
所以它知道没有隐式转换的类型,所以你可以专门化它以便 common_type<double, A<double>>::type
有效并生成类型 double
,像这样:
namespace std {
template<typename T>
struct common_type<T, A<T>> { typedef T type; };
}
这样做是一个非常糟糕的主意! 什么 common_type
应该给出“这两种类型都可以安全地转换成什么类型?”的答案。上面的特化颠覆了它,给出了“这些类型相乘的结果是什么?”的答案。这是一个完全不同的问题!这与特化一样愚蠢 is_integral<std::string>
是真的。
如果您想要回答“expr 之类的一般表达式的类型是什么?”然后使用 decltype(expr)
, 这就是它的用途!
关于c++ - 什么更适合用于自动返回类型:decltype 或 std::common_type<>::type(如果可能)?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11055350/
我正在制作一个简单的程序来更改我的计算机背景。我在网上发现了一个stackoverflow问题,或多或少涵盖了我想做的事情。我现在可以成功地将我的墙纸更改为平铺、居中和从在线图像 URL 拉伸(str
是的,这是另一个每组最大的问题之一!我已经尝试了几天,试图解决这个问题,但无济于事。我也一直在寻找,但我什至不知道我是否在正确的地方寻找。问题的最简化版本如下。 我有 2 个表,一个是多对多表,另一个
我想解析一些数据,我有一个 BNF 语法来解析它。谁能推荐任何能够生成可在移动设备上使用的代码的语法编译器? 由于这是针对 JavaME 的,因此生成的代码必须是: 希望很小 对外来 Java 库的依
我有一个动物园时间序列对象,vels : 2011-05-01 00:00:00 7.52 2011-05-01 00:10:00 7.69 2011-05-01 00:20:00 7.67 2011
我想创建一个供小型制造公司使用的生产管理系统。该系统将允许记录设备制造的不同阶段。要求如下: 1.非基于浏览器的界面。需要基于 Swing 或 AWT 的东西。虽然我了解实现基于浏览器的解决方案的便利
是否有任何 java 或 clojure 邮件库可以实现 lamson 的功能?特别是lamson的邮件路由功能非常酷http://verpa.wordpress.com/2010/11/13/mak
sklearn 中的 fit() 方法似乎在同一界面中服务于不同的目的。 应用于训练集时,像这样: model.fit(X_train, y_train) fit() 用于学习稍后将在测试集上使用 p
我使用 OSM 显示县的边界。它在大多数情况下工作得很好,但在某些情况下,县更大并且不适合 map 。 如何在开始渲染之前调整缩放级别? var map = L.map("mapCnty").setV
我正在致力于缩小和丑化我的 javascript 文件。我想知道合适的尺寸是多大。如果我将所有js文件合并成一个文件(经过缩小和丑化),它会大于1mb。我想,最好将它们分成 2-3 个文件(每个文件
我是 Java 新手。 我想在 GridPane 中放置一个 TextArea。我在过去几个小时内尝试了此操作,结果如下: 如您所见,TextArea 比我的 Gridpane 大得多。这是我的代码:
sklearn 中的 fit() 方法似乎在同一界面中服务于不同的目的。 应用于训练集时,像这样: model.fit(X_train, y_train) fit() 用于学习稍后将在测试集上使用 p
我认为这是一个基本问题,但也许我混淆了这些概念。 假设我使用 R forecast 包中的函数 auto.arima() 将 ARIMA 模型拟合到时间序列。该模型假设方差不变。我如何获得该方差?是残
我使用 OSM 显示县的边界。它在大多数情况下工作得很好,但在某些情况下,县更大并且不适合 map 。 如何在开始渲染之前调整缩放级别? var map = L.map("mapCnty").setV
我有一个很长的标签,这是我的第一个标签,我想把它放在我的单元格中。这就是我所拥有的,但它不起作用。 我有一个自定义的 UITabelviewCell ,里面有几个标签。 -(CGFloat)table
假设我有一个包含 WCS header 的 FITS 文件,这样我就可以执行以下操作: #import healpy as hp #import astropy.io.fits as pyfits #
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 这个问题似乎与 help center 中定义的范围内的编程无关。 . 已关闭10 年前。 Improve
我们正在构建一个与其他系统有多个集成接触点的应用程序。我们有效地使用 Unity 来满足我们所有的依赖注入(inject)需求。整个业务层是用接口(interface)驱动的方法构建的,实际实现在应用
我得到了 MKMapView 和一些注释。我使用下一个代码来显示所有注释: NSArray *coordinates = [self.mapView valueForKeyPath:@"annotat
我在一家托管公司工作,我们经常收到安装、新域、滞后修复等方面的请求。为了大致了解仍然开放的内容,我决定制作一个非常简单的票务系统。我有一点 php 知识和一点 MySQL 知识。目前,我们将根据客户的
我想向我的 UITableView 添加背景图像,它适合 UI,还具有导航 Controller 和工具栏。在那种情况下,我没有找到适合 iPhone 和 iPad 不同屏幕的 tableview 的
我是一名优秀的程序员,十分优秀!