详解C语言中的常量指针和指针常量_C 语言

概述
对于新手来说,指针在c语言里总是一个非常难以理解的概念。在这篇文章中,我们将解释常量指针,指针常量,const pointer to const(ps:楼主以为这可以翻译成指向常量的常量指针)的区别

常量指针
让我们先来理解什么是常量指针。常量指针是指指针指向的地址是常量。换句话说,一旦常量指针指向了一个变量,你不能让该常量指针指向其他变量了

常量指针的声明方法如下:

  <type of pointer> * const <name of pointer> 

常量指针声明示例:

  int * const ptr; 

让我们用一小段代码来说明常量指针:

  #include <stdio.h> 

  int main(void)
  {
    int var1 = 0, var2 = 0; 

    int * const ptr = &var1; 

    ptr = &var2; 

    printf("%d\n", *ptr); 

    return 0;
  } 

在上面的例子中:

  •     我们定义了两个变量var1和var2
  •     声明一个常量指针ptr,并让其指向var1
  •     然后,我们让ptr指向var2
  •     最后,我们试图打印这个指针指向的内容

简而言之,我们定义了一个常量指针,并且试图修改常量指针指向的地址

现在,让我们来编译这个程序:

所以,一旦常量指针指向了某一地址,我们不能更改常量指针的地址(ps:但是可以修改常量指针指向的内容)

指针常量
从名字中就可以明显得出,一个指针,我们无法修改指针指向的内容,这种指针就叫做指针常量。对于这类指针,你可以修改指针指向的地址,但是不能修改指针指向的内容

指针常量的定义如下:

  const <type of pointer> * <name of pointer> 

或者

  <type of pointer> const * <name of pointer> 

指针常量示例如下:

  const int *ptr; 

  int const *ptr; 

让我们用一小段代码来解释指针常量:

   

#include <stdio.h> 

  int main(void)
  {
    int var1 = 0; 

    const int *ptr = &var1; 

    *ptr = 2; 

    printf("%d\n", *ptr); 

    return 0;
  }

在上面代码里:

  •     我们定义了一个变量var1,并将其赋值为0
  •     我们定义了一个指针常量并让它指向变量var1
  •     我们试图修改指针常量指向的值
  •     打印指针常量指向的值

然后我们编译上面的程序:

我们可以看到*ptr是只读属性。这意味着,如果指针常量指向了一个变量,我们不能修改该指针常量指向的值

指向常量的常量指针
如果你理解了上面两种指针类型,那作为上述两种指针的混合形态,你也应该非常好理解。指向常量的常量指针,你既不能修改指针的地址,也不能修改指针指向的内容

指向常量的常量指针定义如下:

  const <type of pointer> * const <name of pointer> 

示例:

  const int * const ptr; 

让我们写一小段代码来理解指向常量的常量指针:

  #include <stdio.h> 

  int main(void)
  {
    int var1 = 0, var2 = 0; 

    const int * const ptr = &var1; 

    *ptr = 1; 

    ptr = &var2; 

    printf("%d\n", *ptr); 

    return 0;
  } 

在上面代码中:

  •     我们声明了两个变量var1和var2
  •     我们声明了一个指向常量的常量指针ptr,并让ptr指向变量var1
  •     我们试图修改指针的地址和指针指向的内容

当我们编译这段代码的时候:

而C/C++中常把指针和常量混合起来使用,其最大的用途就是作为函数的形式参数,保证实参在被调函数中的不可改变的特性,那到底常量指针和指针常量有什么区别呢?

总结
好了,现在来总结一下 常量指针 和 指针常量 的区别

首先一定要明白哪种定义方式是常量指针,哪种是指针常量,这里可以记住三句话加深记忆:

* (指针)和 const(常量) 谁在前先读谁 ;*象征着地址,const象征着内容;谁在前面谁就不允许改变。

好吧,让我们来看这个例子:

int a =3;
int b = 1;
int c = 2;
int const *p1 = &b;//const 在前,定义为常量指针
int *const p2 = &c;//*在前,定义为指针常量

常量指针p1:指向的地址可以变,但内容不可以重新赋值,内容的改变只能通过修改地址指向后变换。  

  •     p1 = &a是正确的,但 *p1 = a是错误的。
  • 指针常量p2:指向的地址不可以重新赋值,但内容可以改变,必须初始化,地址跟随一生。
  •     p2= &a是错误的,而*p2 = a 是正确的。
  • 对于常量指针p1,我们可以改变它指向的地址,但不能改变指向的内容,如果改变了,就会出错,下面是18行代码取消注释后编译器提示的错误:

下面是在Vim编辑器中的调试结果

上述代码在注释 18行 和 24行 代码后才能正确输出,下图是正确结果

输出结果可以看出,对于常量指针p1,改变其地址指向,内容也随着地址的改变而变化了。

而对于指针常量p2,初始化后地址就固定了,内容可以随时重新赋值。

对于常量指针p1,我们可以改变它指向的地址,但不能改变指向的内容,如果改变了,就会出错,下面是18行代码取消注释后编译器提示的错误:

输出结果可以看出,对于常量指针p1,改变其地址指向,内容也随着地址的改变而变化了。

而对于指针常量p2,初始化后地址就固定了,内容可以随时重新赋值。

对于常量指针p1,我们可以改变它指向的地址,但不能改变指向的内容,如果改变了,就会出错,下面是18行代码取消注释后编译器提示的错误:


经过上面的介绍,我想大家应该知道常量指针和指针常量的区别了。

以上是小编为您精心准备的的内容,在的博客、问答、公众号、人物、课程等栏目也有的相关内容,欢迎继续使用右上角搜索按钮进行搜索c
, 指针常量
常量指针
c语言指针详解、c语言二级指针详解、c语言指针详解视频、c语言指针用法详解、c语言合法常量,以便于您获取更多的相关知识。

时间: 2024-08-04 00:18:29

详解C语言中的常量指针和指针常量_C 语言的相关文章

详解C++程序中定义struct结构体的方法_C 语言

什么是结构体?简单的来说,结构体就是一个可以包含不同数据类型的一个结构,它是一种可以自己定义的数据类型,它的特点和数组主要有两点不同,首先结构体可以在一个结构中声明不同的数据类型,第二相同结构的结构体变量是可以相互赋值的,而数组是做不到的,因为数组是单一数据类型的数据集合,它本身不是数据类型(而结构体是),数组名称是常量指针,所以不可以做为左值进行运算,所以数组之间就不能通过数组名称相互复制了,即使数据类型和数组大小完全相同. 结构体的定义 定义结构体使用struct修饰符,例如: struct

详解C++编程中的sizeof运算符与typeid运算符_C 语言

sizeof 运算符产生与 char 类型的大小有关的操作数大小. 语法 sizeof unary-expression sizeof ( type-name ) 备注 sizeof 运算符的结果为 size_t 类型,它是包含文件 STDDEF.H 中定义的整数类型.利用此运算符,你可以避免在程序中指定依赖于计算机的数据大小. sizeof 的操作数可以是下列项之一: 类型名称.若要将 sizeof 用于类型名称,则该名称必须用括号括起. 一个表达式.当用于表达式时,无论是否使用括号都可指定

详解C++编程中的文件流与字符串流_C 语言

C++文件流类与文件流对象 文件流是以外存文件为输入输出对象的数据流.输出文件流是从内存流向外存文件的数据,输入文件流是从外存文件流向内存的数据.每一个文件流都有一个内存缓冲区与之对应. 请区分文件流与文件的概念,不用误以为文件流是由若干个文件组成的流.文件流本身不是文件,而只是以文件为输入输出对象的流.若要对磁盘文件输入输出,就必须通过文件流来实现. 在C++的I/O类库中定义了几种文件类,专门用于对磁盘文件的输入输出操作. 除了标准输入输出流类istream.ostream和iostream

详解C++编程中表达式的语义与计算顺序_C 语言

表达式根据其运算符的优先级和分组来计算. 计算顺序请看以下示例: // expre_pluslang__pluslang_Order_of_Evaluation.cpp // compile with: /EHsc #include <iostream> using namespace std; int main() { int a = 2, b = 4, c = 9; cout << a + b * c << "\n"; cout <<

详解C++编程中的静态成员与可变数据成员_C 语言

静态成员类可以包含静态成员数据和成员函数.当数据成员被声明为"静态"时,只会为类的所有对象保留一个数据副本. 静态数据成员不是给定的类类型的对象的一部分.因此,静态数据成员的声明不被视为一个定义.在类范围中声明数据成员,但在文件范围内执行定义.这些静态类成员具有外部链接.下面的示例阐释了这一点: // static_data_members.cpp class BufferedOutput { public: // Return number of bytes written by a

详解C语言中的#define宏定义命令用法_C 语言

#define命令#define定义了一个标识符及一个串.在源程序中每次遇到该标识符时,均以定义的串代换它.ANSI标准将标识符定义为宏名,将替换过程称为宏替换.命令的一般形式为: #define identifier string 注意: 1.该语句没有分号.在标识符和串之间可以有任意个空格,串一旦开始,仅由一新行结束. 2.宏名定义后,即可成为其它宏名定义中的一部分. 3.宏替换仅仅是以文本串代替宏标识符,前提是宏标识符必须独立的识别出来,否则不进行替换.例如: #define XYZ th

详解C++编程中用数组名作函数参数的方法_C 语言

C++数组的概念 概括地说:数组是有序数据的集合.要寻找一个数组中的某一个元素必须给出两个要素,即数组名和下标.数组名和下标惟一地标识一个数组中的一个元素. 数组是有类型属性的.同一数组中的每一个元素都必须属于同一数据类型.一个数组在内存中占一片连续的存储单元.如果有一个整型数组a,假设数组的起始地址为2000,则该数组在内存中的存储情况如图所示. 引入数组就不需要在程序中定义大量的变量,大大减少程序中变量的数量,使程序精炼,而且数组含义清楚,使用方便,明确地反映了数据间的联系.许多好的算法都与

详解C语言中的getgrgid()函数和getgrnam()函数_C 语言

C语言getgrgid()函数:从组文件中取得指定gid的数据 头文件: #include <grp.h> #include <sys/types.h> 定义函数: strcut group * getgrgid(gid_t gid); 函数说明:getgrgid()用来依参数gid 指定的组识别码逐一搜索组文件, 找到时便将该组的数据以group 结构返回. 返回值:返回 group 结构数据, 如果返回NULL 则表示已无数据, 或有错误发生. 范例 /* 取得gid=3 的组

C语言中宏定义使用的小细节_C 语言

#pragma#pragma 预处理指令详解 在所有的预处理指令中,#Pragma 指令可能是最复杂的了,它的作用是设定编译器的状态或者是指示编译器完成一些特定的动作.#pragma指令对每个编译器给出了一个方法,在保持与C和 C++语言完全兼容的情况下,给出主机或操作系统专有的特征.依据定义,编译指示是机器或操作系统专有的,且对于每个编译器都是不同的. 其格式一般为: #Pragma Para.............etc.. baike.baidu.com/view/1451188.htm

浅谈c语言中转义字符的用法及注意事项_C 语言

c语言中的转义字符: \a 响铃符 \b 退格 \f 换页符 \n 换行符 \r 回车符(回到该行的首位置) \v 纵向制表符 \\ 反斜杠 \? 问号(?经vs10测试可以直接打印) \"(\') 双引号(单引号) \ooo 八进制数(ooo表示一个用8进制数表示出来的对应ANSII代码对应出字符,用此方法可以表示出所有ASCII字符.不过测试发现打不出%号,存疑!) \xhh 十六进制数(功能同八进制数,用hh表示一个十六进制数,如\x20表示空格)  注:使用转义字符的退格符,换行符,回车