Effective C++(一): Const Correctness, Const成员函数和Const Cast

2023-12-02 19:36

本文主要是介绍Effective C++(一): Const Correctness, Const成员函数和Const Cast,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 一、Const成员函数
  • 二、Const Correctness
  • 三、Const Cast


有关 const 的用法是 cpp 中一个非常经典且易错的部分,在面试和日常工作中各种各样的 const 经常让人摸不着头脑,今天就来根据 const 扮演的不同角色来归纳有关 const 的不同用法

一、Const成员函数

const 在成员函数中的用法可谓是面试必问题。一般来说会涉及到以下几点:
我们假设我们自己实现了一个 String 类并且重载了他的运算符[]:

class MyString {
public:MyString(const std::string& str) : text(str) {}// const版本的operator[]const char& operator[](std::size_t position) const {return text[position];}// non-const版本的operator[]char& operator[](std::size_t position) {return text[position];}private:std::string text;
};int main() { MyString mutableString("Hello");const MyString constString("world");mutableString[0] = 'h'; //ok this could be modified char ch = constString[0]; //ok, and here constString "uses" const char& operator[] because it is a const class.constString[0] = 'w'; //error. const char& could not be modified.报错信息如下:/*chapter2.cpp:28:18: error: cannot assign to return value because function 'operator[]' returns a const valueconstString[0] = 'w'; //error. const char& could not be modified.~~~~~~~~~~~~~~ ^chapter2.cpp:10:11: note: function 'operator[]' which returns const-qualified type 'const char &' declared hereconst char& operator[](std::size_t position) const {^~~~~1 error generated.*/return 0;
}

在这里我们可以清楚看到这两个函数的区别,第一个重载的运算符只能用在 const 对象上而第二个重载的运算符[] 只能用在非 const 对象上,同时第一个重载的运算符最后的 const 后缀非常重要,这意味着这个函数不会修改任何类中的成员变量(除了被标记为 mutable 的变量)。这个关键字让编译器知道,这个函数可以在 const 对象上调用。如果没有这个 const 后缀,函数就不能在 const 对象上调用。我们可以试着把这个 const 后缀删除,再次运行就会发现会报错。 报错信息如下:

chapter2.cpp:27:24: error: no viable overloaded operator[] for type 'const MyString'char ch = constString[0]; //ok, and here constString "uses" const char& operator[] because it is a const class.~~~~~~~~~~~^~
chapter2.cpp:10:17: note: candidate function not viable: 'this' argument has type 'const MyString', but method is not marked constconst char& operator[](std::size_t position) {^
1 error generated.

二、Const Correctness

我们都知道被 const 修饰的变量绝大多数情况下是不能被修改的,然而,当我们的变量类型比较复杂,比如是一个指针的时候,如何判断常量指针还是指针常量呢?
下面是一些例子:

char greeting[] = "Hello";// 指针可修改,数据可修改
char* p1 = greeting; 
p1[0] = 'h'; // 正确: 数据可修改
p1 = nullptr; // 正确: 指针可修改// 指针可修改,数据不可修改
const char* p2 = greeting; 
// p2[0] = 'h'; // 错误: 数据不可修改
p2 = nullptr; // 正确: 指针可修改char const* p22 = greeting;
//p22[0] = 'h'; //错误,数据不能被修改
p22 = nullptr //正确,指针可以修改p2 和 p22 也叫做常量指针,即指针可以被修改但是数据不能被修改// 指针不可修改,数据可修改
char* const p3 = greeting; 
p3[0] = 'h'; // 正确: 数据可修改
// p3 = nullptr; // 错误: 指针不可修改p3 也叫做指针常量,即数据可以被修改,但是指针不能被修改// 指针不可修改,数据不可修改
const char* const p4 = greeting; 
// p4[0] = 'h'; // 错误: 数据不可修改
// p4 = nullptr; // 错误: 指针不可修改

三、Const Cast

首先我们先来介绍一下我们为什么需要 const cast。
在第一个知识点中,我们看到对于同一个类对象的运算符重载函数,我们有两种形式,一种是作用于 const 类变量的,一种是作用于非 const 类变量的,这两个函数尽管有这不同的修饰符,但是他们的实现逻辑是一样的,这就造成了函数实现的冗余。在工程中,我们要么需要改变 const 运算符重载符函数的函数实现逻辑,要么改变非 const 运算符重载函数的函数实现逻辑。 一般来说我们会选择后者,因为在 C++中,const 变量拥有着比非 const 变量多的限制,如果我们选择将 const 变量转化为非 const 变量会带来更多的安全隐患。 因此,一般为了优化代码,我们会将非 const 对象中的运算符重载函数改为以下形式:

    const char& operator[](std::size_t position) const {// 假设这里有非常多的代码return text[position];}char& operator[](std::size_t position) {return const_cast<char&>(static_cast<const TextBlock&>(*this)[position]);}

在这里,static_cast的作用是用来处理几乎所有类型转换的类型转换运算符,除了涉及底层的 const 转换。比如 int 到 float 的转换和派生类到基类之间的转换(注意基类到派生类的转换是不允许的) , 而 const_cast的作用是: 删除或者添加(极少数使用) 对象的 const 属性,这意味着当你拥有一个 const 对象但是你需要调用或者返回一个非 const 属性的时候,你需要通过const_cast抹除对象的 const 属性。 当然,你也可以使用 const_cast添加一个对象的 const 属性,但是这不常用,而且很容易出现各种奇怪的 bug。


这篇关于Effective C++(一): Const Correctness, Const成员函数和Const Cast的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Python中bisect_left 函数实现高效插入与有序列表管理

《Python中bisect_left函数实现高效插入与有序列表管理》Python的bisect_left函数通过二分查找高效定位有序列表插入位置,与bisect_right的区别在于处理重复元素时... 目录一、bisect_left 基本介绍1.1 函数定义1.2 核心功能二、bisect_left 与

java中BigDecimal里面的subtract函数介绍及实现方法

《java中BigDecimal里面的subtract函数介绍及实现方法》在Java中实现减法操作需要根据数据类型选择不同方法,主要分为数值型减法和字符串减法两种场景,本文给大家介绍java中BigD... 目录Java中BigDecimal里面的subtract函数的意思?一、数值型减法(高精度计算)1.

C/C++ chrono简单使用场景示例详解

《C/C++chrono简单使用场景示例详解》:本文主要介绍C/C++chrono简单使用场景示例详解,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友... 目录chrono使用场景举例1 输出格式化字符串chrono使用场景China编程举例1 输出格式化字符串示

C++/类与对象/默认成员函数@构造函数的用法

《C++/类与对象/默认成员函数@构造函数的用法》:本文主要介绍C++/类与对象/默认成员函数@构造函数的用法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录名词概念默认成员函数构造函数概念函数特征显示构造函数隐式构造函数总结名词概念默认构造函数:不用传参就可以

C++类和对象之默认成员函数的使用解读

《C++类和对象之默认成员函数的使用解读》:本文主要介绍C++类和对象之默认成员函数的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、默认成员函数有哪些二、各默认成员函数详解默认构造函数析构函数拷贝构造函数拷贝赋值运算符三、默认成员函数的注意事项总结一

C/C++中OpenCV 矩阵运算的实现

《C/C++中OpenCV矩阵运算的实现》本文主要介绍了C/C++中OpenCV矩阵运算的实现,包括基本算术运算(标量与矩阵)、矩阵乘法、转置、逆矩阵、行列式、迹、范数等操作,感兴趣的可以了解一下... 目录矩阵的创建与初始化创建矩阵访问矩阵元素基本的算术运算 ➕➖✖️➗矩阵与标量运算矩阵与矩阵运算 (逐元

C/C++的OpenCV 进行图像梯度提取的几种实现

《C/C++的OpenCV进行图像梯度提取的几种实现》本文主要介绍了C/C++的OpenCV进行图像梯度提取的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的... 目录预www.chinasem.cn备知识1. 图像加载与预处理2. Sobel 算子计算 X 和 Y

C/C++和OpenCV实现调用摄像头

《C/C++和OpenCV实现调用摄像头》本文主要介绍了C/C++和OpenCV实现调用摄像头,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录准备工作1. 打开摄像头2. 读取视频帧3. 显示视频帧4. 释放资源5. 获取和设置摄像头属性

c/c++的opencv图像金字塔缩放实现

《c/c++的opencv图像金字塔缩放实现》本文主要介绍了c/c++的opencv图像金字塔缩放实现,通过对原始图像进行连续的下采样或上采样操作,生成一系列不同分辨率的图像,具有一定的参考价值,感兴... 目录图像金字塔简介图像下采样 (cv::pyrDown)图像上采样 (cv::pyrUp)C++ O

c/c++的opencv实现图片膨胀

《c/c++的opencv实现图片膨胀》图像膨胀是形态学操作,通过结构元素扩张亮区填充孔洞、连接断开部分、加粗物体,OpenCV的cv::dilate函数实现该操作,本文就来介绍一下opencv图片... 目录什么是图像膨胀?结构元素 (KerChina编程nel)OpenCV 中的 cv::dilate() 函