使用以下代码(我的原始代码的简化版本)
#include <iostream>
#include <cmath>
template <typename> class A; // edit 1 following Mark & Matthieu
template <typename X> class A {
X a;
template <typename> friend class A; // edit 1 following Mark & Matthieu
public:
A(X x) : a(x) {}
X get() const { return a; } // edit 2 to avoid using A<Y>::a
template <typename Y>
auto diff(A<Y> const& y) const
-> decltype(a - y.a) // original code causing error with gcc
-> typename std::common_type<X, Y>::type // alternative following Rook
-> decltype(this->get() - // edit 3 not using A<X>::a
y.get()) // edit 2 not using A<Y>::a
{ return a - y.get(); }
};
template <typename X, typename Y>
inline auto dist(A<X> const& x, A<Y> const& y) -> decltype(std::abs(x.diff(y)))
{ return std::abs(x.diff(y)); }
int main()
{
A<double> x(2.0), y(4.5);
std::cout << " dist(x,y)=" << dist(x,y) << '\n'; // <-- error here
}
gcc 4.7.0 出现以下错误:
test.cc: 在函数
decltype (std::abs(x.diff(y))) dist(const A<X>&, const A<Y>&)
中 [withX = double; Y = double; decltype (std::abs(x.diff(y))) = double
]':test.cc:5:5:错误:
double A<double>::a
是私有的突出显示的行:错误:在此上下文中
这个错误信息显然不是很有帮助。我的代码有错误吗?或者这是编译器的问题?
EDIT1:朋友声明没有帮助。
EDIT2:避免使用A<Y>::a
也没有帮助。
EDIT3:与EDIT2一起终于解决了这个问题。的decltype()
定义中dist()
需要decltype()
for A<X>::diff()
,而后者又使用A<X>::a
,它在第一个上下文中是私有的。
EDTI4:Rook 的使用建议typename std::common_type<X,Y>::type
也有效!
EDIT5:但请参阅 Jonathan Wakely 对这个问题的回答