Effective C++ 45-48
45。弄清c++在幕后为你所写。所调用的函数。
假设设置一个空类,c++编译器会声明下面函数:拷贝构造函数。赋值运算符,析构函数,一对取地址运算符函数(const和非const)。而假设你没有声明不论什么构造函数的话,编译器会为你声明一个缺省构造函数。这些函数都是公有的。
编译器生成的缺省构造函数和析构函数实际上什么也不做。生成的析构函数通常是非虚构的,除非继承了一个具有虚析构函数的基类。缺省取地址符仅仅是返回对象的地址,即return this。而拷贝构造函数和赋值运算符,对类的非静态数据成员进行“以成员为单位”逐一拷贝构造或赋值,也就是浅拷贝。
当类中有引用时。默认的拷贝函数无法实现,编译器会报错,有常量也是。有指针是,会发生浅拷贝可是执行上没有错误。对于含有指针,引用和const成员的类须要自定义赋值运算符和复制构造函数。
而假设将派生类中的赋值运算符或拷贝构造函数声明为private,编译器也会拒绝为这个派生类生成对应的赋值运算符和拷贝构造函数。
46.宁可编译和链接时出错,也不要在执行时出错。
当通过编译和链接后,仅仅有极少数情况会让C++抛出异常。如内存耗尽,执行时错误和C++没什么关系。
C++没有执行时检測,要尽量避免执行时错误。
对于执行时错误。在一个执行中没有错误,并不表示其就是正确的了。由于每次程序执行的状态都不一样。
而避免执行时错误的一般方法是对设计做一些小小的修改,就能够在编译期间消除可能产生的执行时错误。一般设计在程序中添加新的数据类型,以在编译时检測数据的安全性。
对于一个日期类,有构造函数:Date(int day,int month,int year)。实现这个类面临的问题是对day和month进行合法性检測。假设不进行检測,因为其内部逻辑可能会导致一些执行时错误。
一种简单的方法是使用枚举
enum Month {Jan = 1,Feb = 2,....,Dec =12};
而构造函数改为:
Date(int day,Month month,int year);
可是这样做没有多大优点。由于枚举类型不用初始化,即直接 Date d(1,Month m,2014),能通过编译。可是执行时出错。
即想免除执行时检查,又要保证足够的安全性,选择使用一个类来实现month。
class Month{
public:
static const Month Jan(){return 1;}//这里事实上是调用隐式构造函数。事实上返回值为 Month(1);
//....
static const Month Dec(){return 12;}//使用静态函数。返回一个常量。防止任意修改
int toInt() const
{return number;}
private:
Month (int n):number(n){}
const int number;
};
这里调用类的静态成员返回相应的Month。而构造函数隐藏。防止用户自己去创建新的month。但即使有了这种类,用户还是能够指定一个非法的month,例如以下:
Month* m;
Data(1, *m ,2014);
消除全部的执行时检測是不切实际的。但将检查由执行时转移到编译或链接时一直值得努力的目标。这样做,会使程序更小,更快,更可靠。
47.确保非局部静态对象在使用前被初始化。
使用对象前一定要初始化。
非局部静态对象是指 : 定义在全局或名字命名空间内。或在一个类中被声明为static。或在一个文件范围内被定义为static。就是值所有的对象。去掉非静态 的局部变量 和函数内的静态变量。
当类依赖与这些非局部静态对象时,如在 一个文件里有一个全局对象theCountry, 在另外一个文件里有一个对象theCity。对city的初始化依赖与country的初始化。而程序的正确执行依赖于它们的初始化顺序。但确定非局部静态对象初始化的正确顺序非常困难,在多个编译单元中确保每一个这种对象初始化是非常困难的,尤其当程序变得更加复杂添加很多其它的这种非局部静态对象的情况下。
单一模式,将每一个非局部静态对象转移到函数中。声明其为static,其次,让函数返回这个对象的引用。这样用户就能够通过函数调用来指明对象,即用函数内部的static对象来代替非局部静态对象。由于对于函数的静态对象什么时候被初始化。c++明白的指出了。这种还有一个优点是假设这个模拟非局部静态对象从没被调用,也就永远没有对象构造和销毁的开销。简单的样例:
class country{....};
country& theCountry(){
static country tc;//定义和初始化theCountry
return tc;//返回它的引用。
}
48.重视编译器警告。
一般程序猿都会忽略编译器警告。毕竟没有出错。要理解编译器的各种警告的含义。书上举个样例:
class A{
public:
virtual void f() const{cout<<"fA";}
};
class B:public A{
public:
virtual void f(){cout<<"fB";}
};
书上说有编译器在这里会出一个 B::f() hides virtual A:::f()的错误,即A中声明的f函数并没有在B中又一次定义。可是被B中新声明的非const的f函数给隐藏了。可是这样没有实现多态。对于使用声明为A的指针的B的对象指向f函数的话,会调用A中的f函数。可是我用的vs2012中并没有提示这条警告。
Effective C++ 45-48的更多相关文章
- Effective C++ -----条款48:认识template元编程
Template metaprogramming(TMP,模板元编程)可将工作由运行期移往编译期,因而得以实现早期错误侦测和更高的执行效率. TMP可被用来生成“基于政策选择组合”(based on ...
- Effective Java 45 Minimize the scope of local variables
Principle The most powerful technique for minimizing the scope of a local variable is to declare it ...
- effective c++ Item 48 了解模板元编程
1. TMP是什么? 模板元编程(template metaprogramming TMP)是实现基于模板的C++程序的过程,它能够在编译期执行.你可以想一想:一个模板元程序是用C++实现的并且可以在 ...
- 读书笔记 effective c++ Item 48 了解模板元编程
1. TMP是什么? 模板元编程(template metaprogramming TMP)是实现基于模板的C++程序的过程,它能够在编译期执行.你可以想一想:一个模板元程序是用C++实现的并且可以在 ...
- Effective Java Index
Hi guys, I am happy to tell you that I am moving to the open source world. And Java is the 1st langu ...
- Effective Java 第三版——71. 避免不必要地使用检查异常
Tips 书中的源代码地址:https://github.com/jbloch/effective-java-3e-source-code 注意,书中的有些代码里方法是基于Java 9 API中的,所 ...
- [转]python3之模块psutil系统性能信息
转自:https://www.cnblogs.com/zhangxinqi/p/9106265.html 阅读目录 1.psutil模块安装 2.获取CPU信息 3.内存信息 4.磁盘信息 5.网络信 ...
- 《Python自动化运维之路》 系统基础信息模块(一)
系统性能收集模块Psutil 目录: 系统性能信息模块psutil 系统性能信息模块psutil psutil能够轻松实现获取系统运行的进程和系统利用率包括(CPU,内存,磁盘 和网络)等.主要用于系 ...
- python3之模块psutil系统性能信息
psutil是个跨平台库,能够轻松实现获取系统运行的进程和系统利用率,包括CPU.内存.磁盘.网络等信息. 它主要应用于信息监控,分析和限制系统资源及进程的管理.它实现了同等命令命令行工具提供的功能, ...
- wazhu之agent功能详解
一.日志数据收集 日志数据收集是从服务器或设备生成的记录中收集的实时过程.此组件可以通过文本文件或Windows事件日志接收日志.它还可以通过远程syslog直接接收日志,这对防火墙和其他此类设备 ...
随机推荐
- Python多线程、多进程
1.from multiprocessing import Process ; from threading import Thread 2.进程之间的数据传输 ,一般会使用到pipes, qu ...
- Web开发中跨域的几种解决方案
同domain(或ip),同端口,同协议视为同一个域,一个域内的脚本仅仅具有本域内的权限,可以理解为本域脚本只能读写本域内的资源,而无法访问其它域的资源.这种安全限制称为同源策略. 出于安全考虑,HT ...
- Struts之 拦截器配置 ( Interceptor )
struts2体系结构的核心就是拦截器. 以链式执行,对真正要执行的方法(execute())进行拦截.首先执行Action配置的拦截器,在Action和Result执行之后,拦截器再一次执行(与先前 ...
- struts2.1.6存在中文乱码的bug
如题,后续版本中已解决:可以通过添加filter的方式解决.
- Xml的读取
using System; using System.Collections.Generic; using System.Linq; using System.Web; namespace WebAp ...
- React容器组件和展示组件
Presentational and Container Components 展示组件 - 只关心它们的样子. - 可能同时包含子级容器组件和展示组件,一般含DOM标签和自定的样式. ...
- TCP简要讲解
TCP被称为面向连接.面向流,可靠的传输层协议.它与UDP最大的不同在于它是可靠性的,所以需要可靠传输的时候就需要用到TCP,如果需要快速传输则选择UDP. TCP的连接又称为3次握手. 1).客 ...
- python函数参数的区别
在运用python的过程中,发现当函数参数为list的时候,在函数内部调用list.append()会改变形参,与C/C++的不太一样,查阅相关资料,在这里记录一下. python中id可以获取对象的 ...
- day13-迭代器、三元表达式、列表推导式、字典生成式、生成器与递归
目录 迭代器 可迭代对象 迭代器对象 for循环原理 三元表达式(三目表达式) 列表推导式 字典生成式 zip()方法 生成器 生成器表达式 递归 递归的两个阶段 迭代器 迭代器即迭代的工具,迭代是一 ...
- Eclipse 使用前的配置
一,修改eclipse对jdk的依赖项 1.查看设置的编译器编译版本:设置成本地jdk一致的版本 点击窗口->首选项 找到java 选择编辑器,查看现在的编译jdk版本 改成本地jdk版本 jd ...