作者:tongqingliu

转载请注明出处:http://www.cnblogs.com/liutongqing/p/7050815.html

C++之const限定符(顶层const,底层const)

const初始化

const的特点:

  • 用const加以限定的变量,无法改变。
  • 由于const对象定义之后就无法改变,所以必须对其进行初始化。
  • const对象的常量特征仅在尝试改变它的时候表现出来,其他时候和变量无异。

const初始化:

const int bufSize = 512; //bufSize无法再改变

const仅在本文件中有效

const对象通常只在本文件内有效,如果希望其在其他文件中也有效,则需要在其前面加上extern关键字。更详细的做法是,在一个文件中定义const,在其他多个文件中声明并使用它。

extern const int bufSize = 512;

顶层const和底层const

首先,const是一个限定符,被它修饰的变量的值不能改变。对于一般的变量来说,其实没有顶层const和底层const的区别,而只有对于指针这类复合类型的基本变量,才有这样的区别。

如何区分顶层const和底层const?

顶层const表示指针本身是个常量;

底层const表示指针所指向的对象是个常量。

指针如果添加const修饰符时有两种情况:

  • 指向常量的指针:代表不能改变其指向内容的指针。声明时const可以放在类型名前后都可,拿int类型来说,声明时:const int和int const 是等价的。声明指向常量的指针也就是底层const,下面举一个例子:
int num_a = 1;
int const *p_a = &num_a; //等价于const int *p_a = &num_a,指向const int 类型的指针,是底层const
//*p_a = 2; //错误,指向“常量”的指针不能改变所指的对象

注意:指向“常量”的指针不代表它所指向的内容一定是常量,只是代表不能通过解引用符(操作符*)来改变它所指向的内容。

上例中指针p_a指向的内容就不是常量,可以通过赋值语句:num_a=2; 来改变它所指向的内容。

  • 常量指针:代表指针本身是常量,声明时必须初始化,之后它存储的地址值就不能再改变。声明时const必须放在指针符号后面,即:const 。声明常量指针就是顶层const,下面举一个例子:
int num_b = 2;
int *const p_b = &num_b; //指向int类型的const指针,是顶层const
//p_b = &num_a; //错误,常量指针不能改变存储的地址值

其实顶层const和底层const很简单,一个指针本身添加const限定符就是顶层const,而指针所指的对象添加const限定符就是底层const。

区分顶层const和底层const的作用

为啥非要区分顶层const和底层const呢,根据C++primer的解释,区分后有两个作用。

1 执行对象拷贝时有限制,常量的底层const不能赋值给非常量的底层const。也就是说,你只要能正确区分顶层const和底层const,你就能避免这样的赋值错误。下面举一个例子:

int num_c = 3;
const int *p_c = &num_c; //指向const int的指针,是底层指针
//int *p_d = p_c; //错误,不能将底层const指针赋值给非顶层const指针
const int *p_d = p_c; //正确,都是指向const int的指针

2 使用命名的强制类型转换函数const_cast时,需要能够分辨底层const和顶层const,因为const_cast只能改变运算对象的底层const。下面举一个例子:

int num_e = 4;
const int *p_e = &num_e;
//*p_e = 5; //错误,不能改变底层const指针指向的内容
int *p_f = const_cast<int *>(p_e); //正确,const_cast可以改变运算对象的底层const。但是使用时一定要知道num_e不是const的类型。
*p_f = 5; //正确,非顶层const指针可以改变指向的内容
cout << num_e; //输出5

3练习

说了这么多,应该练习一下,const int constconst* pppi 是顶层const还是底层const?

答案当然是底层const,因为int前面const限定符,而最后一个*后面没有const限定符。看最后一个例子:

const int a = 1;  //a是顶层const
//int * pi = &a; //错误,&a是底层const,不能赋值给非底层const
const int * pi = &a; //正确,&a是底层const,可以赋值给底层const
const int *const *const ppi = &pi //即是底层const,也是顶层const
const int *const *const *pppi = &ppi; //底层const

指针和const限定符(另一版本理解方法)

  • 指向const对象的指针
  • const指针
  • 指向const对象的const指针
const double *p;// 指向const double类型的指针
double *const p = &pi; //指向double对象的const指针
const double *const p = &pi;//指向const对象的const指针

下面看几个例子:

指向const对象的指针

#include<iostream>
using namespace std;
int main()
{
double a = 1.2;
double *p = &a; //p是指向变量a的指针 const double pi = 3.14;
//p = &pi;//错,要想指向一个const对象,必须用指向const对象的指针
const double *cptr = &pi; //const double类型的指针
//*cptr = 1.5; //错,指向const对象的指针只能指向const对象,因而不能修改其值 system("pause");
}

const指针

指向const对象的const指针

#include<iostream>
using namespace std;
int main()
{
double a = 1.2;
double *p = &a; //p是指向变量a的指针 const double pi = 3.14;
//p = &pi;//错,要想指向一个const对象,必须用指向const对象的指针
const double *cptr = &pi;
cptr = &a; //指向const对象的指针也能指向非const对象
//*cptr = 1.5; //但是不能通过指针对其进行修改 int errNum = 0;
int *const curErr = &errNum; //指向int对象的const指针,必须进行初始化,而且该const指针不能再指向其他对象 const double *const pi_ptr = &pi; //指向const double对象的const指针
//不可以再指向其他的对象,而且也不可以通过指针修改对象的值 system("pause");
}

const写在左边也行,写在右边也行。

const string str1;
string const str2;

上面的两种写法都是对的。

参考:

C++ primer 第五版

http://blog.csdn.net/qq_19528953/article/details/50922303

C++之const限定符的更多相关文章

  1. C++const限定符

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

  2. const 限定符

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

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

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

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

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

  5. 指针和Const限定符

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

  6. C++ const 限定符

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

  7. const限定符的作用

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

  8. 变量和基本类型——复合类型,const限定符,处理类型

    一.复合类型 复合类型是指基于其他类型定义的类型.C++语言有几种复合类型,包括引用和指针. 1.引用 引用并非对象,它只是为一个已存在的对象所起的另外一个名字. 除了以下2种情况,其他所有引用的类型 ...

  9. C++之const限定符(顶层const,底层const)

    作者:tongqingliu 转载请注明出处:http://www.cnblogs.com/liutongqing/p/7050815.html C++之const限定符(顶层const,底层cons ...

随机推荐

  1. poj1990树状数组

    Every year, Farmer John's N (1 <= N <= 20,000) cows attend "MooFest",a social gather ...

  2. 玩玩kafka1 单机安装

    今天主要来一遍kafka单机版的安装,比较简单易上手,关于kafka的介绍我就不贴了,大家可以自行搜索 1.首先将tgz包传到centos目录下(这里使用xftp工具) ok后查看一下tgz包是否已经 ...

  3. Java中Properties类

    1 简介: JDK提供的java.util.Properties类继承自Hashtable类并且实现了Map接口,用map来存储key-value数据,所以存入的数据是无序的.其中键和值都是字符串类型 ...

  4. 生产环境中使用Docker Swarm的一些建议

    译者按: 实践中会发现,生产环境中使用单个Docker节点是远远不够的,搭建Docker集群势在必行.然而,面对Kubernetes, Mesos以及Swarm等众多容器集群系统,我们该如何选择呢?它 ...

  5. FME中通过HTMLExtractor向HTML要数据

    如何不断扩充数据中心的数据规模,提升数据挖掘的价值,这是我们思考的问题,数据一方面来自于内部生产,一部分数据可以来自于互联网,互联网上的数据体量庞大,形态多样,之前blog里很多FMEer已经提出了方 ...

  6. linux防火墙基本操作

    1.查看防火墙运行状态 # firewall-cmd --state 或者 # systemctl status firewalld.service .关闭防火墙 # systemctl stop f ...

  7. mysql中 date datetime time timestamp 的区别

    MySQL中关于时间的数据类型:它们分别是 date.datetime.time.timestamp.year date :"yyyy-mm-dd"  日期     1000-01 ...

  8. 0基础搭建Hadoop大数据处理-编程

    Hadoop的编程可以是在Linux环境或Winows环境中,在此以Windows环境为示例,以Eclipse工具为主(也可以用IDEA).网上也有很多开发的文章,在此也参考他们的内容只作简单的介绍和 ...

  9. windows上安装jdk

    1.下载jdk安装包 jdk官网   http://www.oracle.com/technetwork/java/javase/downloads/index.html 2.安装jdk 傻瓜式下一步 ...

  10. hive网站日志数据分析

    一.说在前面的话 上一篇,楼主介绍了使用flume集群来模拟网站产生的日志数据收集到hdfs.但我们所采集的日志数据是不规则的,同时也包含了许多无用的日志.当需要分析一些核心指标来满足系统业务决策的时 ...