指针常量和常量指针的一些整理

Posted upstreamL

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了指针常量和常量指针的一些整理相关的知识,希望对你有一定的参考价值。

近来学习指针,碰到一个关于指针常量和常量指针的问题,之前被弄得是稀里糊

涂的,今天特意查资料整理了一下,现拿出来和大家分享,有什么弄错的地方请

大家不吝赐教,共同进步。

首先常量指针,咱们可以这样理解,常量指针即是一个指向常量的指针,既然指

向的是一个常量,那它所指向的值当然就不能变了。但是它本身的值,即它的地

址是可以变的,咱们可以把它指向别的地址。

而指针常量,则可以理解为一个常量,什么时候常量??指针常量,即指针是常

量,则它的地址是不可变的,但是它所指向的地址里的值是可以变的。

一个int型的常量指针是这样定义的:int const* 指针名。int const(常量)*(

指针)指针名
一个int型的指针常量是这样定义的:int* const 指针名。int*(指针) const(常

量)指针名
观察上面两个的定义我们是不是可以根据*和const的位置来一眼判断是一个指针

常量还是一个常量指针呢???*前const后是指针常量,const前*后是常量指针!!

!呵呵,当然最重要的还是要理解它们所代表的含义,哪个能改,哪个不能!

现在可以通过下面的一段代码再来看看自们理解的怎么样了。试试看下面的代码

哪里有问题,为什么!

#include<stdio.h>
int main(){
  int i=0;
  int j=1;
  int x;
  const int * pi=&i;
  *pi=22;
  pi=&j;
  int * const pj=&j;
  *pj=22;
  pj=&i;
  const int * px=&x;
  const int h=2;
  const int* ph=&h;
  int* const ph2=&h;
}

 

理解:

#include<stdio.h>
int main(){
  int i=0;
  int j=1;
  int x;
  const int * pi=&i;//定义一个常量指针pi,它指向的地址是i
  *pi=22;//有问题了,pi是一个常量指针,它指向的地址的值你竟敢改
  pi=&j;//没问题,pi虽然指向的是一个常量,但本身还是自由的,还是可以改

变的
  int * const pj=&j;//这里定义一个指针常量,指向的是j的地址。
  *pj=22;//没问题,指针常量只是指针是个常量,地址是常量,但我里面的内容

可以变啊,好歹也有点自由啊
  pj=&i;//这里就不对了,既然是个常量就乖乖地别换地址
  const int * px=&x;//这里也没问题,可能就有人会有疑问了,你px不是一个

常量指针吗,你不是说常量指针是一个指向常量的指针吗,x又不是一个常量,杂

就可以呢?呵呵,这个是我之前有疑问,后来我把px所指向的值输出来,然后再

试图给px赋值,但不行,x已经被赋上了一个垃圾值。
所以说的常量指针是一个指向常量的指针这仅仅是便于我们理解,事实上它还是

可以指向一个变量的,不过一旦定义后该变量也就成了一个不变的量了。
  const int h=2;
  int* const ph2=&h;//有问题,我上面定义了h是一个常量,你怎么可以用一个

指针变量来指向h呢。若是可以的话,你指针常不是可以修改所指向地址的内容,

那不意味着可以修改h的值了,可h是个常量,怎么能让你这么做!!!
  const int* ph=&h;//所以还是用常量指针吧,指向的地址的值不变。
}

其实我觉得就是中文翻译的比较故弄玄虚。

你所谓的常量指针,其实英文是 pointer to const,经常翻译成就叫指向常量的指针。其实这么叫就不那么容易混淆。
另外指针常量,英文是 const pointer,经常也能看见翻译成常指针的,要是这么叫反而感觉容易和上面那个搞混。

而且有关 constant 这个词,做为术语,有一定的歧义。有的时候,它只包含诸如 1, 1L, "abc", 12.5 \'c\' 这样的东西。有的时候他也指 const variable。就是指 const int 这类东西,声明一个变量,但看成常量。(常见的是翻译成“常量变量”或者“常变量”,不过按照你的那个命名逻辑应该叫“变量常量”)。而且很多时候在上下文里其实仅指后者,因为前者比较简单,通常不需要考虑。

后来标准委员会换了一个词,用 literal 来代表 1, \'c\' 这样的东西,一般被翻译成“字面”。比如前者叫整形字面,后者叫字符字面。而常量变量就统称为常量,比如 const int a,就说声明了一个整形常量(const integer)。但中文不知道是不是习惯问题,也有把 literal 这个词翻译成 字面常量 的。

总之翻译还是很混乱的。不过楼主讲的还算清楚,只要别死抠这两个中文术语就行了。

另外我也看到过另一种记忆的方法:
const int * const p;
看成 const int (* const p)。把 * 前的理解成加完 * 之后的东西,括号里的理解成未加星的东西。
当出现比如 : *p = 5; 这样的语句的时候,由于有星了,那么到括号外头找,左边的东西本质上是 const int。
而 p = 5; 找括号里面的是 const p,那么这就是给一个常量赋值,也不行。

这种记法对复杂一点的东西也有用:

程序代码:
int f() { return 0; }

int main()
{
    int (*p)(void);
    p = &f;

    int a[5];
    int (*q)[5];
    q = &a;

    return 0;
}

int (*p)(void)。 p 是一个指针,而 *p 是 int _ (void),所以是个指向函数的指针。
int (*q)[5]。q 是一个指针,而 *q 是 int [5],所以 q 是个指向 int [5] 这样数组的指针。
当然不管怎么记,复杂的东西还是会比较复杂。

以上是关于指针常量和常量指针的一些整理的主要内容,如果未能解决你的问题,请参考以下文章

指针和引用指针常量与常量指针

指针常量和常量指针

常(量)指针和指针常量

指针常量和常量指针

常量指针(pointer to constant)和指针常量(constant pointer)

指针常量和常量指针