您的位置:首页 > 编程语言 > C语言/C++

C++笔记:Const

2014-03-14 08:23 183 查看


const 常量

定义义后就不能被修改,所以定义时必须初始化:const std::string hi = "hello!";


const 对象作用域

在全局作用域定义的非const对象在整个程序中都可以访问,默认为extern. [code1]
const对象默认为文件的局部变量,要使const变量能够在其他的文件中访问,必须地指定它为extern. [code2]

常量表达式

const expression是指值不会改变并且在编译过程就能得到计算结果的表达式
一个对象(或表达式)是不是常量表达式由它的数据类型和初始值共同决定 [code4]
字面值属于常量表达式
用常量表达式初始化的const对象也是常量表达式.

一些const在头文件中的定义

如果const变量是用常量表达式初始化的,那么它应该在头文件中定义(原因类似C的宏#define)
如果const变量不是用常量表达式初始化,那么它就不应该在头文件中定义该const变量应该在一个源文件中定义并初始化,应在头文件中为它添加extern声明,以使其能被多个文件共享

引用

不能定义引用类型的引用,但可以定义任何其他类型的引用.
引用必须用与该引用同类型的对象初始化.
必须在定义引用时进行初始化,初始化后可以赋值,但是不能修改引用的指向.

const 引用

指向const对象的引用
普通引用不能绑定const对象
const引用则可以绑定到不同但相关的类型的对象或绑定到右值. [code3]
非const引用只能绑定到与该引用同类型的对象. code[3]

const和指针

指向const对象的指针(const int *ptr)

允许给 ptr 重新赋值,使其指向另一个 const 对象,但不能通过 ptr 修改其所指对象的值.
不能使用 void* 指针保存 const 对象的地址,必须使用 const void*
允许把非 const 对象的地址赋给指向 const 对象的指针

const指针

int *const curErr.从右向左把上述定义语句读作“curErr 是指向 int 型对象的 const 指针”
与任何 const 量一样,const 指针也必须在定义时初始化. 且初始化过后不能指向另外一个对象

指向 const 对象的 const 指针

const double pi = 3.14159;const double *const pi_ptr = π
既不能修改 pi_ptr 所指向对象的值,也不允许修改该指针的指向.
从右向左阅读上述声明语句:“pi_ptr 首先是一个 const 指针,指向 double 类型的 const 对象”。


typedef和指针

typedef string *pstring;const pstring cstr;==string *const cstr;
声明 const pstring 时,const 修饰的是 pstring 的类型,这是一个指针。因此,该声明语句应该是把cstr 定义为指向 string 类型对象的 const 指针
const 限定符既可以放在类型前也可以放在类型后。string const s1; == const string s2; [Code5]


const和成员函数

将成员函数声明为常量double avg_price() const;
const 成员不能改变其所操作的对象的数据成员
const 必须同时出现在声明和定义中,若只出现在其中一处,就会出现一个编译时错误。

从 const 成员函数返回 *this

在普通的非 const 成员函数中,this 的类型是一个指向类类型的 const 指针,可以改变 this 所指向的值,但不能改变 this 所保存的地址。

在 const 成员函数中,this 的类型是一个指向 const 类类型对象的 const 指针。既不能改变 this 所指向的对象,也不能改变 this 所保存的地址。

不能从 const 成员函数返回指向类对象的普通引用。const 成员函数只能返回 *this 作为一个 const 引用。

const与成员函数重载

Screen& display(std::ostream &os){ do_display(os); return *this; }
const Screen& display(std::ostream &os) const{ do_display(os); return *this; }

const与迭代器

const_iterator:只能用于读取容器内的元素,但不能改变其值

=====
Code1.在全局作用域定义的非const对象

// file_1.cc
int counter; // definition
// file_2.cc
extern int counter; // uses counter from file_1
++counter;
// increments counter defined in file_1


Code2.在全局作用域定义的const对象

// file_1.cc
// defines and initializes a const that is accessible to other files
extern const int bufSize = fcn();
// file_2.cc
extern const int bufSize; // uses bufSize from file_1,extern 标志着bufSize 是一个声明,所以没有初始化式
// uses bufSize defined in file_1
for (int index = 0; index != bufSize; ++index)
// ...


Code3.const引用则可以绑定到不同但相关的类型的对象或绑定到右值

/*const 引用可以初始化为不同类型的对象或者初始化为右值,如字面值常量:*/
int i = 42;
// legal for const references only
const int &r = 42;
const int &r2 = r + i;

/*同样的初始化对于非 const 引用却是不合法的,而且会导致编译时错误。其原因非常微妙,值得解释一下。
观察将引用绑定到不同的类型时所发生的事情,最容易理解上述行为。假如我们编写*/
double dval = 3.14;
const int &ri = dval;

/*编译器会把这些代码转换成如以下形式的编码:*/
int temp = dval;
const int &ri = temp;
// create temporary int from the double
// bind ri to that temporary

/*如果 ri 不是 const,那么可以给 ri 赋一新值。这样做不会修改 dval,而是修改了 temp。期望对 ri 的赋值会修改 dval 的程序员会发现 dval 并没
89有被修改。仅允许 const 引用绑定到需要临时使用的值完全避免了这个问题,因为 const 引用是只读的。*/


Code4. 一个对象(或表达式)是不是常量表达式由它的数据类型和初始值共同决定

const int max_files = 20; // max_files是常量表达式
const int limit = max_files + 1; // limit是常量表达式
int staff_size = 27; // staff_size不是常量表达式
const int sz = get_size(); // sz不是常量表达式
/*尽管staff_size的初始值是个字面值常量,但由于它的数据类型只是一个普通int而非const int,所以它不属于常量表达式。另一方面,尽管sz本身是一个常量,但它的具体值直到运行时才能获取到,所以也不是常量表达式。*/


Code5.const 限定符既可以放在类型前也可以放在类型后

string s;
typedef string *pstring;

//下面三种声明是等价的
const pstring cstr1 = &s;
pstring const cstr2 = &s;
string *const cstr3 = &s;


From:http://blog.csdn.net/liufei_learning/article/details/21219963
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: