当前位置: 首页 > 编程日记 > 正文

C++中的const关键字(zz)

【补充】mutable关键字

有时我们希望类的数据成员即使在const成员函数中,依然是可以修改的,这时就可以把它们声明为mutable来实现。这样的应用,比如记录各种操作的调用次数,这时,即使在const中,依然要修改计数器,就可以将计数器变量设置为mutable的。

【补充】this指针与const

类ClassSample中:

  • 非const的成员函数中,this的类型是一个指向类类型的const指针,相当于:ClassSample const *this;这时,可以改变对象,但不能让this指向另外一个地址;
  • 在const的成员函数中,this的类型是一个指向const类类型的const指针,相当于:const ClassSample const *this;这时,不能改变对象,也不能让this指向另外一个地址;

注意:不能从const成员函数中返回指向类对象的普通引用,const成员函数只能返回*this作为一个const应用。

假定类中一个函数display(),它应该是个const的,因为它不应该修改对象,但是程序员有可能使用如下的方式调用screen.move(4,0).display(),这时还是没什么问题的,但如果这个操作序列变成了screen.display().move(4,0),display返回的是const引用,是不可以调用move函数的。解决的办法就是定义两个display函数:

#include <iostream>
using namespace std;

class Screen
{
public:
    Screen& display(){return *this;}
    const Screen& display() const
    {
        return *this;
    }
};

这样,操作序列中的操作对象是const时会调用const的display,如果操作对象不是const,则调用非const的display,从而操作序列可以正常使用,但又不充分const保证健壮性。

C++中的const关键字的用法非常灵活,而使用const将大大改善程序的健壮性,本人根据各方面查到的资料进行总结如下,期望对朋友们有所帮助。
Const 是C++中常用的类型修饰符,常类型是指使用类型修饰符const说明的类型,常类型的变量或对象的值是不能被更新的。概括来说,const关键字就是用来在编译时检查任何不合规的操作,从而在编译时发现错误的一种机制。下面是一个例子程序,几乎包括了下面讲到的所有的内容:

#include <iostream>

class ConstSample
{
public:
    ConstSample():Value(0) {
        Value=1; //不能用这种方式初始化const member
    }
    ConstSample(int v):Value(v) //使用初始化列表
    {
    }
   
    ConstSample(int v, int w):Value2(v)// "Value2" is not a nonstatic data member or base class of class "ConstSample"
    {

    }
    void non_const_func1(){};
    void const_func1() const{};
    void const_func2() const
    {
        const_func1();
        Value3 = 4; //error:only can access const members;
        non_const_func1();// error: only can call const function in const function;
    }
    void const_param_func1(const int* a, int *b)
    {
        *a = 0; // a is const value, can not be modified;
        *b = 2;
    }
    const int Value;

    const  ConstSample & operator ++()
    {
        Value3++;
        return *this;
    }

private:
    static const int Value2;
    int Value3;

};

class ConstSample_Operator
{
public:
     ConstSample_Operator & operator ++()
    {
        return *this;
    }
};

const int ConstSample::Value2 = 2; // static的可以外部初始化
static const int ConstSample::Value = 2; //error:a nonstatic data member may not be defined outside its class


int main(int argc, char **argv)
{
    const ConstSample sampleRef;
    ConstSample sampleRef2;
    const ConstSample * samplePtr = new ConstSample(2);
    sampleRef.non_const_func1(); // const reference only can call const method;
    samplePtr->non_const_func1();// const pointer only can call const method;
    sampleRef2.non_const_func1();
    sampleRef2.const_func1();
    samplePtr->const_func1();
    ++sampleRef2; //works;
    ++++sampleRef2; //does not work, because the operator ++ of ConstSample return const reference;
    ConstSample_Operator op;
    ++++op; // works;
}


一、Const作用
 
二、Const的使用
1、定义常量
(1)const修饰变量,以下两种定义形式在本质上是一样的。它的含义是:const修饰的类型为TYPE的变量value是不可变的。

TYPE const ValueName = value;
const TYPE ValueName = value;

(2)将const改为外部连接,作用于扩大至全局,编译时会分配内存,并且可以不进行初始化,仅仅作为声明,编译器认为在程序其他地方进行了定义.

extern const int ValueName = value;

2、指针使用CONST
(1)指针本身是常量不可变

(char*) const pContent;
const (char*) pContent;

(2)指针所指向的内容是常量不可变

const (char) *pContent;
(char) const *pContent;

(3)两者都不可变

const char* const pContent;

(4)还有其中区别方法,沿着*号划一条线:
如果const位于*的左侧,则const就是用来修饰指针所指向的变量,即指针指向为常量;
如果const位于*的右侧,const就是修饰指针本身,即指针本身是常量。

3、函数中使用CONST

(1)const修饰函数参数
a.传递过来的参数在函数内不可以改变(无意义,因为Var本身就是形参)

void function(const int Var);

b.参数指针所指内容为常量不可变

void function(const char* Var);

c.参数指针本身为常量不可变(也无意义,因为char* Var也是形参)

void function(char* const Var);

d.参数为引用,为了增加效率同时防止修改。修饰引用参数时:

void function(const Class& Var); //引用参数在函数内不可以改变
 
void function(const TYPE& Var); //引用参数在函数内为常量不可变

这样的一个const引用传递和最普通的函数按值传递的效果是一模一样的,他禁止对引用的对象的一切修改,唯一不同的是按值传递会先建立一个类对象的副本, 然后传递过去,而它直接传递地址,所以这种传递比按值传递更有效.另外只有引用的const传递可以传递一个临时对象,因为临时对象都是const属性, 且是不可见的,他短时间存在一个局部域中,所以不能使用指针,只有引用的const传递能够捕捉到这个家伙.


(2)const 修饰函数返回值
    const修饰函数返回值其实用的并不是很多,它的含义和const修饰普通变量以及指针的含义基本相同.

a. const int fun1() //这个其实无意义,因为参数返回本身就是赋值。
    b. const int * fun2() //调用时 const int *pValue = fun2();
                         
//我们可以把fun2()看作成一个变量,即指针内容不可变。
    c.int* const fun3()   //调用时 int * const pValue = fun2();
                         
//我们可以把fun2()看作成一个变量,即指针本身不可变。

一般情况下,函数的返回值为某个对象时,如果将其声明为const时,多用于操作符的重载。通常,不建议用const修饰函数的返回值类型为某个对象或对某个对象引用的情况。原因如下:如果返回值为某个对象为const(const A test = A 实例)或某个对象的引用为const(const A& test = A实例) ,则返回值具有const属性,则返回实例只能访问类A中的公有(保护)数据成员和const成员函数,并且不允许对其进行赋值操作,这在一般情况下很少用到。

另外一种情况是,若函数的返回值是基本数据类型时,返回值是用值传递的,值存在于某个临时存储区中,用const修饰是没有意义的。

4、类相关CONST

(1)const修饰成员变量
const修饰类的成员函数,表示成员常量,不能被修改,同时它只能在初始化列表中赋值.

class A
    {
        …
        const int nValue;         //成员常量不能被修改
        …
        A(int x): nValue(x) { } ; //只能在初始化列表中赋值
     }

(2)const修饰成员函数
const修饰类的成员函数,则该成员函数不能修改类中任何非const成员函数。一般写在函数的最后来修饰

class A
    {
        …
       void function()const; //常成员函数, 它不改变对象的成员变量.                       

//也不能调用类中任何非const成员函数。
}

对于const类对象/指针/引用,只能调用类的const成员函数,因此,const修饰成员函数的最重要作用就是限制对于const对象的使用。

a. const成员函数不被允许修改它所在对象的任何一个数据成员。

b. const成员函数能够访问对象的const成员,而其他成员函数不可以。

(3)const修饰类对象/对象指针/对象引用

  • const修饰类对象表示该对象为常量对象,其中的任何成员都不能被修改。对于对象指针和对象引用也是一样。
  • const修饰的对象,该对象的任何非const成员函数都不能被调用,因为任何非const成员函数会有修改成员变量的企图。

例如:

class AAA
{
    void func1();
void func2() const;
}
const AAA aObj;
aObj.func1(); //×
aObj.func2(); //正确
 
const AAA* aObj = new AAA();
aObj-> func1(); //×
aObj-> func2(); //正确


三、将Const类型转化为非Const类型的方法

采用const_cast 进行转换。 
用法:const_cast <type_id>  (expression)
该运算符用来修改类型的const或volatile属性。除了const 或volatile修饰之外, type_id和expression的类型是一样的。

  • 常量指针被转化成非常量指针,并且仍然指向原来的对象;
  • 常量引用被转换成非常量引用,并且仍然指向原来的对象;
  • 常量对象被转换成非常量对象。


四、使用const的一些建议

  • 要大胆的使用const,这将给你带来无尽的益处,但前提是你必须搞清楚原委;
  • 要避免最一般的赋值操作错误,如将const变量赋值,具体可见思考题;
  • 在参数中使用const应该使用引用或指针,而不是一般的对象实例,原因同上;
  • const在成员函数中的三种用法(参数、返回值、函数)要很好的使用;
  • 不要轻易的将函数的返回值类型定为const;
  • 除了重载操作符外一般不要将返回值类型定为对某个对象的const引用;
  • 任何不会修改数据成员的函数都应该声明为const 类型。

五、补充重要说明

  • 类内部的常量限制:使用这种类内部的初始化语法的时候,常量必须是被一个常量表达式初始化的整型或枚举类型,而且必须是static和const形式。
  • 如何初始化类内部的常量:一种方法就是static 和 const 并用,在外部初始化,例如:

class A { public: A() {} private: static const int i; file://注意必须是静态的! };

const int A::i=3;另一个很常见的方法就是初始化列表: class A { public: A(int

i=0):test(i) {} private: const int i; }; 还有一种方式就是在外部初始化,

  • 如果在非const成员函数中,this指针只是一个类类型的;如果在const成员函数中,this指针是一个const类类型的;如果在volatile成员函数中,this指针就是一个volatile类类型的。
  • new返回的指针必须是const类型的。

const char*, char const*, char*const的区别问题几乎是C++面试中每次都会有的题目。

事实上这个概念谁都有只是三种声明方式非常相似很容易记混。
Bjarne在他的The C++ Programming Language里面给出过一个助记的方法:
把一个声明从右向左读。

char  * const cp; ( * 读成 pointer to )
cp is a const pointer to char

const char * p;
p is a pointer to const char;

char const * p;
同上因为C++里面没有const*的运算符,所以const只能属于前面的类型。

http://blog.csdn.net/Eric_Jo/article/details/4138548

转载于:https://www.cnblogs.com/whyandinside/archive/2012/09/11/2680322.html

相关文章:

Python让你成为AI 绘画大师,简直太惊艳了!(附代码))

作者 | 李秋键责编 | 李雪敬头图 | CSDN下载自视觉中国引言&#xff1a;基于前段时间我在CSDN上创作的文章“CylcleGAN人脸转卡通图”的不足&#xff0c;今天给大家分享一个更加完美的绘制卡通的项目“Learning to Cartoonize Using White-box Cartoon Representations”。首先…

Vue 2 | Part 4 v-bind绑定元素属性和样式

这期跟大家分享的&#xff0c;是v-bind指令。它可以往元素的属性中绑定数据&#xff0c;也可以动态地根据数据为元素绑定不同的样式。 绑定属性 最简单的例子&#xff0c;我们有一张图片&#xff0c;需要定义图片的src。我们可以直接在元素的属性里面定义&#xff1a; <div …

在 ASP.NET 中执行 URL 重写

在 ASP.NET 中执行 URL 重写 发布日期&#xff1a; 8/23/2004| 更新日期&#xff1a; 8/23/2004Scott Mitchell 4GuysFromRolla.com 适用范围&#xff1a; Microsoft ASP.NET 摘要&#xff1a;介绍如何使用 Microsoft ASP.NET 执行动态 URL 重写。URL 重写是截取传入 Web 请求并…

win8中使用BitLocker加密

一、加密驱动器二、管理三、TPM转载于:https://blog.51cto.com/jimshu/989359

​清华硕士爆料:这些才是机器学习必备的数学基础

现如今&#xff0c;计算机科学、人工智能、数据科学已成为技术发展的主要推动力。无论是要翻阅这些领域的文章&#xff0c;还是要参与相关任务&#xff0c;你马上就会遇到一些拦路虎&#xff1a;想过滤垃圾邮件&#xff0c;不具备概率论中的贝叶斯思维恐怕不行&#xff1b;想试…

Oracle Golden Gate体系架构详解(原创) - CzmMiao的博客生活 - ITeye技术网站

Oracle Golden Gate体系架构详解(原创) - CzmMiao的博客生活 - ITeye技术网站

用C#对ADO.NET数据库完成简单操作

作者&#xff1a;李阳 http://oraasp.vicp.net/article/article.aspx?ID21 数据库访问是程序中应用最普遍的部分。随着C#和ADO.NET的引入&#xff0c;这种操作变得更简单。这篇文章将示范四种最基础的数据库操作。 ● 读取数据。其中包括多种数据类型&#xff1a;整型&#…

用createrepo配置Yum本地源

yum配置本地源, 在网速差的情况下&#xff0c;yum用在线源是一件头痛的事&#xff0c;所以以下为yum的本地源配置可以有好解决这个事。 1,安装createrepo包&#xff0c; 可以用yum安装(yum install createrepo -y); 也可以安装rpm或tar包 &#xff08;网址&#xff1a;createre…

首次在手机端不牺牲准确率实现BERT实时推理,比TensorFlow-Lite快近8倍,每帧只需45ms...

作者 | 王言治 出品 | AI科技大本营&#xff08;ID:rgznai100&#xff09; 基于Transformer的预训练模型在许多自然语言处理&#xff08;NLP&#xff09;任务中取得了很高的准确度。但是这些预训练模型往往需要很大的计算量和内存。由于移动平台的存储空间以及计算能力的限制&a…

[svc]caffe安装笔记-显卡购买

caffe,这是是数据组需要做一些大数据模型的训练(深度学习), 要求 服务器显卡(运算卡), 刚开始老板让买的牌子是泰坦的(这是2年前的事情了). 后来买不到这个牌子的,(jd,tb)看过丽台的,看过gtx系列的哪个型号来着, 也不合适,后来买的特斯拉显卡 [查了下一些知名的显卡牌子](https…

AABO:自适应最优化Anchor设置,性能榨取的最后一步 | ECCV 2020

编译 | VincentLee来源 | 晓飞的算法工程笔记Introduction目前&#xff0c;主流的目标检测算法使用多种形状的anchor box作为初始预测&#xff0c;然后对anchor box进行回归调整&#xff0c;anchor box的配置是检测算法中十分重要的超参数。一般而言&#xff0c;anchor box的配…

Android列表控件选项中添加进度框ProgressBar实现

今天有时间就学习了下在ListView、GridView列表项中清加ProgressBar,小马用最简单的代码实现可以通用的功能&#xff0c;人人都能看懂&#xff0c;哈哈&#xff0c;直接说下&#xff0c;如果你的适配器getView方法返回的View是一个自定义控件的话&#xff0c;有点不好实现哦&am…

写一个通用数据访问组件

出处&#xff1a;http://www.csharp-corner.com willsound&#xff08;翻译&#xff09; 我收到过好多Email来问我如何用一个通用的数据提供者(data provider)在不失自然数据提供者(native data provider)稳定而强大功能的前提下来访问不同的数据源(data sources).一个小伙子…

InstallShield 2015 LimitedEdition VS2012 运行bat文件

转载:http://www.cnblogs.com/fengwenit/p/4271150.html 运行bat文件 网上很多介绍如何运行bat的方法&#xff0c;但我这个是limted 版本&#xff0c;不适用。 1. 打开 Define Setup Requirements and Actions –> Custom Actions 2. 右健 After Register Product –> Ne…

理解C#中的string类型

作者&#xff1a;未知目的 本文的目的在于揭示和DOTNET及C#相关的一些常见的和不常见的问题。在这些问题中我的第一篇文章和string数据类型有关,string数据类型是一种引用类型,但是当和其他引用类型比较的时候,很多开发人员可能并不能完全理解它的行为。 问题 对于常见的引用类…

最全总结!聊聊 Python 操作PDF的几种方法

作者 | 陈熹来源 | 早起Python前言本文主要涉及&#xff1a;os 模块综合应用glob 模块综合应用PyPDF2 模块操作基本操作PyPDF2 导入模块的代码常常是&#xff1a;from PyPDF2 import PdfFileReader, PdfFileWriter这里导入了两个方法&#xff1a;PdfFileReader 可以理解为读取器…

three.js(六) 地形法向量生成

2019独角兽企业重金招聘Python工程师标准>>> 上一节采用 分形算法生成地形的高度值&#xff0c; 接着我们需要生成每个顶点的法向量。 three.js 的PlaneGeometry 自带有法向量&#xff0c; 法向量分为两种 即 平面法向量 和 平面每个定点法向量。 因此一个n*n 块组成…

ASP.NET中使用多个runat=server form

作者&#xff1a;未知ASP.NET 在同一个页面不支持多个 runatserver forms&#xff0c;要解决这个问题&#xff0c;可以把每个 form 放在一个单独的 panel 控件中&#xff0c;这样用户就可以简单地通过单选按钮在不同 panel 间切换。代码如下&#xff1a;2FormExample.aspx<%…

激发企业大“智慧” | 深度赋能AI全场景 揭秘你不知道的移动云

2020年是人工智能技术发展的关键年。疫情之下&#xff0c;世界见证了人工智能在抗击疫情中发挥的积极作用&#xff1b;今年4月&#xff0c;国家发改委正式将人工智能确定为新基建的重要领域之一。在历史机遇下&#xff0c;AI已实现"质变和量变"&#xff0c;正迈入与技…

ExtJS 4.x 得到资源树上任意的节点对象

上半年做ExtJS 4.x 的时候&#xff0c;遇到过对资源树操作的情况&#xff1a; Ext.tree.Panel 如下图&#xff1a;目的&#xff1a; 直接根据每个节点的{任意key : 对应value}&#xff0c;就能找到匹配的节点对象 代码如下&#xff1a; refs : [ { selector : rtree, …

【转载】mysql常用函数汇总

转载地址&#xff1a;http://www.jb51.net/article/40179.htm 一、数学函数ABS(x) 返回x的绝对值BIN(x) 返回x的二进制&#xff08;OCT返回八进制&#xff0c;HEX返回十六进制&#xff09;CEILING(x) 返回大于x的最小整数值EXP(x) 返回值e&#xff08;自然对数的底&…

有关java的一些话

2019独角兽企业重金招聘Python工程师标准>>> 跟着做完TankWar&#xff0c;java才算是入门了&#xff0c;真正学习java从看尚学堂马士兵老师的视频开始&#xff0c;至今三个月已过&#xff0c;感谢马老师的精彩讲解&#xff0c;您才是我真正的java入门老师&#xff0…

ADO.NET 2.0中的SqlCommand.ExecutePageReader

http://blog.joycode.com/liuhuimiao/在.NET 2.0 PDC或Beta1中&#xff0c;可以看到SqlCommand对象新增了个ExecutePageReader方法&#xff0c;该方法实现了分页读取数据的功能。对于分页读取数据&#xff0c;在ADO.NET1.1中&#xff08;当然2.0也适合&#xff09;一般常用动态…

组合游戏系列5: 井字棋、五子棋AlphaGo Zero 算法实战

来源 | MyEncyclopedia上一篇我们从原理层面解析了AlphaGo Zero如何改进MCTS算法&#xff0c;通过不断自我对弈&#xff0c;最终实现从零棋力开始训练直至能够打败任何高手。在本篇中&#xff0c;我们在已有的N子棋OpenAI Gym 环境中用Pytorch实现一个简化版的AlphaGo Zero算法…

2020职场人裸辞三大原因:不开心、工资低、没有盼头

近期&#xff0c;脉脉发布了《2020职场人裸辞现状调研报道》&#xff0c;报道显示2020最让职场人想裸辞的三大原因为&#xff1a;不开心、工资低、没有盼头。报告数据中还显示&#xff0c;工资不满预期是最让人想要裸辞的主要原因&#xff0c;但有超过6成职场人表示&#xff0c…

Oracle PL/SQL编程学习笔记:Merge方法的使用

Oracle11g的Merge很强大&#xff01; 1 create or replace procedure BRANCE_REPORT_MERGE is2 3 begin4 Merge into BRANCHREPORT desttable5 using TEMP_BRANCHREPORT tmptable6 on (desttable.SENDER_IDtmptable.SENDER_ID and desttable.BRANCH_IDtmptable.BRANCH_ID…

2.0中获取数据库连接统计数据

作者&#xff1a; http://blog.joycode.com/liuhuimiao/.NET 2.0中的SqlConnection多了一个StatisticsEnabled属性和ResetStatistics()、RetrieveStatistics()两个方法&#xff0c;用于获取SQLServer的连接统计数据。当然&#xff0c;这样做是以性能损耗为代价的&#xff0c;但…

Python学习day5作业-ATM和购物商城

Python学习day5作业Python学习day5作业ATM和购物商城作业需求ATM&#xff1a;指定最大透支额度可取款定期还款&#xff08;每月指定日期还款&#xff0c;如15号&#xff09;可存款定期出账单支持多用户登陆&#xff0c;用户间转帐支持多用户管理员可添加账户、指定用户额度、冻…

60分钟看懂HMM的基本原理

作者 | 梁云1991来源 | Python与算法之美HMM模型&#xff0c;韩梅梅的中文拼音的缩写&#xff0c;所以又叫韩梅梅模型&#xff0c;由于这个模型的作者是韩梅梅的粉丝&#xff0c;所以给这个模型取名为HMM。开玩笑&#xff01;HMM模型&#xff0c;也叫做隐马尔科夫模型&#xff…

获取远程网卡MAC地址

出自&#xff1a; http://blog.joycode.com/liuhuimiao/朋友mingal急问我有关获取远程网卡MAC地址的ASP.net实现。我一开始以为是获取本机MAC地址&#xff0c;说了几种方法给他。由于他还需要获取服务器&#xff08;本机&#xff09;相关信息&#xff0c;如硬盘序列号、CPU信息…