CPP常识07 -- const在函数前与函数后的区别

const在函数前与函数后的区别

转载自http://blog.csdn.net/clozxy/article/details/5679887

const基础

如果const关键字不涉及到指针,我们很好理解,下面是涉及到指针的情况:

int b =  500;   
const int* a = &b;              // [1]   
int const *a = &b;              // [2]   
int* const a = &b;              // [3]   
const int* const a = &b;        // [4]   

如果你能区分出上述四种情况,那么,恭喜你,你已经迈出了可喜的一步。不知道,也没关系,我们可以参考《effective c++》 item 03上的做法,如果const位于星号的左侧,则const就是用来修饰指针所指向的变量,即指针指向为常量;如果const位于星号的 右侧,const就是修饰指针本身,即指针本身是常量。因此,[1]和[2]的情况相同,都是指针所指向的内容为常量,这种情况下不允许对内容进行更改操 作,如不能*a = 3 ;[3]为指针本身是常量,而指针所指向的内容不是常量,这种情况下不能对指针本身进行更改操作,如a++是错误的;[4]为指针本身和指向的内容均为常量。

另外const的一些强大的功能在于它在函数声明中的应用。在一个函数声明中,const可以修饰函数的返回值,或某个参数;对于成员函数,还可以修饰是整个函数。有如下几种情况,以下会逐渐的说明用法:

A& operator=(const A& a);   
void fun0(const A* a);   
void fun1() const;   // fun1()为类成员函数   
const A fun2();   

const的初始化

先看一下const变量初始化的情况

  1. 非指针const常量初始化的情况:
A b;
const A a = b;   
  1. 指针(引用)const常量初始化的情况:
A* d = new A(); 
const A* c = d; // c指向一个常量,也就是说,通过c不能对其指向的对象做任何修改.

或者:const A* c = new A();

引用:

A f;
const A& e = f; // 这样作e只能访问声明为const的函数,而不能访问一般的成员函数; 

[思考1]: 以下的这种赋值方法正确吗?

const A* c = new A();   
A* e = c;   

[思考2]: 以下的这种赋值方法正确吗?

A* const c = new A();   
A* b = c;

三 作为参数和返回值的const修饰符

其实,不论是参数还是返回值,道理都是一样的,参数传入时候和函数返回的时候,初始化const变量

  1. 修饰参数的const,如
void   fun0(const A*  a);
void   fun1(const A&  a);

调用函数的时候,用相应的变量初始化const常量,则在函数体中,按照const所修饰的部分进行常量化,如形参为const A* a,则不能对传递进来的指针的内容进行改变,保护了原指针所指向的内容;如形参为const A& a,则不能对传递进来的引用对象进行改变,保护了原对象的属性。
[注意]:参数const通常用于参数为指针或引用的情况;

  1. 修饰返回值的const,如
const A fun2();
const A* fun3();

这样声明了返回值后,const按照"修饰原则"进行修饰,起到相应的保护作用。

const rational operator*(const rational& lhs,const rational& rhs)   
{   
  return rational(lhs.numerator() * rhs.numerator(), lhs.denominator() * rhs.denominator());   
}   

返回值用const修饰可以防止这样的操作发生:

rational a,b;   
radional c;   
(a*b) = c;

一般用const修饰返回值为对象本身的情况多用于二目操作符重载函数并产生新对象的时候。

[总结] 一般情况下,函数的返回值为某个对象时,如果将其声明为const时,多用于操作符的重载。通常,不建议用const修饰函数的返回值类型为某个对象或对 某个对象引用的情况。

原因如下:

如果返回值为某个对象为const或某个对象的引用为const ,则返回值具有const属性,则返回实例只能访问类a中的公有数据成员和const成员函数,并且不允许对其进行赋值操作,这在一般情况下很少用 到。

[思考3]: 这样定义赋值操作符重载函数可以吗?

const A& operator=(const A& a);   

四 类成员函数中const的使用

一般放在函数体后,形如:

void fun() const;

如果一个成员函数的不会修改数据成员,那么最好将其声明为const,因为const成员函数中不允许对数据成员进行修改,如果修改,编译器将报错,这大 大提高了程序的健壮性。

五 使用const的一些建议

  1. 要大胆的使用const,这将给你带来无尽的益处,但前提是你必须搞清楚原委;
  2. 要避免最一般的赋值操作错误,如将const变量赋值,具体可见思考题;
  3. 在参数中使用const应该使用引用或指针,而不是一般的对象实例,原因同上;
  4. const在成员函数中的三种用法要很好的使用;
  5. 不要轻易的将函数的返回值类型定为const;
  6. 除了重载操作符外一般不要将返回值类型定为对某个对象的const引用;

六 一个简单的例子

#include <iostream>
using namespace std;

class A {
public:
    int a;
    A() : a(0)
    {}

    //
    // add 对成员变量进行了修改,所以该函数不能够被const修饰.
    //
    void add() {
        a++;
    }

    //
    // doNoting 不被const修饰的函数,在这个函数里,可以对成员变量进行修改,因此const A的实体不能调用这个函数.
    //
    void doNothing() {

    }

    //
    // doNotingConst 被const修饰的函数,也就是说在这个函数里面不能对成员变量有任何修改.
    // 否则的话,编译是通不过的,这个函数可以别一般的A实体,或者const A的实体调用.
    //
    void doNothingConst() const {

    }
};

int main() {
    //
    // 第一组,a是一个const A的实体,b是一个A的实体.
    //
    A b;
    const A a = b;
    //a.doNothing(); // error.
    
    //
    // 第二组,d是一个A的指针, c是一个const A的指针.
    //
    A* d = new A();
    const A* c = d; // c指向一个常量,具体而言,就是说通过c,不能够对其指向的对象做任何的修改.
    b.doNothing(); // ok.
    b.doNothingConst(); // ok.
    cout << c->a << endl; // ok.
    //c->add(); // error
    //c->doNothing(); // error
    c->doNothingConst(); // right,也就是说,一个const对象仅能够调用那些不改变A对象实体的函数,也就是带const版本的函数

    //
    // 第三组,f是一个A对象的实体,e是const A的引用.
    //
    A f;
    const A& e = f;
    //e.doNothing(); // error

    //
    // 第四组,g是一个const A对象的指针,h是一个A对象的指针.
    //
    const A* g = new A();
    //A* h = g; // error,不要妄想通过h对g所指向的const A实体进行修改.

    //
    // 第五组,i是一个A对象的指针常量, j是一个A对象的指针.
    //
    A* const i = new A();
    A* j = i; // ok

    getchar();
    return 0;
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,163评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,301评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,089评论 0 352
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,093评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,110评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,079评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,005评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,840评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,278评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,497评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,667评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,394评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,980评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,628评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,649评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,548评论 2 352

推荐阅读更多精彩内容

  • 前言 把《C++ Primer》[https://book.douban.com/subject/25708312...
    尤汐Yogy阅读 9,516评论 1 51
  • (1)可以定义 const 常量 (2)const 可以修饰函数的参数、返回值. 详细内容: 1、什么是const...
    幽鬼09阅读 707评论 0 4
  • const的用法,特别是用在函数后面 在普通的非 const成员函数中,this的类型是一个指向类类型的 cons...
    IvanRunning阅读 687评论 0 1
  • 最近,我妈刚刚让我们帮她注册了面子书账户。 然后,我们每个周末回家的言谈就是如下: "最近有某某新闻,听说某某家的...
    V之日志阅读 1,059评论 0 5
  • 精彩文章: 15张照片告诉你:坐飞机应该靠窗 下雨?我们司空见惯的;坐飞机,你也许是家常便饭; 但在飞机上看狂风暴...
    无邪书生阅读 693评论 0 3