需要类型转换时定义为非成员函数

需要类型转换时定义为非成员函数

前言:

因为自己查阅文章时经常遇到,这些问题:因为环境不同而导致结果不一致、因为文章跳跃度太大而导致无法理解、因为文章代码运行结果不同而苦恼。

所以为了看我文章的人不会遭遇同样的问题,

以后我的文章将遵循以下原则:

1.会在文章开头标明相关配置与环境

2.尽可能循序渐进(还是看不懂可能是我没有这样的天赋),标出阅读的前提知识

3.列出的代码自己先跑一遍


////////////////////////////////////////

语言:c++


前提知识:

1.类

2.函数重载

3.类型转换


集成开发环境:Visual Studio 2017

////////////////////////////////////////


条款24:若为所有参数皆需类型转换,请为此采用non-member函数

class Num

{

public:

         Num(intN=0,int M=1):n(N),M(M){}//不设explicit,提供隐式转换

         constNum operator*(const Num& rhs)const

         {

                  returnNum(rhs.getN()*n+rhs.getM()*M);

         }

         constint getN()const{return n;}

         constint getM()const{return m;}

private:

         intn;

         intm;

};

int main()

{

         Numnum1(2,3);

         Numnum2(3,4);

         Numnum3=num1*num2;      //成功

         num3=num3*num1;                //成功

         num3=num3*2;               //成功

         num3=2*num1;               //失败

         return0;

}


为何num3=num3*2成功而num3=2*num1却失败

因为num3=num3*2中,函数operator=由num3调用且参数为2,参数2为int可通过构造函数隐式转换为Num,所以成功

而num3=num3*2中,虽然想调用operator=但,这个函数却不属于int,这个基础类型int根本不拥有函数。

使用函数形式重写上面两个例子:

num3=num3.operator*(2);

num3=2.operator*(num3);             //明显错误


上述情况下,编译器会尝试寻找参数为(int,Num)的operator*的非成员函数。

所以只要写一个参数为(int,Num)的operator*的非成员函数即可,这就是我们一般重载乘法运算符时的做法————将重载乘法运算符声明为friend函数,即非成员函数的operator*可以调用类中的private成员进行运算。

friend     constNum operator*(const Num& lhs,const Num& rhs)const

         {

                  returnNum(rhs.getN()*lhs.getN()+rhs.getM()*lhs.getN());

                  //虽然可以直接使用m和n,但这是个好习惯,保持封装

         }

因为声明为友元函数,不再是成员函数,成员函数隐含一个参数this指向调用方,所以友元函数比成员函数多一个参数。

         num3=2*num1;               //成功

         <=>num3=operator*(Num(2),num1);

该条款的重点在于,只有参数位于参数列中,这个参数才有可能被隐式类型转换。


条款46:需要类型转换时请为模板定义非成员函数

条款24的升级版?可能这样说有点不妥,实质上就是把条跨23中的类模板化了,但是一旦模板化了事情就会变得复杂,因为编译器的处理策略不同了。

template

class Num

{

public:

         Num(T&N=0,T& M=1);//不设explicit,提供隐式转换

         constNum operator*(const Num& rhs)const

         {

                  returnNum(rhs.getN()*n+rhs.getM()*M);

         }

         constT getN()const{return n;}

         constT getM()const{return m;}

private:

         Tn;

         Tm;

};

Num num;

num=num*2;           //失败

上面这个例子跟条款24一致,但是却失败了,因为编译器并不清楚我们要使用哪一个函数,我们想要使用operator(Num,Num)函数,但是编译器必须知道T的类型才能具体化出这个函数,但是它不知道。

在第一参数num时编译器可以推导出T为int,而在第二参数2,它是个int,那么编译器从这个int中就推不出T究竟是个啥。在我们的预想中,我们期望编译器能够把int隐式转换为Num,但是编译器不会这么做,因为推导模板实参时并不考虑通过构造函数产生的隐式转换。

那么怎么办呢,这里有个有趣的办法,模板类中友元声明可以指定特定函数,也就是说只要类被具体化,这个函数就会被同时具体化,就不用在依赖实参推导来具体化函数。即是模板类友元函数,不再是个函数模板而是个模板函数。(前者只是个模具无法直接使用,而后者已经具体化可以直接使用)

虽然上述可以通过编译器,但是却无法链接因为我们没有实现它,我们在类中声明了这个友元函数,期待在类外实现它,但是不行因为这个声明是Num内的所作所为,我们无法在类外实现他,因为我们不能提前指定T并让连接器连接到它,那么就必须在Num中实现它,在声明式中实现它,让它成为一个内联函数。

我们并不使用友元可以访问类中私有成员的特性,而是利用其指定特定函数。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,837评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,551评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,417评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,448评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,524评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,554评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,569评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,316评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,766评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,077评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,240评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,912评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,560评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,176评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,425评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,114评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,114评论 2 352

推荐阅读更多精彩内容

  • C++运算符重载-下篇 本章内容:1. 运算符重载的概述2. 重载算术运算符3. 重载按位运算符和二元逻辑运算符4...
    Haley_2013阅读 1,438评论 0 49
  • 再读高效c++,颇有收获,现将高效c++中的经典分享如下,希望对你有所帮助。 1、尽量以const \enum\i...
    橙小汁阅读 1,216评论 0 1
  • 效果预览: 分析过程: 鼠标指针进入图片时,放大图片并且遮罩变为完全透明; 鼠标指针离开图片时,图片恢复原有尺寸并...
    黄啊码阅读 2,276评论 0 0
  • 以前看小说常有这样的桥段:一伙强人在山背后窥伺,看见有大队车马经过,观察到车轮压进路面很深就知道大有油水,于是一声...
    亦农阅读 367评论 2 4