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

相关文章

Django中的函数视图和类视图以及路由的定义方式

《Django中的函数视图和类视图以及路由的定义方式》Django视图分函数视图和类视图,前者用函数处理请求,后者继承View类定义方法,路由使用path()、re_path()或url(),通过in... 目录函数视图类视图路由总路由函数视图的路由类视图定义路由总结Django允许接收的请求方法http

MySQL常用字符串函数示例和场景介绍

《MySQL常用字符串函数示例和场景介绍》MySQL提供了丰富的字符串函数帮助我们高效地对字符串进行处理、转换和分析,本文我将全面且深入地介绍MySQL常用的字符串函数,并结合具体示例和场景,帮你熟练... 目录一、字符串函数概述1.1 字符串函数的作用1.2 字符串函数分类二、字符串长度与统计函数2.1

python使用try函数详解

《python使用try函数详解》Pythontry语句用于异常处理,支持捕获特定/多种异常、else/final子句确保资源释放,结合with语句自动清理,可自定义异常及嵌套结构,灵活应对错误场景... 目录try 函数的基本语法捕获特定异常捕获多个异常使用 else 子句使用 finally 子句捕获所

C++11范围for初始化列表auto decltype详解

《C++11范围for初始化列表autodecltype详解》C++11引入auto类型推导、decltype类型推断、统一列表初始化、范围for循环及智能指针,提升代码简洁性、类型安全与资源管理效... 目录C++11新特性1. 自动类型推导auto1.1 基本语法2. decltype3. 列表初始化3

C++11右值引用与Lambda表达式的使用

《C++11右值引用与Lambda表达式的使用》C++11引入右值引用,实现移动语义提升性能,支持资源转移与完美转发;同时引入Lambda表达式,简化匿名函数定义,通过捕获列表和参数列表灵活处理变量... 目录C++11新特性右值引用和移动语义左值 / 右值常见的左值和右值移动语义移动构造函数移动复制运算符

C++中detach的作用、使用场景及注意事项

《C++中detach的作用、使用场景及注意事项》关于C++中的detach,它主要涉及多线程编程中的线程管理,理解detach的作用、使用场景以及注意事项,对于写出高效、安全的多线程程序至关重要,下... 目录一、什么是join()?它的作用是什么?类比一下:二、join()的作用总结三、join()怎么

postgresql使用UUID函数的方法

《postgresql使用UUID函数的方法》本文给大家介绍postgresql使用UUID函数的方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录PostgreSQL有两种生成uuid的方法。可以先通过sql查看是否已安装扩展函数,和可以安装的扩展函数

MySQL字符串常用函数详解

《MySQL字符串常用函数详解》本文给大家介绍MySQL字符串常用函数,本文结合实例代码给大家介绍的非常详细,对大家学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录mysql字符串常用函数一、获取二、大小写转换三、拼接四、截取五、比较、反转、替换六、去空白、填充MySQL字符串常用函数一、

C++中全局变量和局部变量的区别

《C++中全局变量和局部变量的区别》本文主要介绍了C++中全局变量和局部变量的区别,全局变量和局部变量在作用域和生命周期上有显著的区别,下面就来介绍一下,感兴趣的可以了解一下... 目录一、全局变量定义生命周期存储位置代码示例输出二、局部变量定义生命周期存储位置代码示例输出三、全局变量和局部变量的区别作用域

C++中assign函数的使用

《C++中assign函数的使用》在C++标准模板库中,std::list等容器都提供了assign成员函数,它比操作符更灵活,支持多种初始化方式,下面就来介绍一下assign的用法,具有一定的参考价... 目录​1.assign的基本功能​​语法​2. 具体用法示例​​​(1) 填充n个相同值​​(2)