《Effective C++》重点摘要(四)

《Effective C++》第四章:设计与声明

  1. 让接口容易被正确使用,不易被误用。一个接口由返回类型、接口名称、和参数列表组成,为了让接口容易被正确的使用,需要小心设计返回类型,最好是简单、直接、自然。接口名称选择很重要,做到简单、达意、无歧义。参数列表形参类型需要身份小心,如果能防范非法输入,尽力为之,形参名也尽可能做到同接口名称一样的标准。另外请保持命名习惯的一致性也能收到奇效。
  2. 设计class犹如设计type。用户对语言内置的type支持的行为一般十分熟悉,所以最好自行设计的class能够像内置type一样,让用户使用起来得心应手。为了做到这个,可以问自己下面这些问题:

    1) 新type对象如何被创建和销毁?

    2) 对象的初始化和对象的赋值该有什么差别?

    3) 新type的对象如果被pass-by-value,意味着什么?

    4) 什么是新type的“合法值”?

    5) 新type需要配合某个继承图系吗?

    6) 新type需要什么样的转换?

    7) 什么样的操作符和函数对此新type而言是合理的?

    8) 什么样的标准函数应该驳回?需要拒绝的或是该声明为private的?

    9) 新type的成员访问权限是否合理?

    10) 什么是新type的为声明接口?它对效率、异常安全、资源利用提供何种保证?

    11) 新type有多么一般化?需要template吗?

    12) 这个新type真的是需要的吗?

  3. 宁以pass-by-reference-to-const替换pass-by-value。以const引用作为引用的第一理由是安全,不会发生对象切割,第二理由是高效。但是语言内置的简单类型还是以值传递比较高效。
  4. 必须返回对象时,别妄想返回其reference。为了效率返回引用……额,别想了,函数内部的局部变量在函数结束时会被析构的,引用怎么可以指向一个返回指针吗?可以是可以,但是谁保证释放这个指针指向的内存,并且这种方法也不一定总是有效。也许你可以利用RAII,返回一个智能指针。但是,最好还是返回对象吧。
  5. 将成员变量声明为private。封装得越多,实现者可以在幕后做更多手脚而不影响客户端代码;为封装的成员变量提供访问的函数接口,给用户更一致的使用风格;protected并不比public更具有封装性,因为它在子类中就像public一样。C++中只有两种访问权限:封装的(private)和不封装的。
  6. 宁以non-member且non-friend替换member函数。这一个条款真的不适合几句话说清楚,这里我想说的是如果发现某个成员函数只是调用一序列public类型的函数来完成功能,那么就把它编写为一个non-member且non-friend函数。这样可以带来更大的封装性、包裹弹性和机能扩充性,降低编译依赖性。具体原因非常精彩,将以另一篇博文呈现。
  7. 若所有参数皆需要类型转换,请为此采用non-member函数。比如为复数类重载一个乘法运算符(*),如果作为一个member函数,那么第一参数this将不能轻松实现转型,所以为了便利,将其实现为一个non-member函数会让日子过得更轻松一点。
  8. 考虑写出一个不抛异常的swap函数。应该实现一个member版的swap函数,然后再实现一个non-member版的swap函数,让non-member版调用member版swap函数,一般情况下,这样做是为了copy-and-swap技术,从而做到异常安全,于是,不应该让swap函数抛出异常。
时间: 2024-08-01 05:05:08

《Effective C++》重点摘要(四)的相关文章

BDD in Action 重点摘要

BDD in Action 重点摘要: BDD in Action ?对于需灰度发布的产品, 开发人员可经由BDD, 便可自保证需灰度发布产品的质量? ?所以, BDD 可使产品更快速的达到灰度发布的要求, 而可更快速的获得使用者对产品的反馈? ?在这样的开发模式下, 测试人员可更专注于探索性测试, 挖掘出更深层的产品缺陷, 回归到真正测试人员的专业?

《Effective C++》重点摘要(五)

<Effective C++>第五章:实现 尽可能延后变量定义式的出现时间.只有变量在恰好要使用之前定义,程序的可读性往往会得到提高,因为这样不容易忘记变量说代表的意思.另一方面,这样做可以提高程序性能,如果不需要一个变量时却要为它分配.释放空间,调用构造.析构函数,获取.释放资源--这,真是太浪费了.补充一点,声明式并不会做这些事情,所以可以考虑用声明式替换定义式以尽量延后变量的定义. 尽量少做转型动作.转型意味着出错的可能性大大提升,转型意味着更多的操作.如果不得不转型,考虑使用C++ s

《Effective C++》重点摘要(二)

<Effective C++>第二章:构造/析构/赋值运算 C++默认编写的函数.C++编译器如果没有发现以下函数,就会为类生成一份默认版本的: 1) default构造函数 2) default析构函数 3) copy构造函数 4) copy assignment操作符(=运算符) 前两个函数并不总是产生,它只在编译器需要的时候才产生出来.后两个函数只保证以bitwise语义拷贝non-static成员,所以如果有指针.引用等non-static成员的类要么拒绝编译器产生的版本,要么自行小心

《Effective C++》重点摘要(七)

<Effective C++>第七章:模板与泛型编程 了解隐式接口和编译期多态.面向对象编程总是采用显式地声明一个接口,并在子类中提供特殊的实现,进而实现运行期多态.模板类中的接口往往是隐式的,隐式的意思是,编写模板时,假设需要的接口类T中是存在的,至于实例化模板的类是否真的具有该接口,编译时才知道(如果实例化类型没有实现这些接口,则编译失败).这正是模板多态的展现方式,由实例化模板的类决定具体的行为是什么.这种多态编译期与运行期多态有很大的区别,运行期的多态是一种is-a类型的多态(我自造的

《More Effective C++》重点摘要一:基础议题

仔细区别pointers和references.指针和引用有些相似,他们本身都是对存在于某个地方的对象(不是指class)的指示,但是他们有着本质的区别.指针变量存储所指对象的地址,所指的对象可以是null,只要可以寻址就行.而引用是某个已经存在对象的别名,所以不可以先声明一个引用,经过一段时间(代码)后让它指向某个对象. 最好使用C++转型操作符.C++提供了自己的四种转型操作符: 1) static_cast.拥有与C旧式转型相同的意义与限制,但不能移除表达式的常量性,因为有const_ca

《Effective C++》重点摘要(六)

<Effective C++>第六章:继承与面向对象设计 确定你的public继承塑模出is-a关系.student is a person,所以student可以public继承自person.概念上,正方形是长方形,但是如果让square继承自rectangle呢,square需要像rectangle那样具有长和宽两个成员变量吗?如果是,每次改变正方形的宽的时候,也要改变正方形的长,这很奇怪不是吗?所以is-a关系不仅仅是现实概念上的,也是实现上的,确保对于base class对象身上发生

《Effective C++》重点摘要(八)

<Effective C++>第八章:定制new和delete 了解new-handler的行为.new和delete不是函数,是申请和释放内存的操作符.当new提出获得内存申请失败时会发生什么?老旧的编译器是返回null指针.现在呢,如果申请失败,会先调用一个错误处理函数,那就是new-handler.这就像一个回调函数,系统有一个默认的,用户也可以自行编写一个错误处理函数并使用set_new_handler函数设置之.通常自行编写的错误处理函数可以使用这些策略: 1) 多次尝试申请内存.

《Effective C++》重点摘要(九)

<Effective C++>第九章:杂项讨论 不要轻忽编译器的警告.严肃对待编译器发出的警告信息.努力在最高警告级别下争取无警告.也不要过度依赖编译器给出的警告信息,不同的编译器给出的警告信息是不一样的,甚至给出警告的点也不一样. 让自己熟悉包括TR1在内的标准程序库. 让自己熟悉Boost. C++不想Java和C#那样,语言本身就带有很强大的库,所以外部的程序库对于编写C++程序至关重要.俗话说不要重复发明轮子嘛,所以能用别人的就用吧.当然这是在实际工程中,学习阶段嘛,我还是比较推崇自己

《Effective C++》重点摘要(一)

开篇的话 这个星期不再发布关于数据结构的博客,想把半个月来看的书做一些总结,整理整理,第一本就是<Effective C++>.第一次看这本书是一年多前,准备考研复试的时候,随后陆陆续续,这个月再来看算是第三遍了吧,之前没有看过<深度探索C++对象模型>,所以有的地方看得不是很透彻(现在有的地方也看得不透,但是比以前好点了).还有另一本书就是<深度探索C++对象模型>,看第二遍,吸收得更容易了,看书总是这样,第一遍也许什么也不知道,但是只要坚持看下去,回过头来再读的时候