effictive c++
c++条款
num 1:尽量以const enum inline替换#define
1)对于单纯常量,最好以const对象或enums替换#defines
2)对于形似函数的宏,最好改用inline函数替换#define
num 2:尽可能使用const
1)将某些东西声明为const可帮助编译器侦测出错误用法
2)当const non-const 成员函数有着实质等价的实现时,令nono-const调用const版本可避免重复。
eg:
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]);
//从原始TextBlock转换为const TextBlock,调用const operator[],将const char& 转换为char&.
}
};
num 3:确定对象被使用前已被初始化
1)对内置型对象进行手工初始化,因为c++不保证初始化它们
2)构造函数最好使用成员初始化列表,其排列次序和它们在class中的声明次序相同
3)以local static对象替换non local static对象
当某编译单元内的non-local static对象的初始化动作使用另一个编译单元内的某个nono-local static对象,但所用对象可能尚未初始化。----将non-local static对象搬到自己的专属函数内,这些函数返回一个reference指向它所含的对象。然后用户调用这些函数。保证对象已被初始化
二:构造/析构/赋值运算
num 1:编译器可暗自创建4个成员函数
当class内含reference成员或const成员,必须自己定义赋值操作符
num 2:若不想使用编译器自动生成的函数,就应该明确拒绝
1)所有编译器产出的函数都是public:可将不想使用的函数放入private且只声明不实现
num 3:为多态基类声明virtual析构函数
1)当基类的析构函数为非virtual,是错误做法。如string作为基类
2)希望class成为抽象的class为其声明pure virtual 析构函数。virtual ~AWOV() = 0;
3)带多态性质的基类应声明一个virtual析构函数。若class带有任何virtual函数,它就应该拥有一个virtual析构函数
4)class的设计目的若不是作为基类使用或不是为了具备多态性质,就不应该声明virtual析构函数
num 4:别让异常逃离析构函数
1)析构函数绝对不要吐出异常。析构函数应该捕捉任何异常并吞下它们或结束异常
2)若客户需要对某个操作函数运行期间抛出的异常做出反应,那class应该提供一个普通函数执行该操作
num 5:绝不在构造析构中调用virtual函数
num 6:令operator=返回一个reference to *this
num 7:在operator=中处理自我赋值
eg:Widget& Widget::operator=(const Widget& rhs)
{ if(*this == rhs) return *this; delete pb; pb = new Bitmap(*rhs.pb); return *this;
//2:Bitmap* pOrig = pb; pb = new Bitmap(*rhs.pb); delete pOrig; return *this;
}
num 8:复制对象时勿忘其每一个成分
1)确保复制“对象内的所有成员变量及所有基类成分
2)不要常识以某个copy函数实现另一个copy函数。应该将共同机能放进第三个函数,并有两个copy函数共同调用
三:资源管理
num 1:以对象管理资源
1)在构造函数中获得资源,在析构函数中释放资源
2)使用shared_ptr auto_ptr管理资源。最好使用shared_ptr---计数指针,当引用计数为0时释放对象。当shared_ptr auto_ptr其析构函数调用delete,因此不能与数组相关
num 2:在资源管理类中小心coping行为???
num 3:在资源管理类中提供对原始资源的访问???
num 4:成对使用new delete时要采取相同形式
num 5:以独立语句将newed对象置入智能指针
eg:shared_ptr<Widget> pw(new Widget);
processWidget(pw,priority());
四:设计与声明
num 1:让接口容易被使用,不易被误用
num 2:设计class犹如设计type
1)新type的对象应该如何被创建和销毁?
2)对象的初始化和对象的赋值该有什么样的差别?
3)新typw的对象如果被passed by value,意味着什么?copy 构造函数用来定义一个type的passed by value
4)什么是新type的合法值?
5)你的新type需要配合某个继承图系吗?
6)你的新type需要什么样的转换?
7)什么样的操作符和函数对此新type而言是合理的?
8)什么样的标准函数应该驳回?那些必须声明为private
9)什么是新type的未声明接口?
10)谁该取用新的type成员?
11)是否真的需要一个新type?
num 3:以引用传递代替值传递
1)引用传递更高效
2)该规则并不适合内置类型以及STL的迭代器和函数对象。值传递更适合
num 4:必须返回对象时,别妄想返回其reference??
num 5:将成员变量声明为private
1)protected并不比public更具封装性
num 6:宁以non-member non-friend 替换member函数
num 7:若所有参数皆需类型转换,请为此采用non-member 函数
如算数混合运算,可以使用非成员函数或友元函数进行类型转换
num 8:考虑写出一个不抛异常的swap函数
1)以指针指向一个对象,内含真正数据。当要置换两个对象值,唯一需要做的就是置换其指针---具体实践是将std::swap针对对象特化。
eg:class Widget {
public: void swap(Widget& other) { using std:swap; swap(pImpl,other.pImpl); }
};
namespace std {
template<>
void swap<Widget>(Widget& a,Widget& b) { a.swap(b); }
}
2)偏特化只对类模板有用
3)当std:swap对类型效率不高时,提供一个swap成员函数
4)当提供一个member swap,也提供一个非member swap来调用
五:实现
num 1:尽可能延迟变量定义式的出现时间
num 2:尽量少做转型
num 3:避免返回handles(指针、引用、迭代器)指向对象内部成分
num 4:为异常安全而努力是值得的
num 5:透彻了解inlining
num 6:将文件间的编译依存关系降至最低
1)相依于声明式,不要相依于定义式
六:继承与面向对象设计
num 1:确定public继承是is_a关系
num 2:避免遮掩继承而来的名称
num 3:区分接口继承和实现继承
1)public继承下,子类总是继承父类接口
2)pure virtual函数只具体指定接口继承
3)非纯虚函数具体指定接口继承及缺省实现继承
4)非虚函数指定接口继承和强制实现继承
num 4:考虑virtual函数以外的其他选择
num 5:绝不重新定义继承而来的non-virtual函数
num 6:绝不重新定义继承而来的缺省参数值(默认参数值)
num 7:明智而审慎使用private继承
七:模板与泛型编程
num 1:了解隐式接口与编译期多态
1)class与template都支持接口和多态
2)对class而言接口是显式,以函数签名为中心,多态通过虚函数发生与运行期
3)对template而言,接口是隐式,有效表达式展现。多态通过具现化与函数重载解析发生于编译期
num 2:了解typename双重意义
1)使用template标识嵌套从属类型名称,但不能在基类列表或成员初始化列表内以他作为基类修饰符
num 3:学会处理模板化基类内的名称??
num 4:将与参数无关的代码抽离template
1)template生成多个class和函数,任何template代码都不该与某个参数产生相依关系
effictive c++的更多相关文章
- [effictive c++] 条款04 确定对象被使用前已被初始化
成员初始化 在c和c++ 中,使用为初始化的类型经常会引发不可预料的错误,从而使得我们要花费巨大的时间用于调试查找问题,所以确定对象被使用前已被初始化是个非常好的习惯. 永远在使用之前对对象进行初始化 ...
- Effictive C++ 学习记录
这是前段时间看的书,整理到这里吧,以后查看也方便. 这些条款需要反复查看. 条款01:视C++为一个语言联邦 条款02:尽量用const.enum.inline替换#define 条款03:尽可能的使 ...
- 在对象内部尽量直接訪问实例变量 --Effictive Objective-C 抄书
在对象之外訪问实例变量时,应该总是通过属性来做.在那么在对象内部訪问实例变量的时候,又该怎样呢? 这是 OCer们一直激烈讨论的问题.有人觉得,不管什么情况,都应该通过属性来訪问实例变量;也有人说,& ...
- Effictive Java学习笔记1:创建和销毁对象
建议1:考虑用静态工厂方法代替构造器 理由:1)静态方法有名字啊,更容易懂和理解.构造方法重载容易让人混淆,并不是好主意 2)静态工厂方法可以不必每次调用时都创建一个新对象,而公共构造函数每次调用都会 ...
- BAT的面试经验_摘抄
一.心态 心态很重要! 心态很重要! 心态很重要! 重要的事情说三遍,这一点我觉得是必须放到前面来讲. 找工作之前,有一点你必须清楚,就是找工作是一件看缘分的事情,不是你很牛逼,你就一定能进你想进的公 ...
- 和Java相关的一些好文章(不定期更新)
1.Java 集合类详解 (包括arraylist,linkedlist,vector,stack,hashmap,hashtable,treemap,collection等). 2.Java 理论与 ...
- Android复习指南
基础无外乎几部分:语言(C/C++或java),操作系统,TCP/IP,数据结构与算法,再加上你所熟悉的领域.这里面其实有很多东西,各大面试宝典都有列举. 在这只列举了Android客户端所需要的和我 ...
- 高效的使用STL
高效的使用STL 仅仅是个选择的问题,都是STL,可能写出来的效率相差几倍: 熟悉以下条款,高效的使用STL: 当对象很大时,建立指针的容器而不是对象的容器 1)STL基于拷贝的方式的来工作,任何需要 ...
- 基础1 JavaSe基础
JavaSe基础 1. 九种基本数据类型的大小,以及他们的封装类 boolean 无明确指定 Boolean char 16bits Character byte 8bits Byte short 1 ...
随机推荐
- 2020年腾讯实习生C++面试题&持续更新中(2)
2020年腾讯实习生C++面试题&持续更新中(2) hello,大家好~ 我是好好学习天天,天天编程的天天,一个每天都死磕技术,及时分享的技术宅~ 昨天分享的题目不知道大家是否看过了,以后我计 ...
- ionic + asp.net core webapi + keycloak实现前后端用户认证和自动生成客户端代码
概述 本文使用ionic/angular开发网页前台,asp.net core webapi开发restful service,使用keycloak保护前台页面和后台服务,并且利用open api自动 ...
- MySQL(2)— 数据库的基本操作
二.数据库 2-1.操作数据库(了解) 1.创建数据库 CREATE DATABASE [IF NOT EXIST] myDatabase; 2.删除数据库 DROP DATABASE `myData ...
- Java并发编程入门(二)
1.竞态条件 1.1 定义 当某个计算的正确性取决于多个线程的交替执行时序时,就会发生竞态条件.换句话说,正确的结果要取决于运气. 最常见的竞态条件类型:先检查后执行(Check-Then-Act)操 ...
- Linux上,最常用的一批命令解析【10年精选】
原文链接:https://mp.weixin.qq.com/s/QkqHexs_kOgy_5OwbwyFww 建议点击原文链接查看 不同平台linux客户端连接工具分享: windos终端神器:SSH ...
- PHP设计模式(一)
1)工厂模式 工厂模式是用工厂方法生成对象,而不是直接new一个对象.假设我们在Config命名空间下有一个名叫Db的数据库操作类,用普通的方法,如果我们想去创建一个Db的对象,我们会直接new一个出 ...
- Mybatis配置-简单的使用
导包 基本配置 配置mybatis.config.xml文档 <?xml version="1.0" encoding="UTF-8" ?> < ...
- [CSS布局基础]居中布局的实现方式总结
[原创]码路工人 Coder-Power 大家好,这里是码路工人有力量,我是码路工人,你们是力量. github-pages 博客园cnblogs 做Web开发少不了做页面布局.码路工人给大家总结一下 ...
- 安装superset遇到的坑
实验环境:ubuntu16.04 python环境: 3.6.7 安装参考:https://superset.incubator.apache.org/installation.html 特别提醒: ...
- springboot 启动报错"No bean named 'org.springframework.context.annotation.ConfigurationClassPostProcessor.importRegistry' available"
1.问题 springboot启动报错 "D:\Program Files\Java\jdk-11\bin\java.exe" -XX:TieredStopAtLevel=1 -n ...