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

2023-12-07 04:36

本文主要是介绍条款24:若所有参数都需类型转换,请为此采用non-member函数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.前言

令classes支持隐式类型转换通常是个糟糕的想法,当然这条规则有其例外,最常见的例外是在建立数值类型时候,假如你设计一个class用来表现有理数,允许整数“隐式转换”为有理数似乎颇为合理。

2.实例分析

假设这样开始你的Rational class:

class Rational{public:Rational(int numerator=0,int denominator=1);//构造函数刻意不为explict,允许int-to-        Rational隐式转换int numerator() const;//分子访问函数int denominator() const;//分母的访问函数private:....
};

在这里,你想支持算数运算,比如加法,乘法等等。但你不确定是否该由member函数,non-member函数,或者由non-member friend函数来实现 。

首先研究一下将operator*写成Rational成员函数的写法:

class Rational{public:...const Rational operator*(const Rational& rhs) const;
};

这个设计能够让你能够将两个有理数以最轻松自在的方式相乘:

Rational oneEighth(1,8);
Rational oneHalf(1,2);
Rational result=oneHalf*oneEighth;//很好
result=result*oneEighth;//很好

如果还想要更多的方式,希望能够支持混合式运算,也就是拿Rational和int相乘。然而当尝试混合式算术时,发现只有一半行的通:

result=oneHalf*2;//很好
result=2*oneHalf;//错误

这不是个好兆头,乘法应该满足交换律。

当你以对应的函数形式重写上述两个式子,问题便一目了然了:

result=oneHal;f.operator*(2);//很好
result=2.operator*(oneHalf);//错误

是的,oneHalf是一个内含operator*函数的class对象,所以编译器调用该函数。然而整数2并没有相应的class,也就没有operator*成员函数。编译器也会尝试寻找可以被以下这般调用的non-member operator*(即在命名空间内或者在global作用域内):

result=operator*(2,oneHalf);//错误

但本例并不存在这样一个接受int和Rational作为参数的non-member operator*,因此查找失败。

再分析下先前成功的调用。注意其第二个参数是整数2,但Rational::operator*需要的实参却是个Rational对象,这里发生了什么是事情呢?

答案是这里发生了所谓的隐式转换。编译器知道你正在传递一个int,而函数需要的是Rationl;但它也知道只要调用Rational构造函数并赋予你所提供的int,就可以变出一个适当的Rational来,于是它就这样做了。换句话说此一调用动作在编译器眼里又点像这样:

const Rational temp(2);//根据2建立一个暂时性的Rational对象
result=oneHalf*temp;//等同于oneHalf.operator*(temp);

当然,只因为涉及non-explict构造函数,编译器才会这样做,如果Rational构造函数是explict,以下语句没有一个可以通过编译:

result=oneHalf*2;//错误,在explict构造函数下,无法将2转换为一个Rational
result=2*oneHalf;//同样的错误

这就很难让Rational class支持混合式算术运算了。

然而我的目标不仅在一致性,也要支持混合式算术运算,也就是希望能有个设计能让以上语句通过编译。我们再回到上述两个语句,为什么即使Rational构造函数不是explict,仍然只有一俄国可以通过编译,另一个确不可以:

result=oneHalf*2;//正确(在non-explict构造函数的情况下)
result=2*oneHalf;//错误

答案是:只有当参数列被列于参数列内,这个参数才是隐式类型转换的合格参与者,地位相当于“被调用之成员函数所隶属的那个对象”-即this对象的那个隐喻参数,绝不是隐式转换的合格参与者。这就是为什么上述第一次调用可以通过编译,第二次调用则不能,因为第一次调用伴随一个放在参数列内的参数,第二次调用则没有。

然而你一定也会想要支持混合式算术运算:让operator*成为一个non-member函数,允许编译器在每一个实参身上执行隐式类型转换:

class Rational{....
};
const Rational operator*(const Rational& lhs,const Rational& rhs)//现在做成一个non-           
{return Rational(lhs.numerator()*rhs.numerator(),lhs.denominator()*rhs.denominator());
}
//member函数
Ratioanl oneFourth(1,4);
Rational result;
result=oneFourth*2;//没问题
result=2*oneFourth;//编译通过

不过还有一点需要担心:即operator*是否应该成为Rational class的一个friend函数呢?

就本例而言答案是否定的,因为operator*可以完全由Rational的public接口完成任务,上面代码已经表明此种做法。这导出一个重要的观察:member函数的反面是non-member函数,不是friend函数。

3.总结

如果你需要为某个函数的所有参数(包括被this指针所指的那个隐喻参数)进行类型转换,那么这个函数必须是个non-member.

这篇关于条款24:若所有参数都需类型转换,请为此采用non-member函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/464502

相关文章

Andrej Karpathy最新采访:认知核心模型10亿参数就够了,AI会打破教育不公的僵局

夕小瑶科技说 原创  作者 | 海野 AI圈子的红人,AI大神Andrej Karpathy,曾是OpenAI联合创始人之一,特斯拉AI总监。上一次的动态是官宣创办一家名为 Eureka Labs 的人工智能+教育公司 ,宣布将长期致力于AI原生教育。 近日,Andrej Karpathy接受了No Priors(投资博客)的采访,与硅谷知名投资人 Sara Guo 和 Elad G

hdu1171(母函数或多重背包)

题意:把物品分成两份,使得价值最接近 可以用背包,或者是母函数来解,母函数(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v) 其中指数为价值,每一项的数目为(该物品数+1)个 代码如下: #include<iostream>#include<algorithm>

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

如何在页面调用utility bar并传递参数至lwc组件

1.在app的utility item中添加lwc组件: 2.调用utility bar api的方式有两种: 方法一,通过lwc调用: import {LightningElement,api ,wire } from 'lwc';import { publish, MessageContext } from 'lightning/messageService';import Ca

4B参数秒杀GPT-3.5:MiniCPM 3.0惊艳登场!

​ 面壁智能 在 AI 的世界里,总有那么几个时刻让人惊叹不已。面壁智能推出的 MiniCPM 3.0,这个仅有4B参数的"小钢炮",正在以惊人的实力挑战着 GPT-3.5 这个曾经的AI巨人。 MiniCPM 3.0 MiniCPM 3.0 MiniCPM 3.0 目前的主要功能有: 长上下文功能:原生支持 32k 上下文长度,性能完美。我们引入了

【C++高阶】C++类型转换全攻略:深入理解并高效应用

📝个人主页🌹:Eternity._ ⏩收录专栏⏪:C++ “ 登神长阶 ” 🤡往期回顾🤡:C++ 智能指针 🌹🌹期待您的关注 🌹🌹 ❀C++的类型转换 📒1. C语言中的类型转换📚2. C++强制类型转换⛰️static_cast🌞reinterpret_cast⭐const_cast🍁dynamic_cast 📜3. C++强制类型转换的原因📝

C++操作符重载实例(独立函数)

C++操作符重载实例,我们把坐标值CVector的加法进行重载,计算c3=c1+c2时,也就是计算x3=x1+x2,y3=y1+y2,今天我们以独立函数的方式重载操作符+(加号),以下是C++代码: c1802.cpp源代码: D:\YcjWork\CppTour>vim c1802.cpp #include <iostream>using namespace std;/*** 以独立函数

AI(文生语音)-TTS 技术线路探索学习:从拼接式参数化方法到Tacotron端到端输出

AI(文生语音)-TTS 技术线路探索学习:从拼接式参数化方法到Tacotron端到端输出 在数字化时代,文本到语音(Text-to-Speech, TTS)技术已成为人机交互的关键桥梁,无论是为视障人士提供辅助阅读,还是为智能助手注入声音的灵魂,TTS 技术都扮演着至关重要的角色。从最初的拼接式方法到参数化技术,再到现今的深度学习解决方案,TTS 技术经历了一段长足的进步。这篇文章将带您穿越时

函数式编程思想

我们经常会用到各种各样的编程思想,例如面向过程、面向对象。不过笔者在该博客简单介绍一下函数式编程思想. 如果对函数式编程思想进行概括,就是f(x) = na(x) , y=uf(x)…至于其他的编程思想,可能是y=a(x)+b(x)+c(x)…,也有可能是y=f(x)=f(x)/a + f(x)/b+f(x)/c… 面向过程的指令式编程 面向过程,简单理解就是y=a(x)+b(x)+c(x)

如何确定 Go 语言中 HTTP 连接池的最佳参数?

确定 Go 语言中 HTTP 连接池的最佳参数可以通过以下几种方式: 一、分析应用场景和需求 并发请求量: 确定应用程序在特定时间段内可能同时发起的 HTTP 请求数量。如果并发请求量很高,需要设置较大的连接池参数以满足需求。例如,对于一个高并发的 Web 服务,可能同时有数百个请求在处理,此时需要较大的连接池大小。可以通过压力测试工具模拟高并发场景,观察系统在不同并发请求下的性能表现,从而