你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

c++中的友元

[复制链接]
gaosmile 发布时间:2020-8-11 11:28
一、友元的概念:
1、什么是友元?
  • 友元是c++中的一种关系
  • 友元关系发生在函数与类之间或者类与类之间
  • 友元关系是单项的,不能传递

微信图片_20200811112716.png
2、友元的用法:
  • 在类中以friend关键字声明友元
  • 类的友元可以是其它类或者具体函数
  • 友元不是类的一部分
  • 友元不受类中访问级别的限制
  • 友元可以直接访问具体类的所有成员

3、友元的语法:
在类中使用friend 关键字对函数或者类进行声明:
class Test
{
     double x;
     double y;

     friend void func(Test& t);

};

void func(Test& t)
{


}

注意:func() 全局函数是 Test 类的友元,func() 可以访问Test 类的所有成员,但是 func() 不是 Test 的成员函数。
示例代码:
#include <stdio.h>
#include <math.h>

class Test
{
    double x;
    double y;
public:
    Test(double x,double y)
    {
       this->x=x;
       this->y=y;
    }
    double getX()
    {
       return x;
    }
    double getY()
    {

       return y;
     }
    friend double func(Test& t1,Test& t2);

};

double func(Test& t1,Test& t2)
{
       double ret =0;
       ret = (t2.y-t1.y)*(t2.y-t1.y)+(t2.x-t1.x)*(t2.x-t1.x);

        ret = sqrt(ret);

     return ret;

}
int main()
{
    Test t1(1,2);
    Test t2(10,20);

    printf("t1(%f,%f)\n",t1.getX(),t1.getY());
    printf("t2(%f,%f)\n",t2.getX(),t2.getY());
    printf("(t1,t2)=%f\n",func(t1,t2));

   
    return 0;
}

输出结果:
root@txp-virtual-machine:/home/txp/add# g++ test.cpp
root@txp-virtual-machine:/home/txp/add# ./a.out
t1(1.000000,2.000000)
t2(10.000000,20.000000)
(t1,t2)=20.124612

4、友元的尴尬:
  • 友元是为了兼顾c语言的高效而诞生的
  • 友元直接破坏了面向对象的封装性
  • 友元在实际开发产品中的高效是得不偿失的
  • 友元在软件工程中已经慢慢被遗弃了

5、注意事项:
  • 友元关系不具备传递性
  • 类的友元可以是其它类的成员函数
  • 类的友元可以是某个完整的类
  • 所有的成员函数都是友元

微信图片_20200811112721.png
代码测试:
#include <stdio.h>

class ClassC
{
    const char* n;
public:
    ClassC(const char* n)
    {
        this->n = n;
    }
   
    friend class ClassB;
};

class ClassB
{
    const char* n;
public:
    ClassB(const char* n)
    {
        this->n = n;
    }
   
    void getClassCName(ClassC& c)
    {
        printf("c.n = %s\n", c.n);
    }
   
    friend class ClassA;
};

class ClassA
{
    const char* n;
public:
    ClassA(const char* n)
    {
        this->n = n;
    }
   
    void getClassBName(ClassB& b)
    {
        printf("b.n = %s\n", b.n);
    }
    /*
    void getClassCName(ClassC& c)
    {
        printf("c.n = %s\n", c.n);
    }
    */
};

int main()
{
    ClassA A("A");
    ClassB B("B");
    ClassC C("C");
   
    A.getClassBName(B);
    B.getClassCName(C);
   
    return 0;
}

输出结果:
root@txp-virtual-machine:/home/txp/add# ./a.out
b.n = B
c.n = C

如果把上面屏蔽的那部分代码打开,编译就会报错(因为友元没有传递性哦):
root@txp-virtual-machine:/home/txp/add# g++ test.cpp
test.cpp: In member function ‘void ClassA::getClassCName(ClassC&)’:
test.cpp:5:17: error: ‘const char* ClassC::n’ is private
     const char* n;
                 ^
test.cpp:48:32: error: within this context
         printf("c.n = %s\n", c.n);
                                ^

6、小结:
  • 友元是为了兼顾c语言的高效而诞生的
  • 友元直接破坏了面向对象的封装性
  • 友元关系不具备传递性
  • 类的友元可以是其它类的成员函数
  • 类的友元可以是某个完成的类


二、总结:

好了,今天的分享就到这里,如果文章中有错误或者不理解的地方,可以交流互动,一起进步。

收藏 评论0 发布时间:2020-8-11 11:28

举报

0个回答

所属标签

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版