C++和C中const的区别详解

目录
  • C中的const
    • const修饰局部变量
    • const修饰全局变量
    • const修饰的全局变量有外部链接属性
    • const与指针
  • C++中的const
    • const修饰普通全局变量
    • const修饰普通局部变量
    • const与类
  • 总结
  • C中的const
    • const修饰局部变量
    • const修饰全局变量
    • const修饰的全局变量有外部链接属性
    • const与指针
  • C++中的const
    • const修饰普通全局变量
    • const修饰普通局部变量
    • const与类
  • 总结

const,这个词字面意思为:常数。

这就表示这是一个不可以改变是数。同时,这也是C/C++里面的一个关键字,是一个限定符,但是const在C和C++中的用法是有区别的。

由于本人水平有限,难免会有出错的地方,如果错误,还请指正!

C中的const

const修饰局部变量

在C语言中,const修饰局部变量,那么这个变量就是“常变量”。

void test(){
	const int b = 20;
}

int main() {
	const int a = 10;
	return 0;
}

上面的两个变量,无论是主函数中的a,还是普通的函数中的b,都是被从const修饰的变量,那么就是“常变量”。

“常变量”不可以直接通过变量名来对值进行修改,因为变量名被const修饰后,从原来的“可读可改”的属性,变成了只“可读”,“不可改”的属性。

void test(){
	const int b = 20;
	b = 40;//error
}

int main() {
	const int a = 10;
	a = 30;//error
	return 0;
}

上面的行为是错误的。

但是,“常变量”本质上还是一个“变量”,而不是“常量”。

只要是被const修饰的局部变量,都是在程序运行到这一行代码的时候,才会创建这个变量并且分配空间的。

而分配空间是在栈区分配的,栈区的空间都会有对应的地址,栈区的空间是“可读可写”的。

我们可以通过地址,来对值进行修改。

#include<stdio.h>

void test() {
	const int b = 20;
	int* pb = &b;
	*pb = 40;
	printf("%d\n", b);
}

int main() {
	const int a = 10;
	int* pa = &a;
	*pa = 30;
	printf("%d\n", a);
	test();
	return 0;
}

上面的代码输出的结果是:30和40

也就是说,不论是main函数中的还是普通函数中的局部变量,只要是被const修饰的局部变量,是可以通过地址来进行修改的。

补充

一般我们在定义一个被const修饰的变量的时候,都应该定义并且初始化,如果像上面的那样,是被const修饰的局部变量,如果我们在定义的时候不进行初始化,那么就是一个随机值,想要修改就只能通过指针了。

const修饰全局变量

const修饰的全局变量,也就是定义在函数体之外的变量,内存空间是在文字常量区的,这个内存区域是只读的,不能通过变量名去修改变量的值,也不能通过指针去修改变量的值!

const int a = 10;//全局变量
int main() {
	int* pa = &a;
	*pa = 30;
	printf("%d\n", a);
	return 0;
}

上面的代码时错误的,被const修饰的全局变量不能通过变量名和地址对内容进行修改,程序会报错。

const修饰的全局变量有外部链接属性

在C语言中,只要时全局变量,不论有没有被const修饰,都是默认拥有外部链接属性的,也就是说这个全局变量不仅限于在当前文件下使用,只要我们在其他文件中,加上extern的声明,也是可以使用的。

const与指针

当const修饰非指针的普通变量的时候,不论const放在类型关键字前面还是后面,表达的意思都是一样的

#include<stdio.h>
const int c = 5;
int const c = 5;
void test() {
	const int b = 20;
	int const b = 20;
}

int main() {
	const int a = 10;
	int const a = 10;
	return 0;
}

上面a,b,c,三个变量的两种写法表达的意思都是一样的,当然同名的变量不能重复定义,我只是演示一下而已。

当const修饰指针的时候,不同的写法会代表不同的意思。

int main() {
	const int a = 10;
	//const int* pa = &a;//与下一行的代码表达的意思一样
	int const* pa = &a;
	*pa = 30;
	printf("%d\n", a);
	return 0;
}

上面的代码时有错的,const修饰指针的时候,const在*星号的左边(上面演示的两种情况都可以),那么表示的是,指针pa指向的空间的内容不可以修改,但是指针变量本身的值可以修改,也就是该指针变量可以改变指向的空间。

int main() {
	const int a = 10;
	int b = 20;
	int* const pa = &a;
	*pa = 30;
	pa = &b;//error
	printf("%d\n", a);
	return 0;
}

上面的代码是错误的,const在*星号的右边,那么表示的是指针变量pa里面存放的地址不可以被修改,也就是不能修改当前指针变量所指向的空间,但是空间的内容可以通过指针来进行修改。

C++中的const

const修饰普通全局变量

与C一样,当const修饰普通的全局变量的时候,不能通过变量名和地址来修改变量的值。

另外

与C不一样的是,C语言中的const修饰的普通全局变量默认是外部链接属性的,但是在C++中被const修饰的普通全局变量是内部链接属性的。

也就是说当我们在一个文件中定义了一个如下的全局变量

const int a = 10;//定义全局变量
int main() {
	return 0;
}

我们在另外一个文件中,使用extern来声明,也是不可以的。

//另外一个文件
extern const int a;//在另外的文件中声明

上面这种做法是不可以的,C++中被const修饰的全局变量默认是内部链接属性,不能直接在另外的文件中使用,如果想要在另外的文件中使用,就需要在定义该全局的变量的文件中用extern来修饰。

//定义的文件
extern const int a = 10;
//另外一个文件声明
extern const int a;

const修饰普通局部变量

如果const修饰的局部变量是基础的类型(int char double等等),并且初始化使用字面常量的话,不会给该变量分配空间。

例如:

void test() {
	const int a = 10;//用字面常量10来初始化
	a = 20;//error
}

但是,当我们对这个变量进行取地址的操作的时候,系统会为该变量分配空间。

void test() {
	const int a = 10;
	//a = 20;//error
	int* p = (int*)&a;
	*p = 20;
	cout << a << endl;
	cout << *p << endl;
}

上面的结果是:

10和20

这是因为,当我们定义一个被const修饰并且使用字面常量来初始化的局部变量的时候,系统会把这个变量看作是一个符号,放入到符号表中,这么变量名就是一个符号,值就是这个符号的值,类似于#define的作用。

当我们对这个变量取地址的时候,由于原来没有空间,就没有地址,现在需要取地址,所以才被迫分配一块空间,我们通过地址的解引用可以修改这个空间的值,这也就是为什么第二个结果为20的原因,但是如果我们还是通过变量名来访问数据的话,系统会认为这还是一个符号,直接用符号表里面的值替换。

但是!

如果初始化不是用字面常量而是用变量,那么系统会直接分配空间。

void test() {
	int b = 20;
	const int a = b;
}

这时候的a是有空间的,不会被放入到符号表中。

const与类

如果是自定义数据类型(结构体、对象)

我们在创建对象(结构体)的时候,如果这个对象是被const修饰的话,那么不管这个对象是全局的还是局部的,系统都会直接分配空间

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注我们的更多内容!

const,这个词字面意思为:常数。

这就表示这是一个不可以改变是数。同时,这也是C/C++里面的一个关键字,是一个限定符,但是const在C和C++中的用法是有区别的。

由于本人水平有限,难免会有出错的地方,如果错误,还请指正!

C中的const

const修饰局部变量

在C语言中,const修饰局部变量,那么这个变量就是“常变量”。

void test(){
	const int b = 20;
}

int main() {
	const int a = 10;
	return 0;
}

上面的两个变量,无论是主函数中的a,还是普通的函数中的b,都是被从const修饰的变量,那么就是“常变量”。

“常变量”不可以直接通过变量名来对值进行修改,因为变量名被const修饰后,从原来的“可读可改”的属性,变成了只“可读”,“不可改”的属性。

void test(){
	const int b = 20;
	b = 40;//error
}

int main() {
	const int a = 10;
	a = 30;//error
	return 0;
}

上面的行为是错误的。

但是,“常变量”本质上还是一个“变量”,而不是“常量”。

只要是被const修饰的局部变量,都是在程序运行到这一行代码的时候,才会创建这个变量并且分配空间的。

而分配空间是在栈区分配的,栈区的空间都会有对应的地址,栈区的空间是“可读可写”的。

我们可以通过地址,来对值进行修改。

#include<stdio.h>

void test() {
	const int b = 20;
	int* pb = &b;
	*pb = 40;
	printf("%d\n", b);
}

int main() {
	const int a = 10;
	int* pa = &a;
	*pa = 30;
	printf("%d\n", a);
	test();
	return 0;
}

上面的代码输出的结果是:30和40

也就是说,不论是main函数中的还是普通函数中的局部变量,只要是被const修饰的局部变量,是可以通过地址来进行修改的。

补充

一般我们在定义一个被const修饰的变量的时候,都应该定义并且初始化,如果像上面的那样,是被const修饰的局部变量,如果我们在定义的时候不进行初始化,那么就是一个随机值,想要修改就只能通过指针了。

const修饰全局变量

const修饰的全局变量,也就是定义在函数体之外的变量,内存空间是在文字常量区的,这个内存区域是只读的,不能通过变量名去修改变量的值,也不能通过指针去修改变量的值!

const int a = 10;//全局变量
int main() {
	int* pa = &a;
	*pa = 30;
	printf("%d\n", a);
	return 0;
}

上面的代码时错误的,被const修饰的全局变量不能通过变量名和地址对内容进行修改,程序会报错。

const修饰的全局变量有外部链接属性

在C语言中,只要时全局变量,不论有没有被const修饰,都是默认拥有外部链接属性的,也就是说这个全局变量不仅限于在当前文件下使用,只要我们在其他文件中,加上extern的声明,也是可以使用的。

const与指针

当const修饰非指针的普通变量的时候,不论const放在类型关键字前面还是后面,表达的意思都是一样的

#include<stdio.h>
const int c = 5;
int const c = 5;
void test() {
	const int b = 20;
	int const b = 20;
}

int main() {
	const int a = 10;
	int const a = 10;
	return 0;
}

上面a,b,c,三个变量的两种写法表达的意思都是一样的,当然同名的变量不能重复定义,我只是演示一下而已。

当const修饰指针的时候,不同的写法会代表不同的意思。

int main() {
	const int a = 10;
	//const int* pa = &a;//与下一行的代码表达的意思一样
	int const* pa = &a;
	*pa = 30;
	printf("%d\n", a);
	return 0;
}

上面的代码时有错的,const修饰指针的时候,const在*星号的左边(上面演示的两种情况都可以),那么表示的是,指针pa指向的空间的内容不可以修改,但是指针变量本身的值可以修改,也就是该指针变量可以改变指向的空间。

int main() {
	const int a = 10;
	int b = 20;
	int* const pa = &a;
	*pa = 30;
	pa = &b;//error
	printf("%d\n", a);
	return 0;
}

上面的代码是错误的,const在*星号的右边,那么表示的是指针变量pa里面存放的地址不可以被修改,也就是不能修改当前指针变量所指向的空间,但是空间的内容可以通过指针来进行修改。

C++中的const

const修饰普通全局变量

与C一样,当const修饰普通的全局变量的时候,不能通过变量名和地址来修改变量的值。

另外

与C不一样的是,C语言中的const修饰的普通全局变量默认是外部链接属性的,但是在C++中被const修饰的普通全局变量是内部链接属性的。

也就是说当我们在一个文件中定义了一个如下的全局变量

const int a = 10;//定义全局变量
int main() {
	return 0;
}

我们在另外一个文件中,使用extern来声明,也是不可以的。

//另外一个文件
extern const int a;//在另外的文件中声明

上面这种做法是不可以的,C++中被const修饰的全局变量默认是内部链接属性,不能直接在另外的文件中使用,如果想要在另外的文件中使用,就需要在定义该全局的变量的文件中用extern来修饰。

//定义的文件
extern const int a = 10;
//另外一个文件声明
extern const int a;

const修饰普通局部变量

如果const修饰的局部变量是基础的类型(int char double等等),并且初始化使用字面常量的话,不会给该变量分配空间。

例如:

void test() {
	const int a = 10;//用字面常量10来初始化
	a = 20;//error
}

但是,当我们对这个变量进行取地址的操作的时候,系统会为该变量分配空间。

void test() {
	const int a = 10;
	//a = 20;//error
	int* p = (int*)&a;
	*p = 20;
	cout << a << endl;
	cout << *p << endl;
}

上面的结果是:

10和20

这是因为,当我们定义一个被const修饰并且使用字面常量来初始化的局部变量的时候,系统会把这个变量看作是一个符号,放入到符号表中,这么变量名就是一个符号,值就是这个符号的值,类似于#define的作用。

当我们对这个变量取地址的时候,由于原来没有空间,就没有地址,现在需要取地址,所以才被迫分配一块空间,我们通过地址的解引用可以修改这个空间的值,这也就是为什么第二个结果为20的原因,但是如果我们还是通过变量名来访问数据的话,系统会认为这还是一个符号,直接用符号表里面的值替换。

但是!

如果初始化不是用字面常量而是用变量,那么系统会直接分配空间。

void test() {
	int b = 20;
	const int a = b;
}

这时候的a是有空间的,不会被放入到符号表中。

const与类

如果是自定义数据类型(结构体、对象)

我们在创建对象(结构体)的时候,如果这个对象是被const修饰的话,那么不管这个对象是全局的还是局部的,系统都会直接分配空间

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注我们的更多内容!

(0)

相关推荐

  • C++ 中 const和static readonly区别

    C++ 中 const和static readonly区别 我们都知道,const和static readonly的确很像:通过类名而不是对象名进行访问,在程序中只读等等. 在多数情况下可以混用. 二者本质的区别在于,const的值是在编译期间确定的,因此只能在声明时通过常量表达式指定其值.而static readonly是在运行时计算出其值的,所以还可以通过静态构造函数来赋值. 明白了这个本质区别,我们就不难看出下面的语句中static readonly和const能否互换了: 1. stat

  • 详解C++中的const关键字及与C语言中const的区别

    const对象默认为文件的局部变量,与其他变量不同,除非特别说明,在全局作用域的const变量时定义该对象的文件局部变量.此变量只存在于那个文件中中,不能别其他文件访问.要是const变量能在其他文件中访问,必须显示的指定extern(c中也是)   当你只在定义该const常量的文件中使用该常量时,c++不给你的const常量分配空间--这也是c++的一种优化措施,没有必要浪费内存空间来存储一个常量,此时const int c = 0:相当于#define c 0:    当在当前文件之外使用

  • C语言中const和C++中的const 区别详解

    C语言中const和C++中的const 区别详解 C++的const和C语言的#define都可以用来定义常量,二者是有区别的,const是有数据类型的常量,而宏常量没有,编译器可以对前者进行静态类型安全检查,对后者仅是字符替换,没有类型安全检查. 而C语言中的const与C++也有很大的不同,在C语言中用const修饰的变量仍是一个变量,表示这个变量是只读的,不可显示地更改,而在C++中用const修饰过后,就变成常量了.例如下面的代码: const int n=10; int a[n];

  • C++中const char*、char const*、char * const三者的区别

    目录 一.const char *ptr; 二.char const *ptr; 三.char * const ptr; C/C++ 中关于以下三种定义: const char *ptr; char const *ptr; char * const ptr; 整理三者之间的区别与联系. 一.const char *ptr; 定义一个指向字符常量的指针,这里,ptr是一个指向 char* 类型的常量,所以不能用ptr来修改所指向的内容,换句话说,*ptr的值为const,不能修改.但是ptr的声明

  • C++和C中const的区别详解

    目录 C中的const const修饰局部变量 const修饰全局变量 const修饰的全局变量有外部链接属性 const与指针 C++中的const const修饰普通全局变量 const修饰普通局部变量 const与类 总结 C中的const const修饰局部变量 const修饰全局变量 const修饰的全局变量有外部链接属性 const与指针 C++中的const const修饰普通全局变量 const修饰普通局部变量 const与类 总结 const,这个词字面意思为:常数. 这就表示

  • 从go语言中找&和*区别详解

    *和&的区别 :& 是取地址符号 , 即取得某个变量的地址 , 如 ; &a*是指针运算符 , 可以表示一个变量是指针类型 , 也可以表示一个指针变量所指向的存储单元 , 也就是这个地址所存储的值 . 从代码中验证 : 先构建一个Rect类型 : 1. &是取地址符号, 取到Rect类型对象的地址 2. *可以表示一个变量是指针类型(r是一个指针变量): 3.*也可以表示指针类型变量所指向的存储单元 ,也就是这个地址所指向的值 4.查看这个指针变量的地址 , 基本数据类型直

  • C语言中 & 和 &&的区别详解

    这是c语言的基本语法,但是在学习的过程中也总是搞混.所以记录一下,也和大家分享一下. &:按照位与操作,例如:0010&1101,结果为0000 &是java中的位逻辑运算:       eg: 2&3=2: 分析如下: 2的二进制为10 :3的二进制为11 : 逻辑&之后为10 &&:短路与,表示如果两个条件都成立则执行之后的逻辑: 例如:if(a==0&&b==0),意思就是if a为0并且b为0的时候,进行下一步操作. || 短

  • Servlet中/和/*的区别详解

    目录 本文提纲 版本约定 ✍正文 点拨"市面上"的错误答案 1./用于Servlet,/*用于Filter 2./不会匹配.jsp请求,而/*可以匹配到.jsp请求 3./*匹配范围比/大 4./匹配所有url(路径+后缀),/*只匹配路径型 Servlet四种匹配方式 1. 精确匹配 2. 路径匹配 3. 后缀名匹配 4. 缺省匹配 URL匹配注意事项 匹配顺序 /和/*的区别 DispatcherServlet不拦截.jsp请求根因分析 ✍总结 本文提纲 版本约定 JDK:8 Se

  • javaScript中"=="和"==="的区别详解

    区别: ==, 两边值类型不同的时候,要先进行类型转换,再比较. ==,不做类型转换,类型不同的一定不等. 下面分别说明: 先说 "===",这个比较简单.下面的规则用来判断两个值是否===相等: 1.如果类型不同,就不相等 2.如果两个都是数值,并且是同一个值,那么[相等]:(!例外)的是,如果其中至少一个是NaN,那么[不相等].(判断一个值是否是NaN,只能用isNaN()来判断) 3.如果两个都是字符串,每个位置的字符都一样,那么相等:否则不相等 . 4.如果两个值都是true

  • C/C++编程中const的使用详解

    目录 1 概述:const和define的区别 2. 修饰局部变量 3. 常量指针与指针常量 4. 修饰函数的参数 5. 修饰函数的返回值 6. 修饰全局变量 总结 1 概述:const和define的区别 先看一个典型的程序: #include<iostream> using namespace std; int main() { int num = 1; #define t1 num + num #define t2 t1 % t1 cout << "t2 is &q

  • C字符串与C++中string的区别详解

    在C++中则把字符串封装成了一种数据类型string,可以直接声明变量并进行赋值等字符串操作.以下是C字符串和C++中string的区别:  C字符串 string对象(C++) 所需的头文件名称  <string>或<string.h> <string>或<string.h> 需要头文件 原因 为了使用字符串函数 为了使用string类 声明 方式 char name[20]; string name; 初始化方式 char name[20]="

  • MyBatis中#{}和${}的区别详解

    最近在用mybatis,之前用过ibatis,总体来说差不多,不过还是遇到了不少问题,再次记录下. 先给大家介绍下MyBatis中#{}和${}的区别,具体介绍如下: 1. #将传入的数据都当成一个字符串,会对自动传入的数据加一个双引号.如:order by #user_id#,如果传入的值是111,那么解析成sql时的值为order by "111", 如果传入的值是id,则解析成的sql为order by "id". 2. $将传入的数据直接显示生成在sql中.

  • Mybatis中#{}与${}的区别详解

    前言 在开发中使用Mybatis经常使用到#{}与${},依旧有很多开发者对二者的使用不是很清晰,正所谓好记性不如烂笔头,特此总结一下. 在mybatis中动态 sql 是其主要特性之一,在 mapper 中定义的参数传到 xml 中之后,在执行操作之前 mybatis 会对其进行动态解析.mybatis 提供了两种支持动态 sql 的语法:#{} 以及 $ { }, 其最大的区别则是前者方式能够很大程度防止sql注入(安全),后者方式无法防止Sql注入 .什么??不懂什么是Sql注入?额...

  • mysql中#{}和${}的区别详解

    #{}会将传入的数据当成一个字符串,会对自动传入的数据加一个双引号 order by #{userId}   这里假如userId = 111,那么解析成sql时会变成 order by "111"这里如果userId = idStr,那么解析成sql时会变成 order by "idStr" ${}会将传入的数据直接显示生成在sql中 order by #{userId}  这里假如userId = 111,那么解析成sql时会变成 order by 111这里如

随机推荐