首页 > 编程语言 >C++系列:const关键字

C++系列:const关键字

时间:2024-03-04 17:59:47浏览次数:24  
标签:const 常量 修改 int C++ 关键字 指针

前言

在学习C++时,const关键字的知识点分散在书的各个章节。当我们尝试在编程时使用const时,总会感觉有一些细节被遗忘,因而不能得心应手地使用const关键字。因此,本篇文章尝试着对const关键字的做一些总结。参考书籍《C++ Primer Plus》

const总结

这里是我做的关于const关键字的一些总结,之后的各章便是对书中知识点的理解。

  • const限定符创建的常量不可再次修改
  • 创建常量时记得初始化
  • const创建的常量可以用来声明数组长度。
  • const int * p;p指向常量,p可修改,*p不可修改。
  • int * const p;, p指向变量,p不可修改,*p可修改。
  • const指针可以接受const数据和非const数据。
  • const指针仅可以接收非const数据。
  • 不允许将非const指针的地址赋值给const指针。
  • const引用创建临时变量的两种情况。
  • const全局变量是内部链接性,如static。可用extern更改链接性。可在头文件中使用。
  • cv-限定符。
  • const成员函数,void show() const;,表示函数不会修改调用对象(类成员)。

const限定符

const关键字是C++中较为常用的一个关键字。当我们想创建一个符号常量时,按照C语言的习惯,我们一般会使用#define这种预处理器方法,例如#define ZERO 0。但在C++中,提供了一种更好的处理符号常量的方法,那就是const关键字。

创建常量

创建一个常量的通用格式:const type name = value;
例如:

const int zero = 0; // 一个普通的常量·

这个例子中,被const修饰过的变量zero会变为常量。常量zero被初始化后,其值就被固定了,C++编译器不允许再次修改常量的值。

常量初始化

这里有一点需要注意:在用const声明一个变量时,需要进行初始化。例如下面的代码是错误的:

const int zero;//声明常量时需要进行初始化,否则zero的值未知。
zero = 0;//因为C++编译器不允许再次修改常量的值。所以此处错误。

另外,常量可以用来作为声明名数组时的元素数目。例如:

const int ten = 10;//创建了一个常量。
int array[ten];//用常量创建数组。

一级指针与const

在用const修饰指针时则会出现一些很微妙的地方。在C++中,可以用两种不同的方式将const关键字用于指针。第一种方法是让指针指向一个常量,第二种方法是让指针本身就是常量。其中,第一种方法可以防止使用该指针来修改所指向的值。而第二种方法可以防止改变指针所指的位置。例如:

int value = 0;
const int * p1 = &value; //第一种用法,防止利用p1修改value的值。
int * const p2 = &value; //第二种用法,p2本身不能再修改了。

这里有一个特殊情况令人在意。如果将一个指向变量的指针指向一个常量会发生什么?代码如下:

const int value = 0;
int * p2 = &value; //这是错误的,C++禁止这样的行为。

我们发现,value是常量。但p2是指向value的,那么我们可以通过* p2value的值进行修改。可是做这样的话const的作用就失效了。
事实上,C++禁止这样的用法。也就是说,C++禁止将const常量的地址赋值给非const指针。因此,上面的代码是错误的。在逻辑上也很好解释。
我们可以这样理解:为保证常量不可再次修改的属性,我们不能通过指针修改常量,因此,非const指针仅可以接收非const数据。与之类似,因为我们声明了const指针目的是不会通过当前指针修改其指向的数据,因此其指向的数据一直都是安全的,自然const数据和非const数据都可以。因此,const指针可以接受const数据和非const数据。

二级指针使用const的限制

关于二级指针与常量的关系有些复杂,我们来看下面的代码:

const int ** pp;//这是一个二级const指针
int *p;
const int value = 0;//这是一个常量
pp  = &p;  //这里是错误的,虽然高亮没有提示错误。
           //错误C2440:初始化:无法从int **转换为const int **。
*pp = &value;//两个都是常量,赋值没有问题
*p  = 10; //通过p修改了value的值!

如果pp = &p允许的话,那么我们可以通过二级指针绕开const的限制,如上诉代码一样。C++规定,仅当且只有一层间接关系(如指针指向基本数据类型)时,才能将非const地址赋值给const指针。也就是说,C++不允许将非const指针的地址赋值给const指针。
最后,关于const与指针的关系,下面还有几个例子,请看:

const int value =0;//这是常量
const int *p1;//p1可变,*p1不可变
int * const p2 = &value;//p2不可变,*p2可变

const int ** pp3;//pp3可变,*pp3可变,**pp3不可变
int * const * pp4;//pp4可变,*pp4不可变,**pp4可变
int ** const pp5;//pp5不可变,*pp5不可变,**pp5可变
const int * const *const p6 = &p1;//pp6不可变,*pp6不可变,**pp6不可变

const引用

我们在使用函数的时候,一般会使用引用形参。原因就是因为速度快,无需走复制的流程。当我们使用引用的时候,如果实参与引用参数不匹配,那么C++将产生临时变,关于const引用却有需要了解的知识点。
如果引用形参是const,则C++编译器将在下面两种情况下生成临时变量:

  • 实参的类型正确,但不是左值。
  • 实参的类型不正确,但可以转换为正确的类型。

左值:在C语言中,左值最初指的是可出现在赋值语句左边的实体,但现在,常规变量和const变量都视为左值,因为可以通过地址访问它们。
左值例子:变量,数组元素,结构成员,引用,解除引用的指针等。
非左值例子:字面常量(用引号括起的字符串除外,因为它们是地址),包含多项的表达式等。

代码例子如下:

double refcube (const double &ra){return ra * ra * ra;}
double side = 3.0;
long edge = 5L;

double x1 = refcube(edge);//实参类型不正确,但可以转换为正确的类型
double x2 = refcube(7.0);//实参类型正确,但不是左值(字面常量)
double x3 = refcube(side+10.0);//实参类型正确,但不是左值(表达式)

const全局变量

在C++中,const限定符对默认存储类型稍有影响。默认情况下,C++全局变量的链接性是外部的,但const全局变量的链接性为内部的。也就是说,在c++看来,全局const定义就像使用了static说明符一样。
C++这样做有着很多的好处,这意味着每个文件都有自己的一组常量,而不是所有文件共享一组常量。因此我们可以将常量定义到头文件中,这样只要在两个源代码文件中包括同一个头文件,则它们将获得同一组常量。当然,如果我们希望某个常量的链接性为外部的,那么我们可以使用extern关键字来覆盖默认的内部连接线。extern const int states = 50;

cv-限定符

  • const
  • volatile
    const限定符表明,内存被初始化后,程序便不会再对它进行修改。
    volatile限定符表明,即使程序没有对内存单元进行修改,但其中的值也可能发生变化。可能由于硬件的原因,也可能由其他程序修改,如共享数据。这个关键字的作用主要是为了改善编译器的优化能力。
    防止编译器将该值用缓存的方式进行优化。
    mutable限定符也是需要了解的。当我们声明一个数据结构体为常量,而其中某个成员却需要修改时。我们可以利用mutable限定符对需要修改的成员加以修饰。例子如下:
struct Data{
int x;
mutable int y;//声明此成员是可被再次修改的。
};
const Data data = {1,2};//data实例是常量
data.y=3;//但data的y成员可以被再次修改!

上述代码中,dataconst禁止程序修改data的成员,但由于y成员的mutable限定符说明了datay成员不受这种限制,仍然可以被再次修改。

const成员函数

请看如下的代码片段:

class Data{
int x;
public:
void show(){std::cout <<x;};
}
//
const Data data = {1};
data.show();//这里会报错哦!

C++编译器在data.show();会报错,因为show();函数无法确保调用对象不被修改。很有可能show函数修改了data中的成员,因此C++编译器为了保证data不会被再次修改,禁止了这种调用行为。在C++中,解决这类问题的办法是const成员函数。
请看下面的代码:

class Data{
int x;
public:
void show() const {std::cout <<x;};//这里的const!
}
//
const Data data = {1};
data.show();//这里不会报错了

对于show()的声明应该类似于这样:void show() const;
对于show()的定义应该类似于这样:void Data::show() const {...}
这种方式声明和定义的类函数被称为const成员函数,const成员函数表示函数不会修改调用对象(类成员)。
最后,本人才疏学浅,可能会有很多错误,还望诸君见谅。

标签:const,常量,修改,int,C++,关键字,指针
From: https://www.cnblogs.com/TianyiLi-Tone/p/18052254

相关文章

  • (持续更新)c++结构体
    结构体指针作用:通过指针访问结构体中的成员利用操作符->可以通过结构体指针访问结构体属性 1.指针访问单一结构体#include<iostream>#include<string>#include<ctime>usingnamespacestd;structStudent{stringname;intage;intscore;};i......
  • C++ 动态内存
    C++ 动态内存C++程序中的内存分为两个部分:栈:在函数内部声明的所有变量都将占用栈内存。堆:这是程序中未使用的内存,在程序运行时可用于动态分配内存。很多时候,您无法提前预知需要多少内存来存储某个定义变量中的特定信息,所需内存的大小需要在运行时才能确定。在C++中,您可......
  • C++ 异常处理
    菜鸟教程C语言中文网程序的错误大致可以分为三种,分别是语法错误、逻辑错误和运行时错误:1)语法错误在编译和链接阶段就能发现,只有100%符合语法规则的代码才能生成可执行程序。语法错误是最容易发现、最容易定位、最容易排除的错误,程序员最不需要担心的就是这种错误。2)逻辑错......
  • 关于SAP-APP机器-R3trans -d报错-R3trans: /lib64/libstdc++.so.6: version `GLIBCXX_
    在SAP-应用-APP-机器上执行如下命令报错awpxxx03:prdadm270>R3trans-dR3trans:/lib64/libstdc++.so.6:version`GLIBCXX_3.4.26'notfound(requiredbyR3trans) 其实之前,使用过一种方法解决这个问题,可以参考笔者另一篇文章《关于Redhat-Linux中-compat-sap-c++的说......
  • const修饰指针
    #include<iostream>//const修饰指针,分为常量指针和指针常量//常量指针:int*constnum1=&a;指针的指向不能变化,但是里面的值可以变化//指针常量:constint*num2=&a;指针的指向可以变,但是里面的值不能变化//示例:intmain(void){ inta=9; intb=8; //常量......
  • void关键字
    #include<iostream>//void关键字代表空,可以用作函数前声明返回值类型,代表无返回值//例如:voidfun(){}//可以放在函数的形参位置代表不需要参数voidfun(void){}//*****主要是第三种,void*放在形参位置用于接收任意类型的指针。//但是解引用的时候必须强制转换为明确的......
  • 【C++ STL容器set 】std::set 的全方位解析
    装载自知乎(虽然有AI辅助操作,但是确实写得好好):【C++STL容器set】std::set的全方位解析-知乎(zhihu.com)<imgsrc="https://pic3.zhimg.com/v2-cc8068b8931c7f65e9a89717e2ab404e_b.jpg"data-size="normal"data-rawwidth="1024"data-rawheight="1024......
  • 4_C# 中的 ref 关键字有什么作用
    C#中的ref关键字有什么作用?参数在使用ref关键字进行引用传递时,必须在方法调用之前对其进行初始化。ref关键字既可以在进入方法之前初始化参数的值,也可以在方法内部对参数进行修改。ref参数在进入方法时保持原始值,并在方法结束后将值带回到调用处。ref关键字的作用:......
  • 2_virtual关键字的作用
    virtual作用?virtual关键字用于修改方法、属性、索引器或事件声明,并使它们可以在派生类中被重写(使用override关键字对虚方法重写)。如下是虚方法声明和重写虚方法的示例:声明虚方法publicclassAnimal{publicvirtualvoidSpeak(){Console.WriteLine("动......
  • 1_readonly与const区别
    readonly与const区别?readonly关键字(运行时常量):字段可以在声明或构造函数中初始化,常作为运行时常量使用。const关键字(编译时常量):字段只能在该字段的声明时初始化,常作为编译时常量使用过。C#中的readonly和const都是用于声明常量的关键字,但它们之间存在一些关键的区......