什么是const限定符?

Const限定符是我们通常所说的常量限定符,被const修饰的对象具有常量性质,只能读,不能写。

为什么使用const限定符?

用const变量取代“魔数”,代码更容易理解和维护。例如:以const常变量作为数组的界;const常变量作为switch的条件标号。

C++“最小特权原则”的体现,防止不应该被修改的数据被篡改。例如:很多指针(或迭代器)是边读边移动,而不是边写边移动;很多函数参数是只读不写的,使用const限定符不仅可以限制参数被修改,还可以扩大函数可以接受参数的范围(可以接受const对象作为参数)。

如何使用const限定符?

1、 变量:const 类型 变量名  例如: const int ia = 5;

2、 引用:const 类型 &引用名 例如:const int &ib = ia;

3、 对象:const 类名 对象名

4、 指针:const 类型 * 指针名 (常量指针); 类型 * const 指针名 (指针常量)

5、 数组: const 类型 数组名[Arr_Max]

6、 成员函数: 类名::函数名(形参表) const

PS: 上述几种情况中,除 指针常量 和 const修饰成员函数外,其他几种情况的 const  和 类型或者类名(类是一种自定义的类型)的位置可以互换, 比如: const int ia = 5; 与 int const ia = 5; 等同。 建议坚持使用一种风格。

const 限定符几种用法详解:

1、 指针

使用指针时涉及到两个对象:该指针本身和指针指向的对象。根据这两个对象是否使用const修饰,将会产生以下四种情况:

指向非常量对象的非常量指针 int *ptr;

指向非常量对象的常量指针   int *const ptr1;

指向常量对象的非常量指针   const int *ptr2;

指向常量对象的常量指针     int const *const ptr3;

从右向左解读声明:

Ptr1 is a const pointer to int.       ptr1 是一个const指针,指向int对象。

Ptr2 is a pointer to const int.   ptr2是一个指针,指向const int 对象。

Ptr3 is a const pointer to const int.   ptr3是一个const指针,指向const int 对象。

如果指针本身被修饰为常量,则该指针在定义时就应该初始化,且之后都不能变更指向的对象(即不能给指针赋值),但是可以通过解引用修改指向对象的值。

例如:

int i = 10, j = 20;

int *const ia = &i;  // 定义时必须初始化

*ia = 100;   //正确,可以修改指向变量的值

ia = &j;    //错误,不允许给const修饰的指针赋值

ia = &i;     //错误,不允许给const修饰的指针赋值(即使是赋同样的值)

如果指针指向的对象被修饰为常量,则不能在对指针进行解引用之后修改对象的值,但可以变更指向的对象。

例如:

int i = 10, j = 20;

const int *ia = &i;

*ia = 100;  // 错误,不允许通过指针修改指向对象的值

ia = &j;     // 正确, 指针本身未被const修饰,可以变更指向对象

PS: 试图通过ia 修改指向变量的值是不被允许的,但并不是说明变量的值不能被修改,我们依然可以通过其他方式修改变量的值,比如直接赋值i = 100。变量的值能不能修改取决于该变量本身。 const修饰指针指向的变量,只代表从该指针的角度来看,这个变量具有const特性,不能通过该指针来修改。也完全可能通过另一个指针来修改,比如: int *ib = &i; *ib = 100 ;

如果指针指向的对象本身具体const特性,则被指向对象的值不允许被修改,包括直接修改或者间接修改(通过指针或者引用修改)。 因此,C++语言强制要求指向const对象的指针也必须具有const特性,否则会出现编译错误(C++ primer 中文第四版)。 这意味着,只有指向常量对象的指针才能指向一个const变量。

例如:

const int i = 10;

int *ia = &i ;    //编译错误

const int *ib = &i;  //正确

int *const ic = &i;  //编译错误

const int *const id = &i; //正确

2、 引用

一个变量的引用即一个变量的别名,引用本身不能再次赋值让一个引用成为另一个变量的别名。因此,相对于指针,const在修饰引用变量的时候就只需考虑能否通过引用变量来修改绑定变量的值。即只存在两种引用:const变量的引用 和 非const变量的引用。在其他特性方面,引用和指针一致。

例如:

int i = 10;

const int j = 20;

const int &ra = i;  //正确, 表示不能通过ra去修改i的值

int &rb = j;  //编译错误,只有const变量的引用才能与const对象绑定

const int &rc = j;  //正确

int &const rc = i;  //语法错误,不同于指针,引用不能这样定义

PS:在程序加载的时候,系统将内存划分为5个区域:堆区、栈区、全局区(静态区)、文字常量区和程序代码区。对于const修饰的变量,系统没有划定专门的区域来保护其中的数据不被修改。也就是说,使用常变量的方式对数据进行保护是通过编译器作语法限制来实现的。我们依然可以绕过编译器的限制去修改被const限定为“常量”的区域。例如:C语音中通过指针间接修改const变量的值,C++中借助volatile关键字修改const变量的值。 但是C++标准中,对于修改const变量属于”未定义行为”,其实际结果取决于各种编译器的实现。因此,不建议修改去const变量的值(这不废话嘛,如果要修改,定义为const干什么)。

3、 形参

const限定符修饰形参,本质上也是修饰各种变量。在调用函数的时候,实参到形参采用的是值传递。

对于非引用形参,无论是否有const修饰,都可以给函数传递const实参或者非const实参,实际传递的都是实参的一个副本。区别仅在于形参在函数内部使用的时候需要遵循相应的变量读写规则。特别是,尽管函数的形参定义为const,但编译器也将其视为普通的类型。

例如同时定义函数 int Fib(int n)  和int Fib(const int n),编译器将会报错:error C2084:函数“int Fib(int)”已有主体。

这种用法是为了支持对C语言的向下兼容,因为在C语音中,具有const形参或非const形参的函数并无区别(C++ primer 中文第四版)。

复制实参具有一定的局限性,并不是在所有的情况下都适用,不适用的情况包括但不限于:需要在函数值修改实参的值; 复制大型对象所付出的时间和空间代价过大; 对象无法实现复制。针对上述情况应该将形参定义为引用或者指针类型。

对于引用形参,如果函数具有非const引用形参,则不能通过const对象调用函数。毕竟,此时函数可以修改传递进来的对象,这样就违背了实参的const特性。如果函数不会去修改实参的值,则应该将相应的形参定义为const引用,这样形参才能接受const实参,否则会毫无必要的限制该函数的使用。

4、 成员函数

const限定符可以修饰类的成员函数,一般放在函数体后,形如:int fun() const;

   const实际修饰了成员函数隐含的this指针形参,使this指向的对象本身具有const性质。因此,const成员函数将无法修改数据成员(mutable修饰的数据成员除外),也无法调用其他非const成员函数。任何不会修改数据成员的成员函数都应该声明为const类型,这样在不慎修改到数据成员或者调用到非const成员函数时编译器将会报错。

5、函数返回值

  如果函数返回的是一个左值(如引用类型、指针类型),那么该返回值可以被const修饰。形如:

 const string& shortString(const string &str1, const string &str2)
{
return str1.size() < str2.size() ? str1 : str2;
}

  其返回的变量是const string& 类型,因此函数的返回值也必须具有const性质。

  

浅谈const限定符的更多相关文章

  1. 简谈const限定符

    const修饰的数据类型是常量类型,常量类型的对象和变量在定义初始化后是不能被更新的.其实只用记住这一个概念,就可以明白const操作对象的方法. 1)定义const常量 最简单的: const in ...

  2. C++const限定符

    在C语言中我们使用#define宏定义的方式来处理符号常量.而在C++中有一种更好的处理符号常量的方法,那就是使用const关键字来修改变量声明和初始化.这种处理常量方式的好处不言而喻:如果程序在多处 ...

  3. const 限定符

    1.定义const对象 const限定符把一个对象转换成一个常量 const int Bufsize = 512; 定义Bufsize 为常量并初始化为512.变量Bufsize仍然是一个左值,但是不 ...

  4. C++杂谈(一)const限定符与const指针

    const限定符 c++有了新的const关键字,用来定义常变量,可以替C语言中的#define.关于const限定符,有以下需要注意: 1.创建后值不再改变 2.作用范围在文件内有效 3.添加ext ...

  5. C++ Primer 第二章 引用 指针 const限定符

    1.引用: 为对象起了另外一个名字,引用类型引用另外一种类型,通过将声明符写成&d的形式来定义引用类型,其中d也就是声明的变量名(声明符就是变量名). PS:1.通过图片中编译所提示的报错信息 ...

  6. 指针和Const限定符

    指针和Const限定符 1.指向const对象的指针 如果指针指向的是const对象,则不允许使用指针来改变其所指的const值.C++要求指向const对象的指针具有const特性. const d ...

  7. C++ const 限定符

    C++ const 限定符 作用:把一个对象转换成一个常量 用法:const type name = value; 性质:1. 定义时必须初始化,定义后不能被修改.2. 类中的const成员变量必须通 ...

  8. const限定符的作用

    const限定符的作用:                                     1.定义const常量:const可以将一个对象变成一个常量,不可被修改,所以定义的 时候必须进行初始 ...

  9. C++之const限定符

    作者:tongqingliu 转载请注明出处: C++之const限定符 const初始化 const的特点: 用const加以限定的变量,无法改变. 由于const对象定义之后就无法改变,所以必须对 ...

随机推荐

  1. EntityFramework.DynamicFilters 实现软删除和租户过滤

    EntityFramework.DynamicFilters 实现软删除和租户过滤

  2. LInux中的物理内存管理

    2017-02-23 一.伙伴系统 LInux下用伙伴系统管理物理内存页,伙伴系统得益于其良好的算法,一定程度上可以避免外部碎片为何这么说?先回顾下Linux下虚拟地址空间的分布. 在X86架构下,系 ...

  3. GNU Screen使用入门

    前些天开始学习使用GNU Screen程序,发现这个工具在管理服务器时候确实挺方便的,于是写一篇文章总结一下,顺便介绍Screen的基本使用方法. 简介 GNU Screen是 一个基于文本的全屏窗口 ...

  4. 对Numpy数组按axis运算的理解

    Python的Numpy数组运算中,有时会出现按axis进行运算的情况,如 >>> x = np.array([[1, 1], [2, 2]]) >>> x arr ...

  5. vim的快捷键

    vim的快捷键 (〇)3中模式之间的切换 默认在命令模式. i键,从命令模式,进入插入模式. 冒号(:)键,从命令模式,进入末行模式. Esc键,从插入模式.末行模式,进入命令模式. (一)命令模式下 ...

  6. 002-java语言基础

    一.安装卸载 卸载:控制面板 安装:下载对应版本 注意1.安装路径→尽量不要有空格和汉字 注意2.安装之后,jre可以不用安装,jdk中含有 二.环境变量 环境变量:理解,一些快捷路径.方便快速查找应 ...

  7. Spark 2.2 DataFrame的一些算子操作

    Spark Session中的DataFrame类似于一张关系型数据表.在关系型数据库中对单表或进行的查询操作,在DataFrame中都可以通过调用其API接口来实现. 可以参考,Scala提供的Da ...

  8. PAT 1052 Linked List Sorting [一般]

    1052 Linked List Sorting (25 分) A linked list consists of a series of structures, which are not nece ...

  9. RF是如何工作的?

    随机森林的发展史 谈及随机森林算法的产生与发展,我们必须回溯到20世纪80年代.可以说,该算法是Leo Breiman, Adele Cutler, Ho Tin Kam, Dietterich, A ...

  10. PHP压缩与解压Zip(PHPZip类)

    <?php     class PHPZip     {         private $ctrl_dir     = array();         private $datasec    ...