什么是友元?
友元是 C++ 中的一种关系
友元发生在函数与类之间或者类与类之间
友元关系是单项的,不能传递
友元的用法
在类中以 friend 关键字声明友元
类的友元可以是其他类或者具体函数
友元不是类的一部分
友元不受类中访问级别的限制
友元可以直接访问具体类的所有成员
友元的语法
在类中用 friend 关键字对函数或类进行声明
友元的使用初探
#include <stdio.h>
#include <math.h>
class Point
{
double x;
double y;
public:
Point(double x, double y)
{
this->x = x;
this->y = y;
}
double getX()
{
return x;
}
double getY()
{
return y;
}
friend double func(Point& p1, Point& p2);
};
double func(Point& p1, Point& p2)
{
double ret = 0;
ret = (p2.y - p1.y) * (p2.y - p1.y) +
(p2.x - p1.x) * (p2.x - p1.x);
ret = sqrt(ret);
return ret;
}
int main()
{
Point p1(1, 2);
Point p2(10, 20);
printf("p1(%f, %f)\n", p1.getX(), p1.getY());
printf("p2(%f, %f)\n", p2.getX(), p2.getY());
printf("|(p1, p2)| = %f\n", func(p1, p2));
return 0;
}
第 25 行,将 func 函数设置为 Point 类的友元,func 函数可以访问 Point 类的私有成员
程序运行结果如下图所示:
友元的尴尬
友元是为了兼顾 C 语言的高效而诞生的
友元直接破坏了面向对象的封装性
友元在实际产品中的高效是得不偿失的
友元在现代软件工程中已经逐渐被遗弃
注意事项
友元关系不具备传递性
类的友元可以是其他类的成员函数
类的友元可以是某个完整的类
- 所有的成员函数都是友元
友元的深入分析
#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;
}
第 12 行,我们将 ClassB 声明为 ClassC 的友元,所以 ClassB 可以访问 ClassC 的所有成员变量和成员函数;第 29 行,我们将 ClassA 声明为 ClassB 的友元,所以 ClassA 可以访问 ClassB 的所有成员变量和成员函数;
第 46 行,ClassA 想要访问 ClassC 的私有成员变量,由于友元不具备传递性,ClassA 不是 ClassC 的友元,所以 ClassA 是没有权限访问 ClassC 的私有成员变量的
程序运行结果如下图所示:
小结
友元是为了兼顾 C 语言的高效而诞生的
友元直接破坏了面向对象的封装性
友元关系不具备传递性
类的友元可以是其他类的成员函数
类的友元可以是某个完整的类