条款1:视C++为一个语言联邦

记住:

  ★C++高效编程守则视状况而变化,这取决于你使用C++的哪一部分

    C;

    Object-oriented c++;

    Template c++;

    STL

条款2:尽量以constenuminline替换#define

记住:

  ★对于单纯常量,最好以const对象或enums替换#define

  ★对于形似函数的宏,最好改用inline函数替换#define

------------------------------------------------------------------------------------------------------

延伸:面试时经常会被问到的一个典型问题是:参数宏与函数的区别,我想可以从如下几个角度来回答:

  a 函数调用要先求出实参表达式的值,然后代入形参;宏只是字符替换;

  b 宏替换只占编译时间;函数调用占运行时间;

  c 函数运行时可调试,宏不可;

  d 函数会做参数类型检查,宏不会;

  e 使用宏次数多时宏展开后源程序变长;函数调用不会;

条款3:尽可能使用const

记住:

  ★将某些东西声明为const可帮助编译器侦测出错误用法。const可被施加于任何作用域内的对象、函数参数、函数返回类型、成员函数本体

  ★编译器强制实施bitwise constness,但你写程序时应使用“概念上的常量性”(conceptual constness)

  ★当const和non-const成员函数有着实质等价的实现时,令non-const版本调用const版本可避免代码重复

----------------------------------------------------------------------------------------------------------------------

1 一条规则:

  const出现在*左,表示被指物是常量: const int *p = NULL;

    *右,  指针自身是常量:int *const p = NULL;

    *两边,  两者都是常量:const int *const p = NULL;

  对于被指物是常量的情形,const 写在类型前后无关:const int *p 等价于 int const *p

2 STL迭代器中const的用法:

  迭代器自身是const的用法:  const std::vector<int>::iterator iter = vec.begin();

                *iter = 10; //可以,改变iter所指之物

                ++iter; //错误,iter自身是const

  迭代器所指之物是const的用法:std::vector<int>::const_iterator cIter = vec.begin();

                *cIter = 10; //错,*cIter是const

                ++cIter; //没问题,cIter本身可以变化

3 const成员函数

将const实施于成员函数的目的是为了确认该成员函数可作用于const对象身上。这类成员函数重要的两个原因:

  一、使class接口较容易被理解。因为可得知哪个函数可改动对象内容而哪个不行很是重要;

  二、使“操作const”对象成为可能。

4 bitwise constness(或physical constness)和logical constness

  利用mutable可释放掉non-static成员变量的bitwise constness约束

5 constnon-const成员函数中避免重复

  利用const成员函数实现出其non-const成员函数:举例:

    

  1. class TextBlock {
  2. ...
  3. const char& operator[]( std::size_t position ) const { //const版本
  4. ...
  5. ...
  6. ...
  7. return text[position];
  8. }
  9.  
  10. char& operator[]( std::size_t position ) { //用const实现的non-const版本
  11.  
  12. return
  13. const_cast<char&>(
  14.  
  15. static_cast<const TextBlock&>(*this)[position] //转换后调用const版[]
  16. );
  17. }
  18.  
  19. ...
  20. }

理解:这份代码有两个转型动作

  一、将*this从其原始类型TextBlock&转型为const TextBlock&(这使得接下来调用operator[]时得以调用const版本成员函数)

  二、从const operator[]的返回值中移除const

最后需要注意:non-const成员调用const成员可以,但反之不行!!!

EC读书笔记系列之1:条款1、条款2、条款3的更多相关文章

  1. EC读书笔记系列之12:条款22、23、24

    条款22 将成员变量声明为private 记住: ★切记将成员变量声明为private.这可赋予客户访问数据的一致性.可细微划分访问控制.允诺约束条件获得保证,并提供class作者以充分的实现弹性. ...

  2. EC读书笔记系列之16:条款35、36、37、38、39、40

    条款35 考虑virtual函数以外的其他选择 记住: ★virtual函数的替代方案包括NVI手法及Strategy模式的多种形式.NVI手法自身是一个特殊形式的Template Method模式 ...

  3. EC读书笔记系列之20:条款53、54、55

    条款53 不要轻忽编译器的警告 记住: ★严肃对待编译器发出的警告信息.努力在你的编译器的最高(最严苛)警告级别下争取“无任何警告”的荣誉 ★不要过度依赖编译器的报警能力,∵不同的编译器对待事情的态度 ...

  4. EC读书笔记系列之19:条款49、50、51、52

    条款49 了解new-handler的行为 记住: ★set_new_handler允许客户指定一个函数,在内存分配无法获得满足时被调用 ★Nothrow new是一个颇为局限的工具,∵其只适用于内存 ...

  5. EC读书笔记系列之18:条款47、48

    条款47 请使用traits classes表现类型信息 记住: ★Traits classes使得“类型相关信息”在编译期可用.它们以templates和“templates特化”完成实现 ★整合重 ...

  6. EC读书笔记系列之17:条款41、42、43、44、45、46

    条款41 了解隐式接口与编译器多态 记住: ★classes和templates都支持接口和多态 ★对classes而言接口是显式的(explicit),以函数签名为中心.多态则是通过virtual函 ...

  7. EC读书笔记系列之15:条款32、33、34

    条款32 确保你的public继承塑模出is-a关系 记住: ★public继承意味着is-a.适用于base class身上的每一件事情一定也适用于derived class身上,∵每一个deriv ...

  8. EC读书笔记系列之14:条款26、27、28、29、30、31

    条款26 尽可能延后变量定义式的出现时间(Lazy evaluation) 记住: ★尽可能延后变量定义式的出现.这样做可增加程序的清晰度并改善程序效率 ----------------------- ...

  9. EC读书笔记系列之11:条款20、21

    条款20 宁以pass-by-reference-to-const替换pass-by-value 记住: ★尽量以pass-by-reference-to-const替换pass-by-value.前 ...

随机推荐

  1. C++智能指针初学小结

    本篇随笔仅作为个人学习<C++ Primer>智能指针一节后的部分小结,抄书严重,伴随个人理解.主要介绍shared_ptr.make_shared.weak_ptr的用法和联系. C++ ...

  2. map和reduce

    map()函数接收两个参数,一个是函数,一个是Iterable,map将传入的函数依次作用到序列的每个元素,并把结果作为新的Iterator返回. map()传入的第一个参数是f,即函数对象本身.由于 ...

  3. POJ2446 二分图最大匹配

    问题:POJ2446 分析: 采用黑白相间的方法把棋盘分成两个点集,是否可以用1*2的卡片实现全覆盖等价于二分图是否有完全匹配. AC代码 //Memory: 172K Time: 32MS #inc ...

  4. Service Lane

    Link https://www.hackerrank.com/challenges/service-lane def main(): n, t = map(int, raw_input().spli ...

  5. isinstance()和__name__

    # coding: utf-8 def displayNumType(num): print num, 'is', if isinstance(num, (int, long, float, comp ...

  6. 使用Japserreport填充报表数据(3)

    E中以PDF文件的格式显示静态的中文字符串,在大多数的情况下,打印的数据来自于一些变量,在JasperReports工具中传递数据并填充到 报表只有两种方式,即使用Parameters参数和JRDat ...

  7. Oracle字符函数(转换大小写,替换等)

    在oracle中,有一些字符函数: upper(字符串):转换为大写lower(字符串):转换为小写initcap(字符串):首字母大写replace(字符串1,字符串2,字符串3):将串1中所有的串 ...

  8. QProcess进程间双向通信

    记得以前写过Linux的C程序, 里面用popen打开一个子进程, 这样可以用read/write和子进程通讯, 而在子进程里则是通过从stdin读和向stdout写实现对父进程的通讯. QProce ...

  9. boost库使用:vs2013下boost::container::vector编译出错解决

    boost版本:boost_1_55_0 bug报告地址 https://svn.boost.org/trac/boost/ticket/9332 出错信息 has_member_function_c ...

  10. 深入解析spring中用到的九种设计模式

    转载请注明出处,文章首发于:http://itxxz.com/a/javashili/tuozhan/2014/0601/7.html 设计模式作为工作学习中的枕边书,却时常处于勤说不用的尴尬境地,也 ...