const 是左结合的,若左边为空,则再向右结合

一、指向  const  对象的指针
指向  const  对象的指针,指的是指针指向的对象的内容是const的,不可修改,但指针本身(即指针的值)是可以修改的。
例1:
char acConst[] = "const";
const char *pcConst = acConst; 
*pcConst = ‘C’;         /* Error,像这样修改pcConst指向内容的做法是不允许的 */
pcConst = “const too”;  /* OK,像这样修改pcConst的值是允许的 */
 
二、const  指针 
const  指针,指的是指针本身(即指针的值)不可修改,但是,指针指向的对象是可以修改的。
例2:
char acConst[] = "const";
char *const pcConst = acConst; 
*pcConst = ‘C’;         /* OK,像这样修改pcConst指向内容的做法是允许的, 修改后,acConst的内容变为"Const" */
pcConst = “const too”;  /* Error,像这样修改pcConst的值是不允许的 */
 
上面2种写法是比较容易区分的,下面这个你可能就要混淆了。
char const *pcConst
这个等价于const char *pcConst,还是等价于char *const pcConst?
答案是等价于const char *pcConst。
 
我对const的理解:
const 是左结合的,若左边为空,则再向右结合。
结合上面2个例子:
①const char *pcConst,这里const的左边为空,则右结合,与char结合。说明char是不可改变的,这样,你就能理解例1中*pcConst = ‘C’;这句为什么错了(‘c’是char型的,所以是不能被修改的哦)。
②char const *pcConst,这里const左边不为空,故左结合,所以还是与char结合,所以char const *pcConst等价于const char *pcConst。
③char *const pcConst,这里const左边不为空,故左结合,所以,const修饰的是char *这个指针,所以指针的值是不可变的,指针的内容是可以变的(见例2)。
 
这下你记住了吧^_^,那再给你个难的。
例3:
typedef char *pcString; 
const pcString pcStr;
请问这个pcStr,到底是等价于const char *pcStr呢还是char const *pcStr呢?我将在留言1楼中给出解答。
 
 
 
 
 
 
 
类型声明中const用来修饰一个常量,有如下两种写法.

   1)、const在前面   
const int nValue; //nValue是const   
const char *pContent; //*pContent是const, pContent可变   
const (char *) pContent;//pContent是const,*pContent可变   
char* const pContent; //pContent是const,*pContent可变   
const char* const pContent; //pContent和*pContent都是const    2)、const在后面,与上面的声明对等   
int const nValue; // nValue是const   
char const * pContent;// *pContent是const, pContent可变   
(char *) const pContent;//pContent是const,*pContent可变   
char* const pContent;// pContent是const,*pContent可变   
char const* const pContent;// pContent和*pContent都是const 3)、const与指针
int me;   const int * p1=&me;//p1可变,*p1不可变,此时不能用*p1来修改,但是p1可以转向   
int * const p2=&me;//p2不可变,*p2可变,此时允许*p2来修改其值,但是p2不能转向。   
const int *const p3=&me;//p3不可变,*p3也不可变,此时既不能用*p3来修改其值,也不能转向 const使用:   
1. 用于指针的两种情况:const是一个左结合的类型修饰符.    int const *A; //A可变,*A不可变   
int *const A; //A不可变,*A可变    2.限定函数的传递值参数:    void function(const int Var); //传递过来的参数在函数内不可以改变.   
3.限定函数返回值型.    const int function(); //此时const无意义   
const myclassname function(); //函数返回自定义类型myclassname.    4限定函数类型.    void function()const; //常成员函数, Const成员函数不能改变对象的成员函数。   
例如:   int Point::GetY()   
{   
return yVal;
  }    这个函数被调用时,不改变Point对象,而下面的函数改变Point对象:    void Point:: SetPt (int x, int y)   
{   
xVal=x;   
yVal=y;
  }   
为了使成员函数的意义更加清楚,我们可在不改变对象的成员函数的函数原型中加上const说明:   
class Point   
{   
public:   
int GetX() const;   
int GetY() const;   
void SetPt (int, int);   
void OffsetPt (int, int);   
private:   
int xVal,
yVal;   
};    const成员函数应该在函数原型说明和函数定义中都增加const限定:    int Point::GetY() const   
{   
return yVal;   }    class Set
{   
public:   
Set (void)
{ card = 0; }   
bool Member(const int) const;   
void AddElem(const int);   //...   };   
bool Set::Member (const int elem) const   
{   //...   }    非常量成员函数不能被常量成员对象调用,因为它可能企图修改常量的数据成员:    const Set s;   
s.AddElem(10); // 非法: AddElem不是常量成员函数   
s.Member(10); // 正确    *******但构造函数和析构函数对这个规则例外,它们从不定义为常量成员,但可被常量对象调用(被自动调用)。它们也能给常量的数据成员赋值,除非数据成员本身是常量。 当const所在代码段中不包含括号时,沿着*号划一条线,如果const位于*的左侧,则const就是用来修饰指针所指向的变量,即指针指向为常量;如果const位于*的右侧,const就是修饰指针本身,即指针本身是常量。你可以根据这个规则来看上面声明的实际意义,相信定会一目了然。   
另外,需要注意:对于const (char *) ; 因为char *是一个整体,相当于一个类型(如 char),因此,这时限定指针是const。    一个简单的判断方法:指针运算符*,是从右到左,那么如:char const * pContent,可以理解为char const (* pContent),即* pContent为const,而pContent则是可变的。 我们定义的类的成员函数中,常常有一些成员函数不改变类的数据成员,也就是说,这些函数是"只读"函数,而有一些函数要修改类数据成员的值。如果把不改变数据成员的函数都加上const关键字进行标识,显然,可提高程序的可读性。其实,它还能提高程序的可靠性,已定义成const的成员函数,一旦企图修改数据成员的值,则编译器按错误处理。 const成员函数和const对象    实际上,const成员函数还有另外一项作用,即常量对象相关。对于内置的数据类型,我们可以定义它们的常量,用户自定义的类也一样,可以定义它们的常量对象。 例如,定义一个整型常量的方法为:    const int i=1 ;    同样,也可以定义常量对象,假定有一个类classA,定义该类的常量对象的方法为:    const classA a(2);    这里,a是类classA的一个const对象,"2"传给它的构造函数参数。const对象的数据成员在对象寿命期内不能改变。但是,如何保证该类的数据成员不被改变呢?   为了确保const对象的数据成员不会被改变,在C++中,const对象只能调用const成员函数。如果一个成员函数实际上没有对数据成员作任何形式的修改,但是它没有被const关键字限定的,也不能被常量对象调用。下面通过一个例子来说明这个问题:    class C   
{   
int X;   
public:   
int GetX()   {   return X;   }   
void SetX(int X)   {   this->X = X;   }   };   
void main()   {   const C constC;   cout<<constC.GetX();   }    如果我们编译上面的程序代码,编译器会出现错误提示:constC是个常量对象,它只能调用const成员函数。虽然GetX( )函数实际上并没有改变数据成员X,由于没有const关键字限定,所以仍旧不能被constC对象调用。如果我们将上述代码中:    int GetX()   改写成:   int GetX()const   再重新编译,就没有问题了。    const成员函数的使用    const成员函数表示该成员函数只能读类数据成员,而不能修改类成员数据。定义const成员函数时,把const关键字放在函数的参数表和函数体之间。有人可能会问:为什么不将const放在函数声明前呢?因为这样做意味着函数的返回值是常量,意义完全不同。下面是定义const成员函数的一个实例:    class X   
{   int i;   
public:   
int f() const;   };    关键字const必须用同样的方式重复出现在函数实现里,否则编译器会把它看成一个不同的函数:    int X::f() const   
{   return i;   }    如果f( )试图用任何方式改变i或调用另一个非const成员函数,编译器将给出错误信息。任何不修改成员数据的函数都应该声明为const函数,这样有助于提高程序的可读性和可靠性。

【转】const 是左结合的,若左边为空,则再向右结合的更多相关文章

  1. Codeforces Round #577 (Div 2)

    A. Important Exam 水题 #include<iostream> #include<string.h> #include<algorithm> #in ...

  2. NOIP算法总结

    前言 离NOIP还有一个星期,匆忙的把寒假整理的算法补充完善,看着当时的整理觉得那时还年少.第二页贴了几张从贴吧里找来的图片,看着就很热血的.旁边的同学都劝我不要再放PASCAL啊什么的了,毕竟我们的 ...

  3. LeetCode 笔记总结

    前言 之前把一些LeetCode题目的思路写在了本子上,现在把这些全都放到博客上,以后翻阅比较方便. 题目 99.Recover Binary Search Tree 题意 Two elements ...

  4. 冲刺NOIP复习,算法知识点总结

    前言        离NOIP还有一个星期,匆忙的把整理的算法补充完善,看着当时的整理觉得那时还年少.第二页贴了几张从贴吧里找来的图片,看着就很热血的.当年来学这个竞赛就是为了兴趣,感受计算机之美的. ...

  5. POJ 3667 Hotel(线段树+区间合并)

    http://poj.org/problem?id=3667 题意: 有N个房间,M次操作.有两种操作(1)"1a",表示找到连续的长度为a的空房间,如果有多解,优先左边的,即表示 ...

  6. Codeforces #263 div2 解题报告

    比赛链接:http://codeforces.com/contest/462 这次比赛的时候,刚刚注冊的时候非常想好好的做一下,可是网上喝了个小酒之后.也就迷迷糊糊地看了题目,做了几题.一觉醒来发现r ...

  7. UVALive 5903 Piece it together(二分图匹配)

    给你一个n*m的矩阵,每个点为'B'或'W'或'.'.然后你有一种碎片.碎片可以旋转,问可否用这种碎片精确覆盖矩阵.N,M<=500 WB  <==碎片 W 题目一看,感觉是精确覆盖(最近 ...

  8. hdu 1698:Just a Hook(线段树,区间更新)

    Just a Hook Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total ...

  9. hdu 1754:I Hate It(线段树,入门题,RMQ问题)

    I Hate It Time Limit: 9000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total S ...

随机推荐

  1. 利用SQL语句查询一个数据库中的所有表

    SQL  :  select * from information_schema.tables ORACLE: select table_name from user_tables ACCESS: s ...

  2. dos基本命令

    dir   :列出当前目录下的文件及文件夹 md  :插件目录 rd   :删除目录 cd   :进入指定目录 cd.. :退回到上一级目录 cd/  :退回到根目录 del   :删除文件 exit ...

  3. editplus利用正则表达式快速定位

    例如我要找到user_jj表保存数据的语句 做法:editplus选择正则表达式输入 user_jj.*save 就可以定位到:M('user_jj')->where(array('tgbz_i ...

  4. libyuv 编译 for android

    libyuv is an open source project that includes is an instrumentation framework for building dynamic ...

  5. SU suchart命令学习

  6. HDU4081 Qin Shi Huang's National Road System(次小生成树)

    枚举作为magic road的边,然后求出A/B. A/B得在大概O(1)的时间复杂度求出,关键是B,B是包含magic road的最小生成树. 这么求得: 先在原图求MST,边总和记为s,顺便求出M ...

  7. ural 1222. Chernobyl’ Eagles

    1222. Chernobyl’ Eagles Time limit: 1.0 secondMemory limit: 64 MB A Chernobyl’ eagle has several hea ...

  8. 初识API函数

    我之前是一个只会编写数值计算的程序的OIer,但我并不甘于这种现状,于是我编写了我的第一个使用API函数的C++程序,开发平台是VS2012: // ConsoleApplication.cpp : ...

  9. CSS3 box-shadow快速教程

    box-shadow 属性向框添加一个或多个阴影.这个CSS3的属性很常用,盒阴影.按钮状态等各种地方都有用到,但是你了解并记住各个参数的作用及用法吗? 展示 源码:http://codepen.io ...

  10. C#多线程与异步的区别

    随着拥有多个硬线程CPU(超线程.双核)的普及,多线程和异步操作等并发程序设计方法也受到了更多的关注和讨论.本文主要是想与园中各位高手一同探讨一下如何使用并发来最大化程序的性能. 多线程和异步操作的异 ...