C++之友元函数

时间:2022-09-19 00:10:45

1、为什么要引入友元函数:在实现类之间数据共享时,减少系统开销,提高效率

具体来说:为了使其他类的成员函数直接访问该类的私有变量

即:允许外面的类或函数去访问类的私有变量和保护变量,从而使两个类共享同一函数

优点:能够提高效率,表达简单、清晰

缺点:友元函数破环了封装机制,尽量不使用成员函数,除非不得已的情况下才使用友元函数。

2、什么时候使用友元函数:

1)运算符重载的某些场合需要使用友元。

2)两个类要共享数据的时候

3、怎么使用友元函数:

友元函数的参数:

因为友元函数没有this指针,则参数要有三种情况:

1、  要访问非static成员时,需要对象做参数;--常用(友元函数常含有参数)

2、  要访问static成员或全局变量时,则不需要对象做参数

3、  如果做参数的对象是全局对象,则不需要对象做参数

友元函数的位置:

因为友元函数是类外的函数,所以它的声明可以放在类的私有段或公有段且没有区别。

友元函数的调用:

可以直接调用友元函数,不需要通过对象或指针

友元函数的分类:

根据这个函数的来源不同,可以分为三种方法:

1、普通函数友元函数:

a) 目的:使普通函数能够访问类的友元

b) 语法:声明位置:公有私有均可,常写为公有

声明: friend + 普通函数声明

实现位置:可以在类外或类中

实现代码:与普通函数相同(不需要加friend或类的作用域运算符::)

调用:类似普通函数,直接调用

class INTEGER
{
private:
int num;
public:
friend void Print(const INTEGER& obj);//声明友元函数
};
void Print(const INTEGER& obj)//不使用friend和类::
{
//函数体
}
void main()
{
INTEGER obj;
Print(obj);//直接调用
}

2、类Y的所有成员函数都为类X友元函数—友元类—友元类A中的所有成员函数都能够访问将A声明为友元的私有成员

a)目的:它提供一种类之间合作的一种方式,使类Y的对象可以具有类X和类Y的功能

具体来说:

前提:A是B的友元(=》A中成员函数可以访问B中有所有成员,包括私有成员和公有成员)

b)语法:声明位置:公有私有均可,常写为私有(把类看成一个变量)

声明: friend + 类名---此时声明的是类,而不是对象哈

class girl;

class boy
{
private:
char *name;
int age;
public:
boy();
void disp(girl &);
}; void boy::disp(girl &x) //函数disp()为类boy的成员函数,也是类girl的友元函数
{
cout<<"boy's name is:"<<name<<",age:"<<age<<endl;//正常情况,boy的成员函数disp中直接访问boy的私有变量
cout<<"girl's name is:"<<x.name<<",age:"<<x.age<<endl;
//借助友元,在boy的成员函数disp中,借助girl的对象,直接访问girl的私有变量
//正常情况下,只允许在girl的成员函数中访问girl的私有变量
} class girl
{
private:
char *name;
int age;
friend boy; //声明类boy是类girl的友元
public:
girl();
};
void main()
{
boy b;
girl g;
b.disp(g); //b调用自己的成员函数,但是以g为参数,友元机制体现在函数disp中
}

3、类Y的一个成员函数为类X的友元函数

a)目的:使类Y的一个成员函数成为类X的友元

具体而言:而在类Y的这个成员函数中,借助参数X,可以直接访问X中的私有变量

b)语法:声明位置:声明在公有中 (本身为函数)

声明:friend + 成员函数的声明

调用:先定义Y的对象y---使用y调用自己的成员函数---自己的成员函数中使用了友元机制

class girl;
class boy
{
private:
char *name;
int age;
public:
boy();
void disp(girl &);
}; class girl
{
private:
char *name;
int age;
public:
girl(char *N,int A);
friend void boy::disp(girl &); //声明类boy的成员函数disp()为类girl的友元函数
}; void boy::disp(girl &x)
{
cout<<"boy's name is:"<<name<<",age:"<<age<<endl; //访问自己(boy)的对象成员,直接访问自己的私有变量
cout<<"girl's name is:"<<x.name<<",age:"<<x.age<<endl;
//借助友元,在boy的成员函数disp中,借助girl的对象,直接访问girl的私有变量
//正常情况下,只允许在girl的成员函数中访问girl的私有变量
}
void main()
{
boy b();
girl g();
b.disp(g); }

4、在模板类中使用友元operator<<(对<<运算符的重载)

a)使用方法:

在模板类中声明:

friend ostream& operator<< <>(ostream& cout,const MGraph<VexType,ArcType>& G);

在模板类中定义:

template<class VexType,class ArcType>
ostream& operator<<(ostream& cout,const MGraph<VexType,ArcType>& G)
{
//函数定义
}

b)注意:

把函数声明非模板函数:

friend ostream& operator<< (ostream& cout,const MGraph& G);

把函数声明为模板函数:

friend ostream& operator<< <>(ostream& cout,const MGraph<VexType,ArcType>& G);

说明:
 在函数声明中加入operator<< <>:是将operator<<函数定义为函数模板,将函数模板申明为类模板的友员时,是一对一绑定的
 实际的声明函数:这里模板参数可以省略,但是尖括号不可以省略

friend ostream& operator<< <VexType,ArcType>(ostream& cout,const MGraph<VexType,ArcType>& G);

5、友元函数和类的成员函数的区别:成员函数有this指针,而友元函数没有this指针。

6、记忆:A是B的友元《=》A是B的朋友《=》借助B的对象,在A中可以直接 通过B.成员变量(可以是公有,也可以为私有变量) 的方式访问B

C++之友元函数的更多相关文章

  1. C&plus;&plus;的友元类和友元函数实例

    #include <math.h> #include<iostream> using namespace std; class Point { public: Point(do ...

  2. c&plus;&plus;友元函数

    c++友元函数分两类: 一://友员全居函数 /*#include <iostream>using namespace std;class aaa{    friend void prin ...

  3. 重载运算符:类成员函数or友元函数

    类成员函数: bool operator ==(const point &a)const { return x==a.x; } 友元函数: friend bool operator ==(co ...

  4. 不可或缺 Windows Native &lpar;20&rpar; - C&plus;&plus;&colon; 友元函数&comma; 友元类

    [源码下载] 不可或缺 Windows Native (20) - C++: 友元函数, 友元类 作者:webabcd 介绍不可或缺 Windows Native 之 C++ 友元函数 友元类 示例演 ...

  5. &lbrack;Reprint&rsqb;C&plus;&plus;友元函数与拷贝构造函数详解

    这篇文章主要介绍了C++友元函数与拷贝构造函数,需要的朋友可以参考下   一.友元函数 1.友元函数概述: (1)友元函数是定义在一个类外的普通函数.友元函数和普通函数的定义一样;在类内必须将该普通函 ...

  6. C&plus;&plus;学习12 友元函数和友元类

    友元函数和友元类在实际开发中较少使用,想快速学习C++的读者可以跳过本节. 一个类中可以有 public.protected.private 三种属性的成员,通过对象可以访问 public 成员,只有 ...

  7. C&plus;&plus;:友元(非成员友元函数、成员友元函数、友元类)

    3.8  友元:友元函数和友元类 友元函数 :既可以是不属于任何类的非成员函数,也可以是另一个类的成员函数,统称为友元函数.友元函数不是当前类的成员函数,而是独立于类的外部函数,但它可以访问该类所有的 ...

  8. C&plus;&plus;重载(主要介绍使用友元函数重载)

    重载限制 多数C++运算符都可以用下面的方式重载.重载的运算符不必是成员函数,但必须至少有一个操作数是用户自定义的类型.下面详细介绍C++对用户定义的运算符重载的限制. 1 重载后的运算符必须至少有一 ...

  9. Effective C&plus;&plus; 第二版 17&rpar;operator&equals;检查自己 18&rpar;接口完整 19&rpar;成员和友元函数

    条款17 在operator=中检查给自己赋值的情况 1 2 3 class  X { ... }; X a; a = a;  // a 赋值给自己 >赋值给自己make no sense, 但 ...

  10. &lbrack;置顶&rsqb; c&plus;&plus;&comma;vc6&period;0&comma;中友元函数&comma;无法访问私有字段&lpar;private&rpar;的问题&lpar;problem&rpar;&comma;cannot access private member declared in class &&num;39&semi;Date&&num;39&semi;

    c++,vc6.0,中友元函数,无法访问私有字段(private)的问题(problem),cannot access private member declared in class 'Date' ...

随机推荐

  1. PHP-解析验证码类--学习笔记

    1.开始 在 网上看到使用PHP写的ValidateCode生成验证码码类,感觉不错,特拿来分析学习一下. 2.类图 3.验证码类部分代码 3.1  定义变量 //随机因子 private $char ...

  2. href,src,url 整理

    一.href 和 src 的定义及区别 href:Hypertext Reference(超文本引用),指定网络资源的位置,从而在当前元素或者当前文档和由当前属性定义的需要的锚点或资源之间定义一个链接 ...

  3. blktrace

    统计块设备层io信息. ● 安装 http://blog.csdn.net/hs794502825/article/details/8545133 出现问题,安装新立得软件包管理器:apt-get i ...

  4. Cocos2d-x 核心概念 - 层&lpar;Layer&rpar;

    层(Layer) 一个简单的主界面是由是三个层叠加实现的,从上到下依次为,菜单层,精灵层,背景层 这个次序适用与事件的响应机制,菜单层最先接受到系统事件,然后精灵层,最后背景层 在事件传递的过程中,如 ...

  5. 又一次的Microsoft Visual C&plus;&plus; 10&period;0 is required &lpar;Unable to find vcvarsall&period;bat&rpar;

    ~~~~~~~~~~~My problem is here~~~~~~~~~~~~~~~~~~~~~~ Error: Microsoft visual C++ 10.0 is required (un ...

  6. thinkphp隐藏中url的index&period;php

    在本地进行测试 1.修改apache配置文件将如下代码#去掉 #LoadModule rewrite_module modules/mod_rewrite.so   在index.php 目录下新建文 ...

  7. Java序列化 如何把多个对象存储在一个文件中

    /** * 用于保存模板文件,内容包括: * 1,标志位,1 int * 2,版本   1 int * 3,数据头长度 1 int * 4,预留数据头空间  5120 byte * 5,后续数据长度  ...

  8. WPF组件开发

    在做组件之前,为了适应框架,我们需要有一个基类,并将这个基类打包成一个模板,让大部分组件去使用这个模板. 组件的基类就不多讲了,上篇文章中已经说过了.这是地址: http://www.cnblogs. ...

  9. JavaScript实战

    JavaScript之单例实战 一.概述 所谓单例模式,顾名思义即一个类只有一个实例. 所以,当我们创建一个实例时,就必须判断其是否已经存在了这个实例,如果已经存在了这个实例,那么就返回这个已经存在的 ...

  10. python开发之virtualenv与virtualenvwrapper讲解

    在使用 Python 开发的过程中,工程一多,难免会碰到不同的工程依赖不同版本的库的问题: 亦或者是在开发过程中不想让物理环境里充斥各种各样的库,引发未来的依赖灾难. 此时,我们需要对于不同的工程使用 ...