C++运算符重载代码分析

2024-09-04 22:32

本文主要是介绍C++运算符重载代码分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

0. 说明

通过示例代码,说明运算符重载的调用时机,以及printf的参数处理机制。

1. 示例代码

#include <cstdio>
#include <iostream>class Foo {
public:explicit Foo(int v): value(v) {}operator int() {std::cout << "int()" << std::endl;return value * 3;}Foo operator+(const Foo& other) {return Foo(value + other.value);}private:int value;
};int main()
{Foo foo2(2);Foo foo3(30);Foo foo5 = foo2 + foo3;printf("foo5: %d\n", foo5);printf("foo5: %d\n", (int)foo5);return 0;
}

2. 运行结果

foo5: 32
int()
foo5: 96

3. 分析

3.1 调用运算符重载的时机

通过运行结果,发现第一个print语句,并没有将Foo对象转化成int类型,即没有如预期的调用Foo的operator int(()函数;而第二个print语句因为加了强制类型转换,所以会调用operator int(),运行结果也是符合预期的。

现在分析为什么第一个print语句出现了异常。

3.2 显示规定类型转换

在此之前,我们先做另外的一些示例分析。新增一个函数,并验证:

void print(int n)
{printf("n: %d\n", n);
}int main()
{Foo foo2(2);Foo foo3(30);Foo foo5 = foo2 + foo3;printf("foo5: %d\n", foo5);printf("foo5: %d\n", (int)foo5);print(foo5);return 0;
}

运行结果:

foo5: 32
int()
foo5: 96
int()
n: 96

即因为print()的入参是int类型,所以print(foo5)的时候,会自动调用Foo.operator int()。

3.3 printf对变长参数的处理策略

通过这种对比,基本上确定是printf()函数在处理Foo对象的时候,并没有如预期调用Foo.operator int()。原因何在呢?因为printf没有类型检查,在格式化的时候,遇到了%d,它就把foo5变量看做整型变量,刚好该变量的地址处,可以解析出整数,即foo5对象的成员变量value的值。截图如下:


为了进一步验证,我们在Foo中添加其它一些数据成员。代码如下:

#include <cstdio>
#include <iostream>class Foo {
public:explicit Foo(int v): start(0x1234), value(v), end(0x5678) {}operator int() {std::cout << "int()" << std::endl;return value * 3;}Foo operator+(const Foo& other) {return Foo(value + other.value);}private:int start;int value;int end;
};void print(int n)
{printf("n: %d\n", n);
}int main()
{Foo foo2(2);Foo foo3(30);Foo foo5 = foo2 + foo3;printf("foo5: 0x%x\n", foo5);printf("foo5: %d\n", (int)foo5);print(foo5);return 0;
}

运行结果:

foo5: 0x1234
int()
foo5: 96
int()
n: 96

这个运行结果可以验证之前的分析(或猜想)。

3.4 运算符重载无关的验证示例

进一步地,我们撇开运算符重载,来观察printf对变长参数的检查机制。增加一个结构体的定义,并把结构体对象用%d进行打印。示例代码如下:

struct Point3D {int x;int y;int z;
};int main()
{Foo foo2(2);Foo foo3(30);Foo foo5 = foo2 + foo3;printf("foo5: 0x%x\n", foo5);printf("foo5: %d\n", (int)foo5);print(foo5);Point3D point = {3, 4, 5};printf("point: %d\n", point);return 0;
}

在VS2005下,编译无任何告警和错误。运行结果:

foo5: 0x1234
int()
foo5: 96
int()
n: 96
point: 3


4. Linux GCC

4.1 编译器对比

为了和VS2005对比,我们在Linux环境上做对比。可以看到这里有编译告警,说明比VS2005的编译器做得更好。——在VS2008上,同样没有warning。

flying-bird@flyingbird:~/examples/cpp/operator_test$ g++ --version
g++ (Ubuntu 4.8.2-19ubuntu1) 4.8.2
Copyright (C) 2013 Free Software Foundation, Inc.
This is free software; see the source for copying conditions.  There is NO
warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
flying-bird@flyingbird:~/examples/cpp/operator_test$ g++ test.cc
test.cc: In function ‘int main()’:
test.cc:40:32: warning: format ‘%x’ expects argument of type ‘unsigned int’, but argument 2 has type ‘Foo’ [-Wformat=]
printf("foo5: 0x%x\n", foo5);
^
test.cc:45:32: warning: format ‘%d’ expects argument of type ‘int’, but argument 2 has type ‘Point3D’ [-Wformat=]
printf("point: %d\n", point);
^
flying-bird@flyingbird:~/examples/cpp/operator_test$ ./a.out 
foo5: 0x1234
int()
foo5: 96
int()
n: 96
point: 3
flying-bird@flyingbird:~/examples/cpp/operator_test$ 

4.2 不要忽视编译器告警

谈到printf,我们顺便看一个案例,这是曾经所在的一个项目组中出现过的。当时造成的故障是设备反复重启,为了定位该问题,团队曾经花了不少时间。


代码如下:

#include <stdio.h>int main()
{int error_code = 1;printf("error code: %s\n", error_code);return 0;
}

这里简单的几行代码,是嵌入在当时所在项目(某个模块)数千行中的一部分。结合了本文的讨论,且代码只抽象出了这几行,所以还是容易看出问题所在。

编译、运行的情况:

flying-bird@flyingbird:~/examples/cpp/printf_warning$ gcc test.c
test.c: In function ‘main’:
test.c:7:5: warning: format ‘%s’ expects argument of type ‘char *’, but argument 2 has type ‘int’ [-Wformat=]printf("error code: %s\n", error_code);^
flying-bird@flyingbird:~/examples/cpp/printf_warning$ ./a.out 
Segmentation fault
flying-bird@flyingbird:~/examples/cpp/printf_warning$ 

运行这个小程序是段错误,当当时在设备上体现的就是反复重启。

另外可以看到,在Linux上面是有warning的,指出了错误所在。因为当初该代码在其他的代码中间,且遗留代码没有清空warning,然后新加入的代码也没有去关注warning,所以问题就这么流到后续环节。

这里体现的一个要点就是:千万别忽视编译告警。

作为对比,作者之前曾经经历过另外一个C++跨平台项目,当时的代码规模是5~6万行(NBNC)的规模,因为对代码质量非常关注,所以从项目一开始就构建了完善的CI环境,包括TDD、Code/Function Coverage等度量,另外还包括的一项就是实时清空warning。只要编译出现了warning,就会在CI中体现出现,自然也会在第一时间处理掉,低级问题想要漏到下一个环节是非常困难的。

当时这个项目(C++部分)有10人左右,实现的是真敏捷(包括很多公司没有实现的Pair Programming我们开展得也很好),每天每个人都是非常频繁地代码上库,且从未出现重大功能问题。

5. 其它

关于class对象的布局,可以参考《深入探索C++对象模型》,其第一章《关于对象》就有基本的介绍。



这篇关于C++运算符重载代码分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

性能分析之MySQL索引实战案例

文章目录 一、前言二、准备三、MySQL索引优化四、MySQL 索引知识回顾五、总结 一、前言 在上一讲性能工具之 JProfiler 简单登录案例分析实战中已经发现SQL没有建立索引问题,本文将一起从代码层去分析为什么没有建立索引? 开源ERP项目地址:https://gitee.com/jishenghua/JSH_ERP 二、准备 打开IDEA找到登录请求资源路径位置

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

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

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

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

活用c4d官方开发文档查询代码

当你问AI助手比如豆包,如何用python禁止掉xpresso标签时候,它会提示到 这时候要用到两个东西。https://developers.maxon.net/论坛搜索和开发文档 比如这里我就在官方找到正确的id描述 然后我就把参数标签换过来

06 C++Lambda表达式

lambda表达式的定义 没有显式模版形参的lambda表达式 [捕获] 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 有显式模版形参的lambda表达式 [捕获] <模版形参> 模版约束 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 含义 捕获:包含零个或者多个捕获符的逗号分隔列表 模板形参:用于泛型lambda提供个模板形参的名

poj 1258 Agri-Net(最小生成树模板代码)

感觉用这题来当模板更适合。 题意就是给你邻接矩阵求最小生成树啦。~ prim代码:效率很高。172k...0ms。 #include<stdio.h>#include<algorithm>using namespace std;const int MaxN = 101;const int INF = 0x3f3f3f3f;int g[MaxN][MaxN];int n

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)

计算机毕业设计 大学志愿填报系统 Java+SpringBoot+Vue 前后端分离 文档报告 代码讲解 安装调试

🍊作者:计算机编程-吉哥 🍊简介:专业从事JavaWeb程序开发,微信小程序开发,定制化项目、 源码、代码讲解、文档撰写、ppt制作。做自己喜欢的事,生活就是快乐的。 🍊心愿:点赞 👍 收藏 ⭐评论 📝 🍅 文末获取源码联系 👇🏻 精彩专栏推荐订阅 👇🏻 不然下次找不到哟~Java毕业设计项目~热门选题推荐《1000套》 目录 1.技术选型 2.开发工具 3.功能