Effective C++条款24——若所有参数皆需类型转换,请为此采用non-member涵数(设计与声明)
我在导读中提过,令classes支持隐式类型转换通常是个糟糕的主意。当然这条规则有其例外,最常见的例外是在建立数值类型时。假设你设计一个class用来表现有理数,允许整数“隐式转换”为有理数似乎颇为合理。的确,它并不比C++内置从int至double的转换来得不合理,而还比C++内置从double至int的转换来得合理些。假设你这样开始你的Rational class:
class Rational {
public:// 构造函数刻意不为explicit// 允许int-to-Rational隐士转换Rational(int numerator = 0,int denominator = 1);// 分子和分母的访问函数int numerator() const;int denominator() const;private:// ...
};
你想支持算术运算诸如加法、乘法等等,但你不确定是否该由member函数、
non-member函数,或可能的话由non-member friend函数来实现它们。你的直觉告诉你,当你犹豫就该保持面向对象精神。你知道有理数相乘和 Rational class有关,因此很自然地似乎该在Rational class内为有理数实现operator*。条款23曾经反直觉地主张,将函数放进相关class内有时会与面向对象守则发生矛盾,但让我们先把那放在一旁,先研究一下将operator*写成Rational成员函数的写法:
class Rational {
public:// ...const Rational operator* (const Rational& rhs) const;
};
(如果你不确定为什么这个函数被声明为此种形式,也就是为什么它返回一个const by-value结果但接受一个reference-to-const实参,请参考条款3,20和21。)
这个设计使你能够将两个有理数以最轻松自在的方式相乘:
Rational oneEighth(1, 8);
Rational oneHalf(1, 2);
Rational result = oneEighth * oneHalf; // 很好
result = result * oneEighth; // 很好
但你还不满足。你希望支持混合式运算,也就是拿Rationals和……嗯……例如ints相乘。毕竟很少有什么东西会比两个数值相乘更自然的了——即使是两个不同类型的数值。
然而当你尝试混合式算术,你发现只有一半行得通:
result = oneHalf * 2; // 很好
result = 2 * oneHalf; // 错误
这不是好兆头。乘法应该满足交换律,不是吗?
当你以对应的函数形式重写上述两个式子,问题所在便一目了然了:
result = oneHalf.operator*(2); // 很好
result = 2.opeator*(oneHalf); // 错误
是的,oneHalf是一个内含operator*函数的 class的对象,所以编译器调用该函数。然而整数2并没有相应的class,也就没有operator*成员函数。编译器也会尝试寻找可被以下这般调用的non-member operator*(也就是在命名空间内或在global作用域内):
result = operator*(2, oneHalf); // 错误
但本例并不存在这样一个接受int和Rational作为参数的non-member operator*,因此查找失败。
再次看看先前成功的那个调用。注意其第二参数是整数2,但Rational::operator*需要的实参却是个Rational对象。这里发生了什么事?为什么2在这里可被接受,在另一个调用中却不被接受?
因为这里发生了所谓隐式类型转换(implicit type conversion)。编译器知道你正在传递一个int,而函数需要的是Rational;但它也知道只要调用Rational构造函数并赋予你所提供的int,就可以变出一个适当的Rational来。于是它就那样做了。换句话说此一调用动作在编译器眼中有点像这样:
const Rational temp(2); // 根据2建立一个暂时性的Rational对象
result = oneHalf * temp; // 很好
当然,只因为涉及non-explicit构造函数,编译器才会这样做。如果Rational构造函数是explicit,以下语句没有一个可通过编译;
result = oneHalf * 2; // 错误
result = 2 * oneHalf; // 错误
这就很难让 Rational class支持混合式算术运算了,不过至少上述两个句子的行为从此一致。
然而你的目标不仅在一致性,也要支持混合式算术运算,也就是希望有个设计能让以上语句通过编译。这把我们带回到上述两个语句,为什么即使Rational构造函数不是explicit,仍然只有一个可通过编译,另一个不可以:
result = oneHalf * 2; // 没问题,在non-expplicit下
result = 2 * oneHalf; // 错误
结论是,只有当参数被列于参数列(parameter list)内,这个参数才是隐式类型转换的合格参与者。地位相当于“被调用之成员函数所隶属的那个对象”——即this对象—一的那个隐喻参数,绝不是隐式转换的合格参与者。这就是为什么上述第一次调用可通过编译,第二次调用则否,因为第一次调用伴随一个放在参数列内的参数,第二次调用则否。
然而你一定也会想要支持混合式算术运算。可行之道终于拨云见日:让operator*成为一个non-member函数,俾允许编译器在每一个实参身上执行隐式类型转换:
class Rational {// ... // 不包括operator*
};// 非成员函数
const Rational operator* (const Rational& lhs,const Rational& rhs)
{return Ratioanl(lhs.numerator() * rhs.numerator(),lhs.denominator() * rhs.denominator());
}Rational oneFourth(1, 4);
Ratioanl result;
result = oneFourth * 2; // 没问题
result = 2 * oneFourth; // 没问题
这当然是个快乐的结局,不过还有一点必须操心: operator*是否应该成为Rational class的一个friend函数呢?
就本例而言答案是否定的,因为operator*可以完全糟由Rational的public接口完成任务,上面代码已表明此种做法。这导出一个重要的观察: member函数的反面是non-member函数,不是friend函数。太多C++程序员假设,如果一个“与某class相关”的函数不该成为一个member (也许由于其所有实参都需要类型转换,例如先前的Rational的operator*函数),就该是个friend。本例表明这样的理由过于牵强。无论何时如果你可以避免friend函数就该避免,因为就像真实世界一样,朋友带来的麻烦往往多过其价值。当然有时候friend有其正当性,但这个事实依然存在:不能够只因函数不该成为member,就自动让它成为friend。
本条款内含真理,但却不是全部的真理。当你从 Object-Oriented C++跨进Template C++(见条款1)并让 Rational成为一个class template而非class,又有些需要考虑的新争议、新解法、以及一些令人惊讶的设计牵连。这些争议、解法和设计牵连形成了条款46。
请记住
- 如果你需要为某个函数的所有参数(包括被this 指针所指的那个隐喻参数)进行类型转换,那么这个函数必须是个non-member。
相关文章:
Effective C++条款24——若所有参数皆需类型转换,请为此采用non-member涵数(设计与声明)
我在导读中提过,令classes支持隐式类型转换通常是个糟糕的主意。当然这条规则有其例外,最常见的例外是在建立数值类型时。假设你设计一个class用来表现有理数,允许整数“隐式转换”为有理数似乎颇为合理。的确,它并不比C内置从int…...
决策工具箱:战略分析必备工具与框架
跟随时代的步伐,企业战略也在不断演化。无论是初创企业还是知名企业,都需要有效的战略工具来指导其业务发展。探索这些必备工具,并学习如何最大限度地利用它们,是企业的一个学习目标。 战略分析工具和框架有很多,其中…...
【压力测试指南】没有任何文档,小白也可以做的压力测试
前言 一般在执行压力测试之前,会由开发提供出接口文档,包含一些接口的详细参数,便于测试工程师编写测试脚本。但在某些情况下,接口等相关文档缺失,那作为Tester,我们该如何顺利的实施压力测试呢࿱…...
Linux: memory: memblock: debug
文章目录 代码memblock=debug实例log代码 将内存相关的一些日志打开: static int __init early_memblock(char *p) {if (p && strstr(p, "debug")...
搬家快递服务小程序的便利性
在当今快节奏的生活中,搬家可能是很多人都需要面对的问题。无论是新房子还是新办公室,都需要高效、便捷的搬家服务。本文将介绍如何使用第三方小程序制作平台,如乔拓云平台,开发一款高效便捷的搬家服务小程序。 1. 注册登录第三方…...
软件架构师 Debugging
软件架构师 Debugging 目录概述需求: 设计思路实现思路分析 参考资料和推荐阅读 Survive by day and develop by night. talk for import biz , show your perfect code,full busy,skip hardness,make a better result,wait for change,challenge Survi…...
7.1 项目1 学生通讯录管理:文本文件增删改查(C++版本)(自顶向下设计+断点调试) (A)
C自学精简教程 目录(必读) 作业目标: 这个作业中,你需要综合运用之前文章中的知识,来解决一个相对完整的应用程序。 作业描述: 1 在这个作业中你需要在文本文件中存储学生通讯录的信息,并在程序启动的时候加载这些…...
学习使用php判断阿里云oss图片单图或批量上传、查询图片文件是否存在
学习使用php判断阿里云oss图片单图或批量上传、查询图片文件是否存在 doesObjectExist doesObjectExist 主要函数doesObjectExist /*** Base64上传文件* param string|array $images* param string $model_path* param string $model_type* param string $upload_path* param…...
重磅| Falcon 180B 正式在 Hugging Face Hub 上发布!
引言 我们很高兴地宣布由 Technology Innovation Institute (TII) 训练的开源大模型 Falcon 180B 登陆 Hugging Face! Falcon 180B 为开源大模型树立了全新的标杆。作为当前最大的开源大模型,有180B 参数并且是在在 3.5 万亿 token 的 TII RefinedWeb 数…...
Linux命令行
目录 CLI GUI 命令行界面 图形界面 命令行提示符 # $ 编辑 命令一般由三个部分组成 历史命令,使用上下键,或者使用history,ctrlr搜索历史命令 通配符 *,? 切换用户 su 作业管理 &,jobs,bg,fg CLI GUI 命令行界面 …...
[持续更新]计算机经典面试题基础篇Day1
[通用]计算机经典面试题基础篇Day1 1、jvm的组成 类加载器(Class Loader):负责将编译后的Java类加载到JVM中,并在运行时动态加载所需的类。运行时数据区(Runtime Data Area):是JVM的内存管理区…...
ProcessWindowFunction 结合自定义触发器的陷阱
背景: flink中常见的需求如下:统计某个页面一天内的点击率,每10秒输出一次,我们如果采用ProcessWindowFunction 结合自定义触发器如何实现呢?如果这样实现问题是什么呢? ProcessWindowFunction 结合自定义触发器实现…...
什么是jvm
一、初识JVM(虚拟机) JVM是Java Virtual Machine(Java虚拟机)的缩写,JVM是一种用于计算设备的规范,它是一个虚构出来的计算机,是通过在实际的计算机上仿真模拟各种计算机功能来实现的。 引入Jav…...
kettle通过java步骤获取汉字首拼
kettle通过java步骤获取汉字首拼 用途描述 一组数据,需要获取汉字首拼后,输出; 实现效果 添加jar包 pinyin4j-2.5.0.jar 自定义常量数据 Java代码 完整代码: import net.sourceforge.pinyin4j.PinyinHelper; import net.sou…...
Conformer: Local Features Coupling Global Representationsfor Visual Recognition
论文链接:https://arxiv.org/abs/2105.03889 代码链接:https://github.com/pengzhiliang/Conformer 参考博文:Conformer论文以及代码解析(上)_conformer代码_从现在开始壹并超的博客-CSDN博客 摘要 在卷积神经网络…...
java8-Stream流常用API
什么是 Stream Stream(流)是 Java 8 引入的一个新的抽象概念,它代表着一种处理数据的序列。简单来说,Stream 是一系列元素的集合,这些元素可以是集合、数组、I/O 资源或者其他数据源。 Stream API 提供了丰富的操作方…...
React 任务调度
React 任务池 不同的fiber任务有不同的优先级,为了用户体验,React需要先处理优先级高的任务。 为了存储这些任务,React中有两个任务池: // Tasks are stored on a min heap var taskQueue []; // 存储立即要执行的任务 var tim…...
小白开始学习C++
第一节:控制台输出hello word! #include<iostream> //引入库文件 int main() { //控制台输出 hello word! 之后回车 std::cout << "hello word!\n"; #include<iostream> //引入库文件int main() {//控制…...
SpringMVC入门的注解、参数传递、返回值和页面跳转---超详细教学
前言: 欢迎阅读Spring MVC入门必读!在这篇文章中,我们将探索这个令人兴奋的框架,它为您提供了一种高效、灵活且易于维护的方式来构建Web应用程序。通过使用Spring MVC,您将享受到以下好处:简洁的代码、强大…...
【复习socket】每天40min,我们一起用70天稳扎稳打学完《JavaEE初阶》——28/70 第二十八天
专注 效率 记忆 预习 笔记 复习 做题 欢迎观看我的博客,如有问题交流,欢迎评论区留言,一定尽快回复!(大家可以去看我的专栏,是所有文章的目录) 文章字体风格: 红色文字表示:重难点★✔ 蓝色文字表示:思路以及想法★✔ 如果大家觉得有帮助的话,感谢大家帮忙 点…...
Realistic Vision V5.1镜像免配置部署教程:Docker+本地模型路径自动校验
Realistic Vision V5.1镜像免配置部署教程:Docker本地模型路径自动校验 1. 项目概述 Realistic Vision V5.1虚拟摄影棚是基于Stable Diffusion 1.5生态顶级写实模型开发的本地化工具,专为追求摄影级人像效果的用户设计。这个解决方案通过Docker容器化技…...
自动送料装车系统PLC控制的设计——24页
自动送料装车系统作为工业自动化领域的关键环节,其核心作用在于通过PLC(可编程逻辑控制器)实现物料输送、定位、装载等流程的精准控制。传统人工操作易受疲劳、环境等因素影响,导致效率波动与安全隐患。而PLC控制通过预设逻辑程序…...
如何3步上手语音转换:Retrieval-based Voice-Conversion-WebUI完整实战指南
如何3步上手语音转换:Retrieval-based Voice-Conversion-WebUI完整实战指南 【免费下载链接】Retrieval-based-Voice-Conversion-WebUI 语音数据小于等于10分钟也可以用来训练一个优秀的变声模型! 项目地址: https://gitcode.com/GitHub_Trending/re/R…...
Java并发包中锁机制的底层实现原理剖析
实现java并发包中的锁机制底层主要有两种方式:1.基于jvm的monitor机制和对象头中的mark,synchronized关键字 word实现并通过锁升级(偏向锁→轻量级锁→重量级锁)优化性能;2.java.util.concurrent.locks包中的锁基于abstractquedsynchronizer&…...
GD32F4系列替换STM32F4,HAL库CAN初始化卡死?一个Sleep模式的坑与填坑实录
GD32F4替换STM32F4的CAN初始化陷阱:Sleep模式差异与实战解决方案 最近在将STM32F4项目迁移到GD32F4平台时,遇到了一个令人费解的问题——CAN总线初始化卡死在HAL_CAN_Init()函数中。经过深入排查,发现问题根源在于两款芯片CAN控制器Sleep模式…...
RTKLIB源码解析(五)数据流融合:RINEX、RTCM、NMEA与接收机原始数据的协同处理
1. 多源GNSS数据流融合的核心挑战 在RTKLIB的实际应用中,处理来自不同数据源的GNSS观测数据时,开发者常会遇到三个关键问题:格式差异、时间基准不统一和数据质量参差不齐。以RINEX、RTCM、NMEA和接收机原始数据为例,这些数据源的…...
CANopen协议学习与实践干货分享
CANopen协议代码,学习资料,包含CANfestival官方代码框架,官方字典生成工具,可自主设定心跳,pdo,sdo等内容参数,并包含已经移植完成的且带有详细注释的一个主站程序两个从站能正常通信࿰…...
避坑指南:RuoYi-Vue2集成Flowable 6.7.2时,关于database-schema-update和nullCatalogMeansCurrent的配置详解
深度解析:RuoYi-Vue2集成Flowable 6.7.2的数据库配置陷阱与实战策略 当企业级应用需要引入工作流引擎时,Flowable因其轻量化和高性能成为许多开发团队的首选。然而在RuoYi-Vue2框架中集成Flowable 6.7.2版本时,数据库配置环节往往成为开发者的…...
CSS 嵌套语法最佳实践:从入门到精通的完整指南
CSS 嵌套语法最佳实践:从入门到精通的完整指南 CSS 是流动的韵律,JS 是叙事的节奏。而 CSS 嵌套,是让这份韵律更加优雅、结构更加清晰的魔法。 一、CSS 嵌套:现代样式表的革命 CSS 嵌套(Nesting)是 CSS 原…...
暗黑破坏神2存档编辑器完全指南:从技术原理到实战应用
暗黑破坏神2存档编辑器完全指南:从技术原理到实战应用 【免费下载链接】d2s-editor 项目地址: https://gitcode.com/gh_mirrors/d2/d2s-editor 价值定位:为什么d2s-editor能重塑你的游戏体验 你是否曾因反复刷不到心仪装备而失去耐心࿱…...
