1为何要使用友元函数
在实现类之间数据共享时,减小系统开销,提升效率。若是类A中的函数要访问类B中的成员(例如:智能指针类的实现),那么类A中该函数要是类B的友元函数。具体说:为了
使其余类的成员函数直接访问该类的私有变量。即:容许外面的类或函数去访问类的私有变量和保护变量,从而使两个类共享同一函数。
实际上具体大概有下面两种状况须要使用友元函数:(1)运算符重载的某些场合须要使用友元。(2)两个类要共享数据的时候。ios
使用友元函数的优缺点
优势:可以提升效率,表达简单、清晰。
缺点:友元函数破环了封装机制,尽可能不使用成员函数,除非不得已的状况下才使用友元函数。函数
2.友元函数的使用this
友元函数的参数:
由于友元函数没有this指针,则参数要有三种状况:
要访问非static成员时,须要对象作参数
要访问static成员或全局变量时,则不须要对象作参数;
若是作参数的对象是全局对象,则不须要对象作参数;spa
友元函数的位置
由于友元函数是类外的函数,因此它的声明能够放在类的私有段或公有段且没有区别。指针
友元函数的调用
能够直接调用友元函数,不须要经过对象或指针对象
友元函数的分类:
根据这个函数的来源不一样,能够分为三种方法:blog
普通函数友元函数继承
目的:使普通函数可以访问类的友元io
语法:
声明: friend + 普通函数声明
实现位置:能够在类外或类中
实现代码:与普通函数相同
调用:相似普通函数,直接调用class
代码:
class Interger { friend void Print(const INTEGER& obj);//声明友元函数 }; void Print(const INTEGER& obj) { //函数体 } void main() { Interger obj; Print(obj);//直接调用 }
类Y的全部成员函数都为类X友元函数—友元类
目的:使用单个声明使Y类的全部函数成为类X的友元,它提供一种类之间合做的一种方式,使类Y的对象能够具备类X和类Y的功能。
语法:
声明位置:公有私有都可,常写为私有(把类当作一个变量)
声明: friend + 类名(不是对象哦)
代码:
class girl; class boy { public: void disp(girl &); }; void boy::disp(girl &x) //函数disp()为类boy的成员函数,也是类girl的友元函数 { cout<<"girl's name is:"<<x.name<<",age:"<<x.age<<endl;//借助友元,在boy的成员函数disp中,借助girl的对象,直接访问girl的私有变量 } class girl { private: char *name; int age; friend boy; //声明类boy是类girl的友元 };
类Y的一个成员函数为类X的友元函数
目的:使类Y的一个成员函数成为类X的友元,具体而言:在类Y的这个成员函数中,借助参数X,能够直接以X的私有变量
语法:
声明位置:声明在公有中 (自己为函数)
声明:friend + 成员函数的声明
调用:先定义Y的对象y---使用y调用本身的成员函数---本身的成员函数中使用了友元机制
代码:
实现代码和上一代码中的实现及其类似只是设置友元的时候变为friend void boy::disp(girl &);
小结:其实一些操做符的重载实现也是要在类外实现的,那么一般这样的话,声明为类的友元是必须滴。
友元函数和类的成员函数的区别
成员函数有this指针,而友元函数没有this指针。
友元函数是不能被继承的,就像父亲的朋友未必是儿子的朋友。
友元函数
代码:
/// /// @file friendfunc.cc /// @author Burgess Fan(1406832037@qq.com) /// @date 2016-06-20 20:38:18 /// #include <iostream> using namespace std; class SalesData{ friend void PrintInfo(SalesData &sdata); public: SalesData(const char*s,unsigned int n,double p) :BookNum(s) ,SaleNum(n) ,BookPrice(p) { cout<<"SalesData()"<<endl; } ~SalesData() { cout<<"~SalesData"<<endl; } private: const char* BookNum; unsigned int SaleNum; double BookPrice; }; void PrintInfo(SalesData &sdata){ cout<<"BookNum is:"<<sdata.BookNum<<endl; cout<<"SaleNum is:"<<sdata.SaleNum<<endl; cout<<"BookPrice is:"<<sdata.BookPrice<<endl; } int main(void){ SalesData s("2016006",10,20.5); PrintInfo(s); return 0; }