网站后台管理怎么进,电脑iis做网站,公司网站框架,双鸭山网站开发食用指南#xff1a;本文在有C基础的情况下食用更佳 #x1f340;本文前置知识#xff1a; C类 ♈️今日夜电波#xff1a;クリームソーダとシャンデリア—Edo_Ame江户糖 1:20 ━━━━━━️#x1f49f;──────── 3:40 … 食用指南本文在有C基础的情况下食用更佳 本文前置知识 C类 ♈️今日夜电波クリームソーダとシャンデリア—Edo_Ame江户糖 1:20 ━━━━━━️──────── 3:40 ◀️ ⏸ ▶️ ☰ 关注点赞收藏您的每一次鼓励都是对我莫大的支持 目录
一、运算符重载基本概念
什么是运算符重载
运算符重载简要干货
可重载的运算符有哪些
二、前置知识-友元函数 什么是友元函数 友元函数的语法 三、运算符重载
运算符重载的语法
一步一步带你实现运算符重载以为例
运算符重载作为成员函数以及全局函数的实现以为例
全局函数
成员函数
四、和--运算符重载 重要、常用
具体实现用例 一、运算符重载基本概念
什么是运算符重载 运算符重载 就是对已有的运算符重新进行定义 赋予其另一种功能 以适应不同 的数据类型。 运算符重载(operator overloading)只是一种”语法上的方便”,也就是它只是另一种函 数调用的方式。 在 c中 可以定义一个处理类的新运算符。 这种定义很像一个普通的函数定义只是函数的名字由关键字 operator 及其紧跟的运算符组成。 差别仅此而已。 它像任何其他函数一样也是一个函数 当编译器遇到适当的模式时 就会调用这个函数。 运算符重载简要干货 运算符重载的目的简化操作 让已有的运算符 适应适应不同的数据类型。 语法函数的名字由关键字operator及其紧跟的运算符组成 比如重载运算符 operator 重载号运算 operator 注意重载运算符 不要更改 运算符的本质操作是数据的相加 不要重载成相减 栗子以下为重载了运算符的类
class Data
{friend ostream operator(ostream out, Data ob);//友元函数经常与运算符重载搭配使用
private:int a;int b;
public:Data(){cout 无参的构造函数 endl;a 0;b 0;}Data(int a, int b) :a(a), b(b){cout 有参构造 endl;//this‐a a;//this‐b b;}void showData(void){cout a a , b b endl;}~Data(){cout 析构函数函数 endl;}
};ostream operator(ostream out, Data ob){out a ob.a , b ob.b;return out;} 解释 为了简化类中访问私有数据较为困难的问题运用友元函数下小点会提到同重载运算符的结合得以运用我们较为常用的直接输出数据。 可重载的运算符有哪些 几乎 C 中所有的运算符都可以重载 但运算符重载的使用时相当受限制的。 特别是不能使用 C 中当前没有意义的运算符(例如用**求幂)不能改变运算符优先级 不能改变运算符的参数个数。 这样的限制有意义 否则 所有这些行为产生的运算符只会混淆而不是澄清寓语意。 一张图囊括~ 二、前置知识-友元函数 什么是友元函数 一句话概括C允许 友元 访问 私有数据。 友元函数的语法
friend定义的函数 注意 friend关键字只出现在声明处 其他类、类成员函数、全局函数都可声明为友元 友元函数不是类的成员不带this指针 友元函数可访问对象任意成员属性包括私有属性。 栗子 创建一个房间类你只准许你的朋友进入你的卧室但是客厅是谁都可以进的
class Room{//将goodGayVisit作为类的友元函数//goodGayVisit 访问 类中所有数据 但是 它不是类的成员friend void goodGayVisit(Room room);private:string bedRoom;//卧室public:string sittingRoom;//客厅public: Room(){this- bedRoom 卧室;this- sittingRoom 客厅;}};// 普通全局函数 作为 类的友元//好基友 访问 我的房间void goodGayVisit(Room room){cout 好基友访问了你的 room.sittingRoom endl;cout 好基友访问了你的 room.bedRoom endl;//ok}void test01(){Room myRoom;goodGayVisit(myRoom);
} friend在这里可以访问对象任意成员属性包括私有属性。因此本来不能访问的私有数据在friend的情况下就可以访问了结果如下 此为普通全局函数 作为 类的友元 。当然也有类的某个成员函数 作为 另一个类的友元一个类整体 作为 另一个类的友元等等。 而我们的友元函数大多应用在重载运算符上 本文仅仅对友元函数做简单介绍如果大家需要详解请在评论区或者私信踢我一脚o(╯□╰)o作者肯定会出一篇的 三、运算符重载
运算符重载的语法
根据自身改变的返回类型operator 重载的运算符根据实际情况改变的传参 函数声明运算符重载是通过在类中定义特殊的成员函数来实现的。这些成员函数被称为运算符重载函数。例如如果要重载运算符则需要在类中声明一个名为operator的函数。 函数名运算符重载函数的命名规则是以operator关键字开始后面跟着要重载的运算符符号。例如要重载“运算符函数名应为operator”。 参数列表运算符重载函数的参数列表取决于所重载的运算符。例如对于二元运算符如, “-”, “*”, “/“等参数列表应包含一个额外的参数表示右操作数。对于一元运算符如””, – – 等参数列表不需要额外的参数。 返回类型运算符重载函数的返回类型取决于所重载的运算符。例如对于运算符返回类型通常是所操作对象的类型。 成员函数或友元函数运算符重载函数可以作为类的成员函数或友元函数来定义。成员函数形式的运算符重载函数将使用对象本身作为左操作数而友元函数形式的运算符重载函数将不使用任何对象。 一步一步带你实现运算符重载以为例 注意此代码未能实现重载 下文为对用cout来输出类的一个引入
#define _CRT_SECURE_NO_WARNINGS 01
#include iostream
#includestring.husing namespace std;class Person{private:char* name;int num;public:Person(char* name, int num){this- name new char[strlen(name) 1];strcpy(this- name, name);this- num num;cout 有参构造 endl;}//普通的成员函数void printPerson(void){cout name name , num num endl;}~Person(){if (this- name ! NULL){delete[] this- name;this- name NULL;}cout 析构函数 endl;}};int main(int argc, char* argv[]){char arr[] lucy;Person ob1(arr, 18);//普通的成员函数 遍历信息//ob1.printPerson();//cout默认输出方式 无法识别 自定义对象 输出格式coutob1endl;//errreturn 0;} 运行改代码我们发现编译器报错如下图 这个时候我们就需要对运算符进行重载了 那么问题又来了如何重载运算符呢根据上文所提到的语法我们做出以下的操作 运用operator来重载运算符 ostream operator(ostream out, Person ob)//outcout, ob ob1{//重新实现 输出格式out ob.name , ob.num;//每次执行为 返回值得到coutreturn out;} 注意ostream为cout的类型定义ostream为返回类型是为了作为起到链接的效果如:
coutob1ob2endl;ostream返回out然后再次被后面所调用一直反复调用下去。 然而进行了运算符重载就能实现我们想要的效果了吗答案是不能见下图 造成这样的原因是什么呢还是类的封装问题私有的数据不能被外界所访问这时我们就需要用到友元函数来帮助我们实现了 于是我们将operator设置成友元
#define _CRT_SECURE_NO_WARNINGS 01
#include iostream
#includestring.husing namespace std;class Person{//设置成友元函数 在函数内 访问Person类中的所有数据friend ostream operator(ostream out, Person ob);private:char* name;int num;public:Person(char* name, int num){this- name new char[strlen(name) 1];strcpy(this- name, name);this- num num;cout 有参构造 endl;}//普通的成员函数void printPerson(void){cout name name , num num endl;}~Person(){if (this- name ! NULL){delete[] this- name;this- name NULL;}cout 析构函数 endl;}};ostream operator(ostream out, Person ob)//outcout, ob ob1{//重新实现 输出格式out ob.name , ob.num;//每次执行为 返回值得到coutreturn out;}int main(int argc, char* argv[]){char arr[] lucy;Person ob1(arr, 18);//普通的成员函数 遍历信息//ob1.printPerson();//cout默认输出方式 无法识别 自定义对象 输出格式coutob1endl;//errreturn 0;} 实现效果如下 运算符重载作为成员函数以及全局函数的实现以为例
全局函数 这里同上面的栗子大致一样不过多叙述
#include iostream#includestring.husing namespace std;class Person{//设置成友元函数 在函数内 访问Person类中的所有数据friend ostream operator(ostream out, Person ob);friend Person operator(Person ob1, Person ob2);private:char* name;int num;public:Person(){this- name NULL;this- num 0;cout 无参构造 endl;}Person(char* name, int num){this- name new char[strlen(name) 1];strcpy(this- name, name);this- num num;cout 有参构造 endl;}//普通的成员函数void printPerson(void){cout name name , num num endl;}~Person(){if (this- name ! NULL){delete[] this- name;this- name NULL;}cout 析构函数 endl;}};//全局函数作为友元 完成运算符重载ostream operator(ostream out, Person ob)//outcout, ob ob1{//重新实现 输出格式out ob.name , ob.num;//每次执行为 返回值得到coutreturn out;}//全局函数作为友元 完成运算符重载Person operator(Person ob1, Person ob2)//ob1 ob2{ //namename(字符串追加)char* tmp_name new char[strlen(ob1.name) strlen(ob2.name) 1];strcpy(tmp_name, ob1.name);strcat(tmp_name, ob2.name);//numnum数值相加int tmp_num ob1.num ob2.num;Person tmp(tmp_name, tmp_num);//释放tmp_name的空间if (tmp_name ! NULL){delete[] tmp_name;tmp_name NULL;}return tmp;}void test02(){char arr[] lucy;Person ob1(arr, 18); char arr2[] bob;Person ob2(arr2, 19);cout ob1 endl;cout ob2 endl;//Person ob3 operator(ob1,ob2);Person ob3 ob1 ob2;cout ob3 endl;}int main(int argc, char* argv[]){test02();return 0;}
成员函数
#include iostream#includestring.husing namespace std;class Person{ //设置成友元函数 在函数内 访问Person类中的所有数据friend ostream operator(ostream out, Person ob);private:char* name;int num;public:Person(){this- name NULL;this- num 0;cout 无参构造 endl;}Person(char* name, int num){this- name new char[strlen(name) 1];strcpy(this- name, name);this- num num;cout 有参构造 endl;}//成员函数 完成运算符重载 ob1用this代替 ob2用参数ob代替Person operator(Person ob){//this ob1//namename(字符串追加)char* tmp_name new char[strlen(this- name) strlen(ob.name) 1];strcpy(tmp_name, this- name);strcat(tmp_name, ob.name);//numnum数值相加int tmp_num this- num ob.num;Person tmp(tmp_name, tmp_num);//释放tmp_name的空间if (tmp_name ! NULL){delete[] tmp_name;tmp_name NULL;}return tmp;}//普通的成员函数void printPerson(void){cout name name , num num endl;}~Person(){if (this- name ! NULL){delete[] this- name;this- name NULL;}cout 析构函数 endl;}};//全局函数作为友元 完成运算符重载ostream operator(ostream out, Person ob)//outcout, ob ob1{//重新实现 输出格式out ob.name , ob.num;//每次执行为 返回值得到coutreturn out;}void test03(){char arr[] lucy;char arr2[] bob;Person ob1(arr, 18);Person ob2(arr2, 19);//Person ob3 ob1.operator(ob2);Person ob3 ob1 ob2;cout ob3 endl;}int main(int argc, char* argv[]){ test03();return 0;} 在运算符重载运算符时如果我们以成员函数的方式定义则可以直接访问类中的数据无需再使用友元函数来定义。因此我们在重载运算符时最好是以成员函数的方式重载 四、和--运算符重载 重要、常用 不知道大家有没有一个疑惑如果我们实现前置 、后置 以及前置- -、后置--运用operator时如何区分他们呢 此时我们又要提到一个概念当编译器看到a(前置)它就调用operator(a),当编译器看到a后置它就会去调用operator(a,int)。 - -也是同样的道理具体实现如下 具体实现用例
#include iostreamusing namespace std;class Data{friend ostream operator(ostream out, Data ob);private:int a;int b;public:Data(){cout 无参的构造函数 endl;a 0;b 0;}Data(int a, int b) :a(a), b(b){cout 有参构造 endl;//this‐a a;//this‐b b;}void showData(void){cout a a , b b endl;}~Data(){cout 析构函数函数 endl;}//成员函数 重载前置 ob1 (先加 后使用)//编译器 默认识别 operator(a) //但是a可以用this代替 从而化简 operator()Data operator()//ob1{ //先加a;//this‐a this‐a 1b;//this‐b this‐b 1//后使用return *this;}//成员函数 重载后置 ob1 (先使用 后加)//编译器 默认识别 operator(a,int) //但是a可以用this代替 从而化简 operator (int)Data operator(int)//ob1{//先使用(备份加之前的值)static Data old *this;//后加a;b;//返回备份值return old;}//重载前置‐‐ ‐‐ob3//编译器 默认识别 operator(a) //但是a可以用this代替 从而化简 operator‐‐()Data operator--(){//先减a--;b--;//后使用(返回)return *this;}//重载后‐‐ ob4‐‐//编译器 默认识别 operator(a,int) //但是a可以用this代替 从而化简 operator(int)Data operator--(int){//先使用static Data old *this;//再减a--;b--;return old;}};//普通全局函数 作为类的友元 重载运算符ostream operator(ostream out, Data ob){out a ob.a , b ob.b;return out;}void test01(){Data ob1(10, 20);ob1.showData();//重载直接输出自定义对象的值//operator(cout,ob1);cout ob1 endl;//成员函数 重载 运算符cout ob1 endl;Data ob2(10, 20);cout ob2 endl;cout ob2 endl;//成员函数 重载 ‐‐运算符Data ob3(10, 20);cout ob3 ob3 endl;cout --ob3 endl;Data ob4(10, 20);cout ob4 ob4 endl;cout ob4-- endl;cout ob4 ob4 endl; }int main(int argc, char* argv[]){test01();return 0;} 效果如下 感谢你耐心的看到这里ღ( ´ᴗ )比心如有哪里有错误请踢一脚作者o(╥﹏╥)o 给个三连再走嘛~