跟我学C++中级篇——explicit的分析

2024-08-31 02:04

本文主要是介绍跟我学C++中级篇——explicit的分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、C++中的类型转换

在学习C++(包括C语言)的过程中,有一个细节非常容易被开发者忽略,就是有一些类型编译器可以自做主张的进行转换。最简单的就是short,int,long,char这些基本类型,编译器会假定开发者就是想转换,所以它会自动转一下。这个在前面分析过有符号和无符号类型时,跳过大坑儿。不过随着开发工具和编译器的升级,这种一般会提供显示的警告。
而在一些类对象中,也存在着这种问题,比如一些常见的库类,会默认对一些对象进行自动的转换,而这种自动转换,往往不是开发者需要的或者说没有考虑到的。那么其行为和后果,是一种不可预知的状态。而这恰恰是非常危险的,不是开发者想看到的。

二、explicit的目的

正如反复强调的,解决某种问题并不需要一蹴而就,一步步解决就行。而explicit关键字就是这么一种方法 ,先搞定一些想通过构造函数进行隐式转换的行为。看下面的代码:

#include <iostream>
#include <string>
using namespace std;
class Data {
public:/×explicit×/ Data(int d) {d_ = d;std::cerr << "set value:" << d << std::endl;}private:int d_ = 0;
};
int main() {Data d = 10;return 0;
}

在上面的代码中,程序会正常的执行,类对象可以使用一个普通的整形值进行赋值。这就是刚刚提到的隐式转换。但是如果在构造函数前增加explicit关键字,那么编译器会报一个编译错误:
“error: conversion from ‘int’ to non-scalar type ‘Data’ requested”,同时指出代码的位置。
和explicit相对应的是implicit。这样看就容易理解explicit这个关键字了,它就是显示的指定,这样也就容易明白它的用法了,想使用它限定的函数(主要就是构造函数,当然有些情况下也可以使用其它一些函数),就必须明确的调用它,而不是让编译器自做主张。说得更直白一些,就是不要约定俗成,必须明白的告诉编译器开发者想怎么做。

三、explicit注意点

使用explicit关键字,有几个需要注意的地方,在开发者经常会被忽略。
1、它只对一个参数的构造函数有意义
这种一个参数,既包括真正只有一个参数的构造函数,也包括其它形式的如虽然有多个参数但其它参数都有默认值的。因为拥有多个参数的构造函数是无法进行自动转换的。所以在一些编程风格的要求中,一个参数的构造函数,都必须显示使用explicit关键字。
2、防止隐式转换的情况
一般开发者只是认为防止转换构造函数(单参数的),其实对于转换运算符的隐式转换也是可以控制的。也就是上面提到的,其它情况下的一些函数,看代码:

class Data {
public:/*explicit*/ Data(int d) {d_ = d;std::cerr << "set value:" << d << std::endl;}/*explicit*/ operator int() const { return d_; }private:int d_ = 0;
};
int main() {Data d = 10;int d1 = d;return 0;
}

使用explicit最主要目的只有一个,防止编译器的隐式转换产生不可预知的结果。而这一结果进而提高了代码的可维护性。

四、总结

其实一直对这个关键字没有注意过,虽然也会使用,但也没有什么深刻的印象。有的时候看代码发现多个参数的构造函数也应用了这个关键字,也没有发现有什么不妥。直到后来一次实际操作中,才发现其实多个参数使用它没有实际意义。虽然不至于有什么问题,但仍然是对细节的把控不到位。
写这篇文章就是和诸君共勉!

这篇关于跟我学C++中级篇——explicit的分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

C++右移运算符的一个小坑及解决

《C++右移运算符的一个小坑及解决》文章指出右移运算符处理负数时左侧补1导致死循环,与除法行为不同,强调需注意补码机制以正确统计二进制1的个数... 目录我遇到了这么一个www.chinasem.cn函数由此可以看到也很好理解总结我遇到了这么一个函数template<typename T>unsigned

C++统计函数执行时间的最佳实践

《C++统计函数执行时间的最佳实践》在软件开发过程中,性能分析是优化程序的重要环节,了解函数的执行时间分布对于识别性能瓶颈至关重要,本文将分享一个C++函数执行时间统计工具,希望对大家有所帮助... 目录前言工具特性核心设计1. 数据结构设计2. 单例模式管理器3. RAII自动计时使用方法基本用法高级用法

深入解析C++ 中std::map内存管理

《深入解析C++中std::map内存管理》文章详解C++std::map内存管理,指出clear()仅删除元素可能不释放底层内存,建议用swap()与空map交换以彻底释放,针对指针类型需手动de... 目录1️、基本清空std::map2️、使用 swap 彻底释放内存3️、map 中存储指针类型的对象

C++ STL-string类底层实现过程

《C++STL-string类底层实现过程》本文实现了一个简易的string类,涵盖动态数组存储、深拷贝机制、迭代器支持、容量调整、字符串修改、运算符重载等功能,模拟标准string核心特性,重点强... 目录实现框架一、默认成员函数1.默认构造函数2.构造函数3.拷贝构造函数(重点)4.赋值运算符重载函数

C++ vector越界问题的完整解决方案

《C++vector越界问题的完整解决方案》在C++开发中,std::vector作为最常用的动态数组容器,其便捷性与性能优势使其成为处理可变长度数据的首选,然而,数组越界访问始终是威胁程序稳定性的... 目录引言一、vector越界的底层原理与危害1.1 越界访问的本质原因1.2 越界访问的实际危害二、基

c++日志库log4cplus快速入门小结

《c++日志库log4cplus快速入门小结》文章浏览阅读1.1w次,点赞9次,收藏44次。本文介绍Log4cplus,一种适用于C++的线程安全日志记录API,提供灵活的日志管理和配置控制。文章涵盖... 目录简介日志等级配置文件使用关于初始化使用示例总结参考资料简介log4j 用于Java,log4c

Android 缓存日志Logcat导出与分析最佳实践

《Android缓存日志Logcat导出与分析最佳实践》本文全面介绍AndroidLogcat缓存日志的导出与分析方法,涵盖按进程、缓冲区类型及日志级别过滤,自动化工具使用,常见问题解决方案和最佳实... 目录android 缓存日志(Logcat)导出与分析全攻略为什么要导出缓存日志?按需过滤导出1. 按

C++归并排序代码实现示例代码

《C++归并排序代码实现示例代码》归并排序将待排序数组分成两个子数组,分别对这两个子数组进行排序,然后将排序好的子数组合并,得到排序后的数组,:本文主要介绍C++归并排序代码实现的相关资料,需要的... 目录1 算法核心思想2 代码实现3 算法时间复杂度1 算法核心思想归并排序是一种高效的排序方式,需要用

Linux中的HTTPS协议原理分析

《Linux中的HTTPS协议原理分析》文章解释了HTTPS的必要性:HTTP明文传输易被篡改和劫持,HTTPS通过非对称加密协商对称密钥、CA证书认证和混合加密机制,有效防范中间人攻击,保障通信安全... 目录一、什么是加密和解密?二、为什么需要加密?三、常见的加密方式3.1 对称加密3.2非对称加密四、

MySQL中读写分离方案对比分析与选型建议

《MySQL中读写分离方案对比分析与选型建议》MySQL读写分离是提升数据库可用性和性能的常见手段,本文将围绕现实生产环境中常见的几种读写分离模式进行系统对比,希望对大家有所帮助... 目录一、问题背景介绍二、多种解决方案对比2.1 原生mysql主从复制2.2 Proxy层中间件:ProxySQL2.3