C语言中const和指针的秘密你知道吗

 更新时间:2022年02月16日 15:37:57   作者:叫我小秦就好了  
这篇文章主要为大家详细介绍了C语言中const和指针的秘密,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助

指针的赋值

问题

将一个类型为 char** 的值赋值给一个 const char** 类型的对象是否合法呢?

先说结果,在vs的环境下,编译器不会报错也不会有任何警告。

但在linux环境下用gcc编译就会出现下面的警告:

warning: assignment from incompatible pointer type

警告:来自不兼容指针类型的赋值

为了代码的可移植性我们显然不能写出这样的代码,下面就让我们一步步探索这其中的奥妙。

首先来了解ANSI C有关标准

ANSI C 有关简单赋值的标准

要使赋值形式合法,必须满足下列条件之一:

  • 1.两个操作数都是指向有限定符或无限定符的相容类型的指针
  • 2. 左边指针所指向的类型必须具有右边指针所指向类型的全部限定符

还有一个关于类型的说明:

const float* 类型不是一个有限定符的类型——它的类型是 “指向一个具有 const 限定符的 float 类型的指针”,也就是说 const 限定符是修饰指针所指向的类型,而不是指针本身。

问题解决

在解决问题之前,我们先来看一组简单的。

char* 和 const char*

char* 和 const char* 是匹配的。它之所以合法,是因为在下面的代码中:

char* cp;
const char* cpp;
cpp = cp; 
  • 左操作数是一个指向有const限定符的char的指针;
  • 右操作数是一个指向没有限定符的char的指针;
  • char类型与char类型是相容的,左操作数所指向的类型具有右操作数所指向类型的限定符(无),再加上自身的限定符(const)。

注意,反过来就不能进行赋值。

char* cp;
const char* cpp;
cp = cpp; //此时左操作数不具有右操作数的const限定符

char** 和 const char**

由上面的知识我们可以得知,char** 和 const char** 都是没有限定符的指针类型,但他们的指向的类型不一样(前者指向char*,后者指向const char*),这违反了上面赋值标准的第一条,所以它们是不相容的。

用这种方式理解这个有一点困难。可以用下面这种方法进行理解:

char** pp1;
const char** pp2;
pp2 = pp1;
  • 左操作数的类型是 const char**,它是一个指向 const char* 类型的指针,而 const char* 是一个没有限定符的指针,它指向一个带有 const 限定的 char 类型;
  • 右操作数的类型是 char**,它是一个指向 char* 的指针,而 char* 是一个没有限定符的指针,它指向一个没有限定符的 char 类型。

const char* 和 char* 是相容的,而且他们本身没有限定符,所以符合标准的约束条件,两者之间的赋值是合法的。但 const char** 和 char** 之间的关系又有不同,虽然二者都没有限定符,但二者所指向的对象类型不相容,所以不能进行赋值。

const修饰

const修饰变量

首先,关键字const并不能把变量变成常量!在一个符号前加上const限定符只是表示这个符号不能被赋值。也就是说const修饰的变量是只读的,不可以被直接修改,但它不能防止被间接修改。

例如:

#include <stdio.h>

int main()
{
	const int i = 10;
	int* p = &i;
	printf("before:%d\n", i);
	*p = 20;
	printf("after:%d\n", i);//这里打印值变成了20,说明可以间接修改

	return 0;
}

const修饰指针

const修饰指针变量有多种位置,下面我们将逐个介绍。

const int* p

注:const int* p 与 int const p 写法不同,作用是一样的。
这种写法的意思是:const修饰p,不能通过解引用(p)的方式直接修改所指向的变量,但可以通过改变指针指向的方式来修改p。
例如:

#include <stdio.h>

int main()
{
	//通过下方直接解引用的方式来修改编译器会直接报错
	//int i = 10;
	//const int* p = &i;
	//*p = 20;

	int i = 10;
	const int* p = &i;
	printf("before:%d\n", *p);
	int j = 20;
	p = &j;//通过这样改变p的指向,可以间接修改*p值
	printf("after:%d\n", *p);
	
	return 0;
}

int* const p

这种写法的意思是:const修饰p,不能通过改变指针指向的方式修改*p的值,但可以通过解引用(*p)的方式直接修改所指向的变量。

例如:

#include <stdio.h>

int main()
{
	int i = 10;
	int* const p = &i;
	printf("before:%d\n", *p);
	*p = 20;//不能改变p的指向,但可以直接解引用修改值
	printf("after:%d\n", *p);

	return 0;
}

const int* const p

这种写法是同时修饰p和*p,既不能改变p的指向,也不能用解引用直接修改。

总结

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

相关文章

  • C++中的throw关键字详解

    C++中的throw关键字详解

    throw关键字是在C语言中用来抛出异常的关键字,它通常与try和catch一起使用,用于在程序中发生错误时进行异常处理,当遇到无法处理的错误情况时,我们可以使用throw关键字主动抛出异常,所以本文给大家详细的介绍一下C++中的throw关键字,需要的朋友可以参考下
    2023-09-09
  • C++ 约瑟夫环问题案例详解

    C++ 约瑟夫环问题案例详解

    这篇文章主要介绍了C++ 约瑟夫环问题案例详解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-08-08
  • 浅谈c++ stl迭代器失效的问题

    浅谈c++ stl迭代器失效的问题

    下面小编就为大家带来一篇浅谈c++ stl迭代器失效的问题。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧,祝大家游戏愉快哦
    2016-12-12
  • 一文带你了解C++中的右值引用与移动语义

    一文带你了解C++中的右值引用与移动语义

    本篇文章主要为大家详细介绍了C++中的右值引用与移动语义的相关知识,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2023-03-03
  • 贪心算法 WOODEN STICKS 实例代码

    贪心算法 WOODEN STICKS 实例代码

    贪心算法 WOODEN STICKS 实例代码,需要的朋友可以参考一下
    2013-05-05
  • C++实现LeetCode(19.移除链表倒数第N个节点)

    C++实现LeetCode(19.移除链表倒数第N个节点)

    这篇文章主要介绍了C++实现LeetCode(19.移除链表倒数第N个节点),本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-07-07
  • Ubuntu 20.04 下安装配置 VScode 的 C/C++ 开发环境(图文教程)

    Ubuntu 20.04 下安装配置 VScode 的 C/C++ 开发环境(图文教程)

    这篇文章主要介绍了Ubuntu 20.04 下安装配置 VScode 的 C/C++ 开发环境,本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-05-05
  • C++ vector数组用法及解析

    C++ vector数组用法及解析

    这篇文章主要给大家分享的是C++ vector数组用法及解析,什么是什么是vector数组呢?下面文章将对打家做详细介绍,感兴趣的小伙伴可以参考一下
    2021-10-10
  • C++迷宫问题的求解算法

    C++迷宫问题的求解算法

    这篇文章主要为大家详细介绍了C++迷宫问题的求解算法,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-03-03
  • 基于重启后消失的注册表键值的详细介绍

    基于重启后消失的注册表键值的详细介绍

    本篇文章是对重启后消失的注册表键值进行了详细的分析介绍,需要的朋友参考下
    2013-05-05

最新评论