5

这更像是一个 c++ 标准问题。考虑以下代码:

template <typename T>
class has_Data
{
   typedef char one;
   typedef long two;

   template <typename C> static one test( typeof(&C::Data) ) ;
   template <typename C> static two test(...);

public:
   enum { value = sizeof(test<T>(0)) == sizeof(char) };
};

class MyClass {
private:
   struct Data {
   };
};


void function(bool val = has_Data<MyClass>::value) {}

上面的代码适用于gcc (GCC) 4.4.3

然而随着clang version 3.3 (2545b1d99942080bac4a74cda92c620123d0d6e9) (2ff97832e593926ea8dbdd5fc5bcf367475638a9)

它给出了这个错误:

test_private_data.cpp:7:54: error: 'Data' is a private member of 'MyClass'
   template <typename C> static one test( typeof(&C::Data) ) ;
                                                     ^
/devshared/home/rhanda/test_private_data.cpp:7:37: note: while substituting explicitly-specified template arguments into function template 'test'
   template <typename C> static one test( typeof(&C::Data) ) ;
                                    ^
/devshared/home/rhanda/test_private_data.cpp:21:26: note: in instantiation of template class 'has_Data<MyClass>' requested here
void function(bool val = has_Data<MyClass>::value) {}
                         ^
1 error generated.

哪一个是对的?

从标准文档(n3485)中,我发现了一个似乎比 gcc 更同意 clang 的声明。

访问控制统一应用于所有名称,无论名称是从声明还是表达式中引用。

4

1 回答 1

3

我认为 GCC 是正确的。

首先要注意的是,没有非friend代码应该能够肯定地报告给定私有成员的存在。因此,如果这是您尝试做的事情,则必须修改您的设计。一个类可以用它的私有成员做任何事情,其他代码(除了朋友)应该没有办法知道它。这是设计使然。

但是,有SFINAE原则:替换失败不是错误。由于MyClass::Data是私有的,所以has_Data在我看来,代码应该表现得好像根本没有C::Data成员一样。因此,第一个函数会导致替换失败,它会被默默地忽略,而第二个函数就是使用的那个。添加更多代码,我的 GCC 4.7.2 编译它没有问题并且has_Data<MyClass>::value评估为false. 我认为正确的 SFINAE。

试图用您提到的文件中的引用来支持这一观点,我在第 14.8.2 节第 8 段中找到了以下内容:

注意:访问检查是替换过程的一部分。

这是标准中的非规范性注释,但对我来说似乎是一个非常易读和明确的指示,表明 SFINAE 实际上应该适用于这种情况,就像 GCC 处理它的方式一样。

编辑:正如@hvd在评论中指出的那样,以上仅适用于 C++11。在旧版本的标准中,情况过去有所不同。问题 1170:模板参数推导期间的访问检查包含有关该更改的详细信息。

由于是 GNU 扩展-std=c++03,GCC 不会编译此代码。仍然编译代码的事实可能被认为是不合适的,但由于前进的方向是使用 C++11 语义,我不会费心提交有关此的报告。-std=c++11typeof-std=gnu++03

于 2013-04-25T08:47:50.263 回答