详解C/C++中const限定符总结

 更新时间:2020年02月19日 09:33:54   作者:smile_zyk  
const是一种限定符,被const所限定的变量其值不可以被改变。。这篇文章主要介绍了C/C++中const限定符总结,通过实例代码给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下

const限定符

const是一种限定符,被const所限定的变量其值不可以被改变。

const的初始化

由于const一旦创建其值就不能够被改变,所以我们必须对其进行初始化

const int a;//错误,const变量必须进行初始化!
const int b=10;//正确,编译时初始化
const int c=get_size();//正确,运行时初始化

相同类型的变量相互初始化时,不论变量是否被const限定我们都可以进行随意的相互拷贝。因为在拷贝过程中我们只会用到等式右边变量的右值属性,无须在意其是否可以改变。

int m = 5;
const int n = m;
int j = n;

const与指针

顶层const与底层const

对于指针来说,由于其指向另一片内存的特点,有三种不同的const情况,即:

  1. 指向常量的指针(const int *)
  2. 常量指针(int * const)
  3. 指向常量的常量指针(const int *const)

我们一般称符合第一种情况的为具有底层const属性。

符合第二种情况的为具有顶层const属性。

第三种情况兼而有之。

关于带有const的指针的相互赋值(或者初始化)问题

  • 顶层const并不会影响变量间的相互拷贝(原因是顶层const只保证自身的值不会改变,const没有改变自身的变量类型,在拷贝时只是使用该类型的右值)。
  • 如果等号右边是底层const,那么等号左边必须保证为相同的底层const(或者等号右边的类型可以转换成等号左边的类型),否者表达式无法通过编译。

关于底层与顶层const的一些想法

const的底层与顶层属性似乎只在指针上存在。但是c++primer中有这样的代码和注释:

const int ci=1,&cr=ci;
auto b=ci;//b是一个整数(ci的顶层const特性被忽略掉了)
auto c=cr;//c是一个整数(cr是ci的别名,ci本身是一个顶层const)

这段代码是为了说明auto说明符一般会忽略掉顶层const的特性,在注释中明确写着 ci本身是一个顶层const 。

这也与我的看法一致,底层与顶层const实际上并不是指针所特有的,只要是不能改变对象自身的对象都具有顶层const,而不能改变自己所指向的对象的对象都具有底层const。

从这个角度看,引用实际上自带顶层const。

底层const的隐式转换

上面提到,只有在等号右边和等号左边的类型具有相同的底层const属性,才可以进行赋值或者初始化。

然而有些时候等号右边可能并不具有和等号左边一致的底层const却依然可以成立,这是因为等号右边的类型发生了隐式转换从而具有了和等号左边类型相同底层const属性。

例如:

int i=5;
int *p=&i;
const int *cp=p//int*隐式转换称为了const int*

为什么int 转换成const int 被设定为合法的呢,因为在将int 转换为const int 的过程中用户的权限变小了,在这一转换过程中并不会使程序变得不可靠。

由此我们可以得知非底层const的指针是可以通过隐式转换转变成底层const的。

const与引用

可以把引用绑定在const的变量上,称为const的引用,对常量的引用。

与普通的引用不同的是,对常量的引用不能被用作修改它所绑定的对象

const int ci=5;
const int &r=ci;
r=6//错误不可以通过常引用来修改值
int &r2=ci//错误,试图让一个非常量引用指向一个常量对象。

const引用的初始化

我们知道对于引用来说初始化时一定要用一个对象初始化,且该对象的类型需要与之匹配。

但是const的引用是个例外,在初始化常量引用时允许用任意表达式作为初始值,只要该表达式的结果能转换成引用的类型即可,甚至允许为一个常量引用绑定非常量的对象、字面值或者是表达式。

int i=42;
const int &r1=i;//允许将const int&绑定到一个普通int对象上
const int &r2=3.14;//正确:r2是一个常量引用
const int &r3=r1*2;//正确:r3是一个常量引用
int &r4 =r1*2;//错误,非常量引用不能用表达式初始化

C++primer中给出了可以这么做的原因:

要想理解这种例外情况的原因,最简单的方法是弄清楚当一个常量引用被绑定到另外一种类型上都时到底发生了什么:

double dval=3.14;
const int &ri=dval;

此处ri引用了一个int型的数。对ri的操作应该是整数运算,但dval却是一个双精度浮点数而非整数。因此为了确保让ri绑定一个整数,编译器把上述代码变成了如下形式:

const int temp=dval;//由双精度浮点数生成一个临时的整型常量
const int &ri=temp;//让ri绑定这个临时量

在这种情况下,ri绑定了一个临时量对象。所谓临时量对象就是当编译器需要一个空间来暂存表达式的求值结果时临时创建的一个未命名的对象。C++程序员们常常把临时量对象简称为临时量。

const与auto类型说明符

auto类型说明符是C++11中新引入的类型说明符,可以自动推断类型。

编译器推断出来的auto类型有时候和初始值的类型并不完全一样,编译器会适当的改变结果类型使其更符合初始化规则。

auto在推断带有const的对象时,编译器一般会忽略掉顶层const,同时底层const则会保留下来。 另外对于引用由于引用没有真正的实体,所以如果用一个引用来初始化auto类型时,auto实际上为引用所指向的对象的类型,而非引用,如果要说明其为引用类型,需要使用auto&。

const int ci=i,&cr=ci;
auto b=ci;//b是一个整数(ci的顶层const特性被忽略掉了)
auto c=cr;//c是一个整数(cr是ci的别名,ci本身是一个顶层const)
auto d=&i;//d是一个整型指针(整数的地址就是指向整数的指针)
auto e=&ci;//e是一个指向整数常量的指针(对常量对象取地址是一种底层const)

如果希望推断出的auto类型是一个顶层const,需要明确指出:

const auto f=ci;//ci的推演类型为int,f是const int。

总结

以上所述是小编给大家介绍的C/C++中const限定符总结,希望对大家有所帮助,也非常感谢大家对脚本之家网站的支持!

相关文章

  • C语言字符串数组详解

    C语言字符串数组详解

    这篇文章主要介绍了C语言字符串数组详解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-08-08
  • C语言实现会员管理系统

    C语言实现会员管理系统

    这篇文章主要为大家详细介绍了C语言实现会员管理系统,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-03-03
  • C++ explicit关键字的使用详解

    C++ explicit关键字的使用详解

    这篇文章主要介绍了C++ explicit关键字的使用详解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-09-09
  • C++ 指针常量和常量指针的区别小结

    C++ 指针常量和常量指针的区别小结

    在C++学习使用过程中,每个人都不可避免地使用指针,而且都或多或少的接触过常量指针或指针常量,本文主要介绍了C++ 指针常量和常量指针的区别小结,感兴趣的可以了解一下
    2023-12-12
  • Python绘图之详解matplotlib

    Python绘图之详解matplotlib

    这篇文章主要介绍了Python绘图之详解matplotlib,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-07-07
  • C++ lambda函数详解

    C++ lambda函数详解

    小编可以明确告诉大家:lambda函数是C++11中最重要的,使用最广泛的,最具现代风格的内容,lambda函数的出现改变了C++编程的思维方式。所以快和小编学习一下C++11中lambda函数的使用吧
    2023-02-02
  • C语言实现扫雷小游戏(适合初学者)

    C语言实现扫雷小游戏(适合初学者)

    这篇文章主要为大家详细介绍了C语言实现扫雷小游戏,适合初学者练习,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-03-03
  • c++ 类函数作为模板参数实现方式详解

    c++ 类函数作为模板参数实现方式详解

    这篇文章主要介绍了c++ 类函数作为模板参数实现方式,在实现中加入增强逻辑,这种方式对代码侵入性过高,而且无法控制该逻辑是否需要,如果不需要的话又得重新修改代码实现,需要的朋友可以参考下
    2023-03-03
  • c++多线程之死锁的发生的情况解析(包含两个归纳,6个示例)

    c++多线程之死锁的发生的情况解析(包含两个归纳,6个示例)

    这篇文章主要介绍了c++多线程之死锁的发生的情况解析(包含两个归纳,6个示例),需要的朋友可以参考下
    2018-01-01
  • 详解C/C++ Linux出错处理函数(strerror与perror)的使用

    详解C/C++ Linux出错处理函数(strerror与perror)的使用

    我们知道,系统函数调用不能保证每次都成功,必须进行出错处理,这样一方面可以保证程序逻辑正常,另一方面可以迅速得到故障信息。本文主要为大家介绍两个出错处理函数(strerror、perror)的使用,需要的可以参考一下
    2023-01-01

最新评论