1

我正在调试一个与第三方库交互的应用程序,其代码不可用,只有标题和 .so 可用。现在我可以将它加载到调试器中并检查在第三方库中声明的类的私有成员的变量值,但是由于对象的数量很大,我想创建一些机制来在控制台上打印它,以便稍后进行分析。我想出了这样的东西

第三方标头

class A
{
    private:
    int i;
};

我没有在上面的课程中包含额外的细节

调试打印机.cpp

#include <thirdpartheaders>

template <typename T> class debugprinter
{
    friend class T;
    public :
    void printonconsole()
    {
        T a;
        std::cout << std::endl << a.i << std::endl;
        return;
    }
}

现在我尝试在上面编译,但似乎我不能将未定义的类型 T 声明为我的模板类的朋友并得到这个错误

错误:无法从 x::acc() 访问 i

现在我可以通过创建非模板调试打印机来解决这个问题,但是出于好奇,有没有一种方法可以创建一个模板类,它可以成为它的输入类型参数的朋友?

谢谢

4

3 回答 3

5

宣布A为 的朋友是无济于事的debugprinter<A>friend关系不是对称的。这种关系只是单向的。如果您想访问 的私人成员,您需要debugprinter<A>成为 的朋友。为此,您需要修改类本身。AAA

这就像现实世界的友谊。你不能强迫别人成为你的朋友。你只能对别人表现得友好。

于 2013-09-23T12:38:29.963 回答
3

你可以宣布T为朋友,但这对你没有任何好处。需要一种方法让您的模板获取 中的私有数据T,反之亦然:T必须将您的模板声明为朋友。关于友谊的决定是由提供友谊的班级做出的;不能从其他班级要求。没有朋友声明,就没有合法的方式从外部获取班级的私密部分;这就是私人的意思。

于 2013-09-23T12:37:08.827 回答
2

第一件事是,正如其他人提到的那样,友谊不是对称的,并且您正在尝试错误的方向(授予对 . 的A访问权限debugprinter<A>。话虽如此,只是为了人们搜索问题:

在旧版本的标准 (C++03) 中,无法直接将模板参数声明为友元。这个限制在 C++11 中被取消了,尽管它需要稍微不同的语法:

template <typename T>
class test {
   friend T;             // note, not 'friend class T'!!!
};

不过,在 C++03 中,您可以通过使用额外的间接级别来实现相同的行为。您不能与模板参数成为朋友,但您可以与依赖类型成为朋友,并且通过使用identity元函数,您可以实现您想要的:

template <typename T>
struct identity {
   typedef T type;
};
template <typename T>
class test {
   friend class identity<T>::type;
};
于 2013-09-23T12:59:52.297 回答