深入理解const char*p,char const*p,char *const p,const char **p,char const**p,char *const*p,char**const p

C C++ Heller 318℃ 0评论

一、可能的组合:

     (1)const char*p

     (2)char const*p

     (3)char *const p

     (4)const char **p

     (5)char const**p

     (6)char *const *p

     (7)char **const p

     当然还有在(5)、(6)、(7)中再插入一个const的若干情况,不过分析了以上7中,其他的就可类推了!

二、理解助记法宝:

     1。关键看const 修饰谁。

     2。由于没有const *的运算,若出现const*的形式,则const实际上是修饰前面的。

     比如:char const*p,由于没有const*运算,则const实际上是修饰前面的char,因此char const*p等价于const char*p。也就是说上面7种情况中,(1)和(2)等价。 同理,(4)和(5)等价。在(6)中,由于没有const*运算,const实际上修饰的是前面的char*,但不能在定义时转换写成 const(char *)*p,因为在定义是”()”是表示函数。

三、深入理解7种组合

     (1)char *p:可以对任意位置   (非系统 敏感区域)进 行读操作和写操作  

     (2)const char*p或者char const *p(因为没有const*p运算,因此const修饰的还是前面的char):可以对任意位置(非系统敏感区域)进行“只读”  操作。(“只读”是相对于char *p来说所限定的内容)

     (3)char *const p(const 修饰的是p):只能对“某个固定的位置”   进 行读写操作,并且在定义p时就必须初始化(因为在后面不能执行“p=..”的操作,因此就不能在后面初始化,因此只能在定义时初始化)。(“某个固定的位 置”是相对于char *p来说所限定的内容)
可以总结以上3点为:char *p中的指针p通常是”万 能”的工作指针   , 而(2)和(3)只是在(1)的基础上加了些特定的限制   ,这 些限制在程序中并不是必须的,只是为了防止程序员的粗心大意而产生事与愿违的错 误。
另外,要明白“每块内存空间都可有名字;都有每块内存空间 内容皆可变(除非有所限)  ”  。比如函数里定义的char s[]=”hello”;事实上在进程的栈内存里开辟了7个变量共10个字节的空间,其中6个字符变量的名字分别为:s1[0]、s1[1]、 s1[2]、s1[3]、s1[4]、s1[5](内容是’\0’) ,他们共占了6个字节,还有一个4字节的指针变量s 。不过s是“有所限制”的,属于char *const类型,也就是前面说的   (3)这种情况,s一直指向s[0],即(*s==s[0]==’h’),可以通过*s=’k’来改变s所指向的 s[0]的值,但不能执行(char *h=“aaa”;s=h;)来对s另外赋值。

     (4)上面的(2)和(3)只是对p进行限定,没有也不能对p所指向的空间进行限定,对于”char s[]=”hello”;const char*p=s;” 不能通过*(s+i)=’x’来修改数组元素s[0]~s[4]的值,但可以通过s[i]=’x’来修改原数组元素的值。

     (5)const char **p或者char const**p :涉及两个指针p和 *p。由于const修饰char ,对指针p没有任何限定,对指针*p进行了上面情况(2)的限定。

    (6)char *const *p:涉及两个指针p和 *p。由于const修饰前面的char*,也就是对p所指向的内容*p进行了限定(也属于前面的情况(2))。而对*p来说,由于不能通 过”*p=…”来进行另外赋值,因此属于前面的情况(3)的限定。

    (7)char **const p : 涉及两个指针p和 *p,const修饰p,对p进行上面情况(3)的限定,而对*p,没有任何限定。

四、关于char **p 、const char **p的类型相容性问题

    1。问题

        char *p1;const *p2=p1;//合法

        char **p1;const  char**p2=p1;//不合法,会有警告warning: initialization from incompatible pointer type

        char **p1;char const**p2=p1;//不合法,会有警告warning: initialization from incompatible pointer type

        char**p1;char*const*p2=p1;//合法

    2。判断规则

        明确const修饰的对象!对于指针p1,和p2,若要使得p2=p1成立,则可读做  

             “p1是指向X类型的指 针,p2是指向“带有const限定”的X类型的指针   “。

        char *p1;const *p2=p1;//合法:p1是指向(char)类型的指针,p2是指向“带有const限定”的(char)类型的指针。

        char **p1;const  char**p2=p1;//不合法:p1是指向(char*)类型的指针,p2是指向 ((const char)*)类型的指针。

        char **p1;char const**p2=p1;//不合法;与上等价。

        char**p1;char*const*p2=p1;//合法:  p1是指向(char *)类型的指针,p2是指向“带有const限定”的(char*)类型的指针。

五、其他

    1。 含有const的单层或双层指针的统一读法:

         “p是一个指针,是一个[“带有 const限定”的]指向[”带有const限定”的]X类型的指针”。

    2。定义时const修饰的对象是确定的,但不能在定义时加括号,不然就和定义时用“()”表示的函数类型相混淆了!因此定义时不能写(char *)const *p或者(const char) **p。


原文链接:http://www.cnblogs.com/rushuizhijing/archive/2011/08/26/2154737.html


转载请注明:无名小站 » 深入理解const char*p,char const*p,char *const p,const char **p,char const**p,char *const*p,char**const p

喜欢 (0)or分享 (0)
发表我的评论
取消评论

表情

Hi,您需要填写昵称和邮箱!

  • 昵称 (必填)
  • 邮箱 (必填)
  • 网址