深入浅出const

  §通常,如果一个对象通过引用方式传到函数f中,而函数f又不会通过修改对象的数据成员的值改变该对象的状态,那么,我们最好将f的参数标记为const,这样可以预防对参数的误写,同时有些编译器还可对这种情况进行一些优化。

  如下:将函数setName的string类型参数n标记为const,表明setName不会改变n,只是将n赋值给数据成员name。

 1 class  C
 2 {
 3     public:
 4         void setName(const string &n)
 5             {
 6                 name = n;
 7             }
 8         //...other public members
 9     private:
10         string name;
11 };

  §如果一个成员函数不需要直接或间接(通过调用其他的成员函数考改变其队形状态)地改变该函数所属对象的任何数据成员,那么最好将这个成员函数标记为const。

  如下,由于get成员函数不需要改变类C 的任何数据成员,因此将get成员函数标记为const。

1 class C  {
2     public:
3         void set(int n) { num = n;}
4         int get() const  {return num;}
5     private:
6         int num;
7 };

  说明:

    <1>由于get成员函数不更改任何数据成员,因此这种类型的函数被称为只读函数。

    <2>将成员函数标记为const可以预防对该函数所属对象的数据成员的误写,同时有些编译器还可以对这种情况进行一些优化。

    <3>一个const成员函数仅能调用其他const成员函数;因为const成员函数不允许直接或间接地改变对象的状态,而调用非const成员函数可能会间接地改变对象的状态。

    例如:

 1 class  C {
 2     public:
 3         void m1(int x) const {
 4             m2(x);     //***error
 5         }
 6
 7         void m2(int x){dm = x;}
 8     private:
 9         int dm;
10 };

  §const关键字三种不同的用法实例

  ——在成员函数set中,因为set不改变string类型参数n,n被标记为const。

  ——成员函数get返回数据成员name的一个const型引用,此处的const表明谁也不能通过这个引用来修改数据成员name的值。

  ——成员函数get本身被标记为const,因为get不会改变类C唯一的数据成员name的值。

  例如:

1 class  C{
2     public:
3         void set(const string &n)  {name = n;}
4         const string &get() const  {return  name;}
5     private:
6         string name;
7 };

  因此有:

    (1)某函数如果采用const返回,则其返回值只能付给一个const类型的局部变量。

    (2)如果该const返回值是一个类的指针或者引用的话,则不能用该指针或引用调用该类的non-const成员函数,因为这些函数可能会改变该类的数据成员的值。

综合以上我们来看这样一个例子:

 1 #include<iostream>
 2 using namespace std;
 3 class  sz
 4 {
 5     public:
 6         int &f1()      //可修改mazT,可修改返回值
 7         {mszT = 5; return mszT;}
 8
 9         int f2() cosst        //不能修改mszT,可修改返回值
10         {return mszT;}
11
12         const int &f3()     //可修改mszT, 不可修改返回值
13         {mszT = 6; return  mszT;}
14
15         const int& f4 const     //不可修改mszT, 不可修改返回值
16         {return mszT;}
17
18         void print()
19         {cout<<mszT<<endl;}
20
21     private:
22         int mszT;
23 };
24
25 int main()
26 {
27     sz f;
28     int &s1 = f.f1();            //OK
29     cout<<s1<<endl;         //输出:5
30     s1 = 20;                       //可修改返回值
31     f.print();                       //输出:20
32
33     int &s3 = f.f3();           //Error, 返回const值,但s3不是const
34     const int & s3 = f.f3()    //OK
35     cout<<s3<<endl;          //输出:5
36     s3 = 20;                        //Error,修改返回值
37
38     return 0;
39 }
时间: 2024-10-21 01:21:29

深入浅出const的相关文章

ES6深入浅出-1 新版变量声明:let 和 const-2.视频 let和const

以前的var方式声明不好用 a=1回声明一个全局变量, 输出了1 说明a=1确实声明了一个全局变量. 但是你把放在其他的地方,就不是声明全局变量了. 如果外面有个全局变量a 那么函数里面就是给a赋值 有两层函数的时候,直接使用的是局部变量a,也就是fn函数里面的a 如果当前没有a就声明一个全局变量a.如果有a就利用这个a.下面当前就是有a的 这样用的就是最外面的var a 如果外面也没有,这时候才会去声明一个全局变量 所以a=1会声明一个全局变量是错的,只有在没有a这个变量的情况下才会隐式的去声

【C++深入浅出】智能指针之auto_ptr学习

起:  C++98标准加入auto_ptr,即智能指针,C++11加入shared_ptr和weak_ptr两种智能指针,先从auto_ptr的定义学习一下auto_ptr的用法. template<class _Ty> class auto_ptr { // wrap an object pointer to ensure destruction public: //定义_Myt类型,作用域局限于类中,便于书写和理解 typedef auto_ptr<_Ty> _Myt; typ

深入浅出实例解析linux内核container_of宏

做一件事情首先应该知道它的目的是什么. container_of的目的:如何通过结构中的某个变量获取结构本身的指针. 总体思路:假想一下,你的结构体中有好几个成员,你如何通过里面的"任一成员"获取整个结构体的首地址呢.container_of的做法就是通过typeof定义一个与"任一成员"同类型的指针变量pvar_a(假设变量名就是pvar_a),并让指针变量pvar_a指向这个"任一成员",然后用 "pvar_a的地址" 减

深入浅出Cocoa之消息

[Cocoa]深入浅出Cocoa之消息 深入浅出Cocoa之消息    罗朝辉 (http://www.cnblogs.com/kesalin/) 本文遵循“署名-非商业用途-保持一致”创作公用协议 在入门级别的ObjC 教程中,我们常对从C++或Java 或其他面向对象语言转过来的程序员说,ObjC 中的方法调用(ObjC中的术语为消息)跟其他语言中的方法调用差不多,只是形式有些不同而已. 譬如C++ 中的: Bird * aBird = new Bird(); aBird->fly(); 在

[Cocoa]深入浅出 Cocoa 之消息

深入浅出 Cocoa 之消息    罗朝辉(http://blog.csdn.net/kesalin) 转载请注明出处 在入门级别的ObjC 教程中,我们常对从C++或Java 或其它面向对象语言转过来的程序猿说,ObjC 中的方法调用(ObjC中的术语为消息)跟其它语言中的方法调用差点儿相同,仅仅是形式有些不同而已. 譬如C++ 中的: Bird * aBird = new Bird(); aBird->fly(); 在ObjC 中则例如以下: Bird * aBird = [[Bird al

Android深入浅出之Binder机制(转)

Android深入浅出之Binder机制 一 说明 Android系统最常见也是初学者最难搞明白的就是Binder了,很多很多的Service就是通过Binder机制来和客户端通讯交互的.所以搞明白Binder的话,在很大程度上就能理解程序运行的流程. 我们这里将以MediaService的例子来分析Binder的使用: l         ServiceManager,这是Android OS的整个服务的管理程序 l         MediaService,这个程序里边注册了提供媒体播放的服

[深入浅出Windows 10]模拟实现微信的彩蛋动画

9.7 模拟实现微信的彩蛋动画 大家在玩微信的时候有没有发现节日的时候发一些节日问候语句如“情人节快乐”,这时候会出现很多爱心形状从屏幕上面飘落下来,我们这小节就是要模拟实现这样的一种动画效果.可能微信里面实现的动画效果都是采用固定的小图片来最为动画的对象,但是我们这小节要对该动画效果增加一些改进,也就是要实现的彩蛋动画的针对的图形形状是动态随机生成的,所以看到从屏幕上面飘落的图形都是形状不太一样的.下面来看一下如何实现星星飘落动画. 9.7.1 实现的思路 首先,我们来分析一下星星飘落动画的实

深入浅出Cocoa之消息(二)-详解动态方法决议(Dynamic Method Resolution) 【转】

序言 如果我们在 Objective C 中向一个对象发送它无法处理的消息,会出现什么情况呢?根据前文<深入浅出Cocoa之消息>的介绍,我们知道发送消息是通过 objc_send(id, SEL, ...) 来实现的,它会首先在对象的类对象的 cache,method list 以及父类对象的 cache, method list 中依次查找 SEL 对应的 IMP:如果没有找到且实现了动态方法决议机制就会进行决议,如果没有实现动态方法决议机制或决议失败且实现了消息转发机制就会进入消息转发流

Android深入浅出之 AudioTrack分析

Android深入浅出之Audio 第一部分 AudioTrack分析 一 目的 本文的目的是通过从Audio系统来分析Android的代码,包括Android自定义的那套机制和一些常见类的使用,比如Thread,MemoryBase等. 分析的流程是: l         先从API层对应的某个类开始,用户层先要有一个简单的使用流程. l         根据这个流程,一步步进入到JNI,服务层.在此过程中,碰到不熟悉或者第一次见到的类或者方法,都会解释.也就是深度优先的方法. 1.1 分析工