c++ two classes as each others' friends

In this case, Box need access to Cup.func, AND Cup need access to Box.func, both of which are private because I don‘t want any other class to have access to neither Box.func nor Cup.func.

 

So they need to declare each other as friend. 

 

Suppose you organise code like Box1.h, Box1.cpp, Cup1.h, Cup1.cpp, it would cause a problem like this: http://www.cnblogs.com/qrlozte/p/4099032.html

 

The compiler will complain. So the solution is also simple, see Cup2.h, Cup2.cpp, other files remain unchanged. (Of couse, you can change Box1.h Box1.cpp using the same pattern as Cup2.h Cup2.cpp, although it would make your code looks more consistent, but only changing Cup1.h, Cup1.cpp is already enough to satisfy your need.)

 

Another question, what if Box.doSomething is a function template? If you change main1.cpp to main2.cpp, and Box1.h, Box1.cpp to Box2.h, Box2.cpp, the linker will complain that it cannot find the implementation of Box.doSomething (undefined reference error) (after all, the compiler need all the detail of a template to generate code, if you hide the implementation of Box.doSomething in Box2.cpp, how can you expect the compiler can generate code for Box.doSomething<int>() in main2.cpp when main2.cpp doesn‘t include Box2.cpp, right?).

 

So, because function template (and class template) must define (just declare is NOT enough) in the header, you might modify Box2.h and Box2.cpp as shown in Box3.h and Box3.cpp. And you‘ll find the problem is solved! (Now we got main2.cpp, Box3.h, Box3.cpp, Cup2.h, Cup2.cpp compiled successfully).

 

Now consider what if Cup.doSomething also need to be a function template? (The same as Box.doSomething in Box3.h). And... yes, you have to include the definition of Cup.doSomething in Cup.h, too!

 

And you if you that, the compiler will complain Box is an incomplete type thus in Cup.doSomething, "b.func" cannot be resolved. As shown in main3.cpp, Cup3.h, Cup3.cpp.

 

The reason is because in Cup3.h "class Box;" it indeed declares class Box, but just the class name, no information is provided about the interface of the class at all. 

 

Can you replace "class Box;" with "#include "Box.h"" ? No. Because that‘ll cause the same problem as Box1.h, Box1.cpp, Cup1.h, Cup1.cpp caused.

 

The solution ? At least for now, I don‘t know(If I can seperate Cup.doSomething() into two parts, one part is template and one part is normal function and the template part doesn‘t have to access memebers of the parameter ‘b‘, then there‘s a simple solution, just let the template part in the header, and implement the normal function in Cup.cpp). This situation is not made up by me, I really encountered this problem when I was making a little program. Maybe there‘s something wrong with my design.

 

main1.cpp

 1 #include "Box.h"
 2 #include "Cup.h"
 3 
 4 int main()
 5 {
 6     Cup c;
 7     Box b;
 8     c.doSomething(b);
 9     b.doSomething(c);
10     return 0;
11 }

 

Box1.h

 1 #ifndef BOX_H
 2 #define BOX_H
 3 
 4 #include "Cup.h"
 5 
 6 class Box
 7 {
 8     friend class Cup;
 9     public:
10         Box();
11         ~Box();
12         void doSomething(const Cup &c);
13     private:
14         void func() const;// only visible to Cup
15 };
16 
17 #endif // BOX_H

 

Box1.cpp

 1 #include "Box.h"
 2 
 3 #include <iostream>
 4 
 5 Box::Box()
 6 {
 7 
 8 }
 9 Box::~Box()
10 {
11 
12 }
13 void Box::doSomething(const Cup &c)
14 {
15     c.func();
16 }
17 void Box::func() const
18 {
19     using namespace std;
20     cout << "Box.func" << endl;
21 }

 

Cup1.h

 1 #ifndef CUP_H
 2 #define CUP_H
 3 
 4 #include "Box.h"
 5 
 6 class Cup
 7 {
 8     friend class Box;
 9     public:
10         Cup();
11         ~Cup();
12         void doSomething(const Box &b);
13     private:
14         void func() const; // only visible to Box
15 };
16 
17 #endif // CUP_H

 

Cup1.cpp

 1 #include "Cup.h"
 2 
 3 #include <iostream>
 4 
 5 Cup::Cup()
 6 {
 7 
 8 }
 9 Cup::~Cup()
10 {
11 
12 }
13 void Cup::doSomething(const Box &b)
14 {
15     b.func();
16 }
17 
18 void Cup::func() const
19 {
20     using namespace std;
21     cout << "Cup.func" << endl;
22 }

 

Cup2.h

 1 #ifndef CUP_H
 2 #define CUP_H
 3 
 4 class Box;
 5 
 6 class Cup
 7 {
 8     friend class Box;
 9     public:
10         Cup();
11         ~Cup();
12         void doSomething(const Box &b);
13     private:
14         void func() const; // only visible to Box
15 };
16 
17 #endif // CUP_H

 

Cup2.cpp

 1 #include "Cup.h"
 2 
 3 #include "Box.h"
 4 
 5 #include <iostream>
 6 
 7 Cup::Cup()
 8 {
 9 
10 }
11 Cup::~Cup()
12 {
13 
14 }
15 void Cup::doSomething(const Box &b)
16 {
17     b.func();
18 }
19 
20 void Cup::func() const
21 {
22     using namespace std;
23     cout << "Cup.func" << endl;
24 }

main2.cpp

 1 #include "Box.h"
 2 #include "Cup.h"
 3 
 4 int main()
 5 {
 6     Cup c;
 7     Box b;
 8     c.doSomething(b);
 9     b.doSomething<int>(1, c);
10     return 0;
11 }

 

Box2.h

#ifndef BOX_H
#define BOX_H

#include "Cup.h"

class Box
{
    friend class Cup;
    public:
        Box();
        ~Box();
        template <typename T> void doSomething(const T &obj, const Cup &c);
    private:
        void func() const;// only visible to Cup
};

#endif // BOX_H

 

Box2.cpp

 1 #include "Box.h"
 2 
 3 #include <iostream>
 4 
 5 Box::Box()
 6 {
 7 
 8 }
 9 Box::~Box()
10 {
11 
12 }
13 template <typename T> void doSomething(const T &obj, const Cup &c)
14 {
15     c.func();
16 }
17 void Box::func() const
18 {
19     using namespace std;
20     cout << "Box.func" << endl;
21 }

Box3.h

 1 #ifndef BOX_H
 2 #define BOX_H
 3 
 4 #include "Cup.h"
 5 
 6 class Box
 7 {
 8     friend class Cup;
 9     public:
10         Box();
11         ~Box();
12         template <typename T> void doSomething(const T &obj, const Cup &c);
13     private:
14         void func() const;// only visible to Cup
15 };
16 
17 template <typename T> void Box::doSomething(const T &obj, const Cup &c)
18 {
19     c.func();
20 }
21 
22 #endif // BOX_H

 

Box3.cpp

 1 #include "Box.h"
 2 
 3 #include <iostream>
 4 
 5 Box::Box()
 6 {
 7 
 8 }
 9 Box::~Box()
10 {
11 
12 }
13 
14 void Box::func() const
15 {
16     using namespace std;
17     cout << "Box.func" << endl;
18 }

main3.cpp

 1 #include "Box.h"
 2 #include "Cup.h"
 3 
 4 int main()
 5 {
 6     Cup c;
 7     Box b;
 8     c.doSomething<int>(1, b);
 9     b.doSomething<int>(1, c);
10     return 0;
11 }

 

Cup3.h

 1 #ifndef CUP_H
 2 #define CUP_H
 3 
 4 class Box;
 5 
 6 class Cup
 7 {
 8     friend class Box;
 9     public:
10         Cup();
11         ~Cup();
12         template <typename T> void doSomething(const T &obj, const Box &b);
13     private:
14         void func() const; // only visible to Box
15 };
16 
17 template <typename T> void doSomething(const T &obj, const Box &b)
18 {
19     b.func();
20 }
21 
22 #endif // CUP_H

 

Cup3.cpp

 1 #include "Cup.h"
 2 
 3 #include "Box.h"
 4 
 5 #include <iostream>
 6 
 7 Cup::Cup()
 8 {
 9 
10 }
11 Cup::~Cup()
12 {
13 
14 }
15 
16 void Cup::func() const
17 {
18     using namespace std;
19     cout << "Cup.func" << endl;
20 }

 

郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。