《Effective C++ 》学习笔记——条款03
***************************************转载请注明出处:http://blog.csdn.net/lttree********************************************
一、 Accustoming Yourself to C++
Rule 03: Use const whenever possible.
条款03:尽可能的使用const
const就是常量,它同意你指定一个语义约束,编译器会强制实施这项约束。
多才多艺的keywordconst 有多种用途:
① 在classes 外部修饰global 或 namespace 作用域中的常量
② 修饰文件、函数 或 区块作用域(block scope )中被声明为static的对象
③ 用它修饰classes内部的 static 和 non-static 成员变量
几个关键的数据
1.对于指针
对于指针,能够设定 指针自身 、 指针所指物 或 两者都是(或都不是) const
分辨方法也非常easy:假设const出如今星号(*)左面,表示被指针所指的物是常量。假设出如今右面,表示指针自身是常量。
自此。也能够扩展到 迭代器 ( iterator )
假设你想表示这个迭代器不得指向不同的东西,但它所指的东西的内容是能够修改的:
const std::vector<int>::iterator iter
反之,迭代器能够指向不同的东西。但所指的内容不可修改:
std::vector<int>::const_iterator iter
2.最具威力的使用方法——对于函数声明时的应用
在一个函数声明中,const 能够和 函数返回值、各參数、函数自身 产生关联。
①函数的返回值
令函数返回一个常量值,往往能够减少因客户错误而造成的意外。而又不至于放弃安全性和高效性。
②函数的參数
除非你有须要修改參数或local const对象,否则请将它们声明为const。只几个字符,能够帮你省下非常多的烦恼。
③函数自身(const成员函数)
const实施于成员函数的理由有两个:
<1> 它们使classes接口比較easy理解(能够明白知道哪个函数能够修改对象内容。而哪个不能)
<2> 它们使“操作const对象”成为可能。 这个是 "pass by reference to const 方式传递对象"技术的基础。
并且,在重载方面,另一个非常重要的特性。
对于一个类,两个函数:
class TextBlock
{
public:
....
const char& operator[](std::size_t position) const
{ return text[position]; }
char& operator[] ( std::size_t position )
{ return text[position]; }
private:
std::string text;
};
仅仅要重载 operator[] 并对不同的版本号给予不同的返回类型,就能够令 const 和 non-const TextBlocks 获得不同的处理:
std::cout<<tb[0]; // 没问题,读一个non-const
tb[0] = ' x '; // 没问题。写一个non-const
std::cout<<ctb[0]; // 没问题,读一个const
ctb[0] = ' x '; // 错误,写一个const
错误的原因是 企图对一个“由const版之operator[]返回”的const char& 施行赋值动作。
此处要注意一下 non-const operator[] 的返回类型是个 reference to char (引用char型),不是char,
否则 tb[0] = ' x '将无法通过编译。
对于 成员函数假设是const意味性 有两个流派:bitwise 和 logical
——bitwise const 阵营
奉行: 成员函数仅仅有在不更改对象之不论什么成员变量(static除外)时才干够说是const。也就是说它不更改对象内的不论什么一个bit(位)。
长处:非常easy侦測违反点,编译器仅仅须要寻找成员变量的赋值动作就可以。
不足:不幸的是很多成员函数尽管不十足具备const性质却能通过bitwise測试。
能够看以下这个样例:
const CTextBlock cctb("Hello");
char* pc = &cctb[0];
*pc = 'J';
这样cctb如今存储的是 "Jello"这个内容,显然不符合const的定义
——logical const 阵营
奉行:一个const成员函数能够改动它所处理的对象内的某些bits,但仅仅有在client监測不出的情况下才如此。
比如 你的CTextBlock class 有可能快速缓存(cache)文本区块的长度以便询问:
class CTextBlock
{
public:
...
std::size_t length() const;
private:
char* pText;
std::size_t textLength;<span style="white-space:pre"> </span>// 近期一次计算的文本区块长度。
bool lengthIsValid;<span style="white-space:pre"> </span>// 眼下长度是否有效
};
std::size_t CTextBlock::length() const
{
if( !lengthIsValid)
{
textLength = std::strlen( pText );<span style="white-space:pre"> </span>// 错误! 在const 成员函数内不能赋值给这俩者
lengthIsValid = true;
}
return textLength;
}
length实现肯定不是 bitwise const 。由于textLength 和 lengthIsValid都可能被改动。它们两者被改动对const CTextBlock 对象而言尽管能够接受。但编译器不允许。
所以就用一个mutable(可变的)来解决,用mutable来释放掉non-static 成员变量的bitwise constness约束
class CTextBlock
{
public:
...
std::size_t length() const;
private:
char* pText;
mutable std::size_t textLength; // 这些成员变量即使在
mutable bool lengthIsValid; // const成员函数内也可更改
}; std::size_t CTextBlock::length() const
{
if( !lengthIsValid)
{
textLength = std::strlen( pText ); // 如今能够这样
lengthIsValid = true;
}
return textLength;
}
So 在const 和 non-const成员函数中避免反复
这个问题是由上面问题产生的,mutable固然能够解决变量的问题,可是假设这个东西非常长,
我们则须要些两份非常长非常长的怪物,so scary!
这就须要non-const来调用const的东西来避免反复,这样修改也方便一些,
{
为什么不让const调用non-const的?
拜托,const成员函数承诺绝不改变其对象的逻辑状态,non-const没有
}
这样就须要 调用 转型 这个概念,看以下的样例:
class TextBlock
{
public:
...
const char& operator[](std::size_t position) const
{
...
...
...
return text[position];
}
char& operator[](std::size_t position)
{
return
const_cast<char&>(
static_cast<const TextBlock&>(*this) [position]
);
}
...
};
这里用到了两个转型:
static_cast 将non-const对象转换成const对象
const_cast 移除const
结束语:
const是一个很奇异的且非比平常的东西,
它能够用在 指针和迭代器上;
在指针、迭代器及reference指涉的对象身上;
在函数參数和返回类型上;
在local变量身上;
在成员函数身上,
等等。。
。。
Please remember:
<1> 将某些东西声明为 const 可帮助编译器侦測出错误使用方法。const可被施加于不论什么作用域内的对象、函数參数、函数返回类型、成员函数本体。
<2> 编译器强制实施bitwise constness。但你编敲代码时应该使用 conceptual constness(概念上的常量性)。
<3> 当const和non-const 成员函数有着实质等价的实现时,令non-const版本号用const版本号可避免代码反复。
***************************************转载请注明出处:http://blog.csdn.net/lttree********************************************
《Effective C++ 》学习笔记——条款03的更多相关文章
- Effective C++学习笔记 条款07:为多态基类声明virtual析构函数
一.C++明确指出:当derived class对象经由一个base class指针被删除,而该base class带着一个non-virtual析构函数,其结果未定义——实际执行时通常发生的是对象的 ...
- Effective C++学习笔记 条款06:如不想使用编译器自动生成的函数,就该明确拒绝
一.为驳回编译器自动提供的机能,可将相应成员函数声明为private并且不予实现.(如果你仅仅是自己不实现的话,编译器会帮你实现) 如: class A { public: A(const strin ...
- Effective C++学习笔记 条款05:了解C++默默编写并调用的哪些函数
一.如果用户没有提供构造函数.copy构造函数.copy assignment操作符和析构函数,当且仅当这些函数被需要的时候,编译器才会帮你创建出来.编译器生成的这些函数都是public且inline ...
- Effective C++学习笔记 条款04:确定对象被使用前已先被初始化
一.为内置类型对象进行手工初始化,因为C++不保证初始化它们. 二.对象初始化数据成员是在进入构造函数用户编写代码前完成,要想对数据成员指定初始化值,那就必须使用初始化列表. class A { pu ...
- Effective C++学习笔记 条款02:尽量以const,enum,inline替换 #define
尽量使用const替换 #define定义常量的原因: #define 不被视为语言的一部分 宏定义的常量,预处理器只是盲目的将宏名称替换为其的常量值,导致目标码中出现多分对应的常量,而const定义 ...
- Effective STL 学习笔记 39 ~ 41
Effective STL 学习笔记 39 ~ 41 */--> div.org-src-container { font-size: 85%; font-family: monospace; ...
- Effective STL 学习笔记 Item 38 : Design functor classes for pass-by-value
Effective STL 学习笔记 Item 38 : Design functor classes for pass-by-value */--> div.org-src-container ...
- Effective STL 学习笔记 Item 34: 了解哪些算法希望输入有序数据
Effective STL 学习笔记 Item 34: 了解哪些算法希望输入有序数据 */--> div.org-src-container { font-size: 85%; font-fam ...
- Effective STL 学习笔记 32 ~ 33
Effective STL 学习笔记 32 ~ 33 */--> div.org-src-container { font-size: 85%; font-family: monospace; ...
随机推荐
- 网站服务器、VPS和虚拟主机的联系与区别
网站服务器是指在互联网数据中心中存放网站的服务器.主要用于网站的互联网中的发布.应用,是网络应用的基础硬件设施.简单的说服务器就是一台电脑,只是这台电脑因为要24 小时高速运行,所以配置要比一般的家 ...
- HMM的学习笔记1:前向算法
HMM的学习笔记 HMM是关于时序的概率模型.描写叙述由一个隐藏的马尔科夫链随机生成不可观測的状态随机序列,再由各个状态生成不可观測的状态随机序列,再由各个状态生成一个观測而产生观測的随机过程. HM ...
- Sybase配置中文语言支持及字符集
在windows平台上,Sybase ASE15.0.7安装完成后默认语言是:英语(us_english),默认字符集为:cp850.cp850是一个西欧字符集,虽然也能使用cp850字符集保存汉字, ...
- python 笔记4-- 函数式编程
高阶函数 把函数作为参数传入,这样的函数称为高阶函数,函数式编程就是指这种高度抽象的编程范式. 在python中 函数也是一种变量 def add(x, y, f): return f(x) + f( ...
- jar包和war包的区别:
jar包就是别人已经写好的一些类,然后将这些类进行打包,你可以将这些jar包引入你的项目中,然后就可以直接使用这些jar包中的类和属性了,这些jar包一般都会放在lib目录下. war是一个web模 ...
- W3C-XML
XML XML Extensible Markup Language,可扩展标记语言 1 XML和HTML的区别 XML主要用来传输数据 HTML主要用来呈现数据内容 2 XML的主要用途 传输数据 ...
- Git使用过程
Git-------目前世界上最先进的分布式版本控制系统(没有之一) 什么是版本控制系统? 说简单点,就是一个文件,对其增加.删除.修改都可以被记录下来,不仅自己可以修改,其他人也可以进行修改 每次对 ...
- SQL Performance Improvement Techniques(转)
原文地址:http://www.codeproject.com/Tips/1023621/SQL-Performance-Improvement-Techniques This article pro ...
- JAVA多线程下,获取递增的序列号
场景描述: 1,目前我们的系统可以简单归纳成MVC的架构模式 2,每个前端的请求过来,都会在C层开启事务,最后处理结束后,也在在C层关闭事务(实际是在C层的底层统一做了事务的开启和提交): ...
- Apache+tomcat的整合
http://blog.csdn.net/stefyue/article/details/6918542 为什么要做这个整合呢?当然,首先想到是就是Apache和Tomcat的区别.正因为有区别,有各 ...