C++易错细节——基础问题
1、过分积极的注释
注释必须和代码一起维护,不应该描述显而易见的事,或把别的地方已说清楚的东西在说一遍。
2、幻数(字面常量)
字面常量没有语义,也没有真正的内存地址。
所以不可以取其地址,也不可以用于初始化普通引用,例如:long &r1 = 40000;
但是const long &r1 = 40000 却是合法的。
所以,尽量不要使用字面常量,而应该使用枚举常量和初始化过的常变量
3、全局变量
全局变量增加了模块间的耦合。
当全局变量(静态变量)用来初始化的值不能在编译时就计算妥当,则该初始化动作就会拖到运行期,这容易导致致命错误。可以使用单价模式解决这个问题
4、未能区分函数重载和形式参数默认值
函数重载主要用于一组抽象意义相同,但实现不同的函数。
而形式参数默认值主要出于简化,为函数提供更简洁的接口。
5、对引用的认识误区
(1)由于引用没有地址,声明引用的引用、指向引用的指针或引用的数组都是不合法的。
(2)引用不可以有常量性或挥发性,所以不可以用关键字const或volatile来修饰引用。
(3)任何能作为左值的复杂表达式都能作为引用的初始化物
(4)char *cp = reinterpret_cast<char *>(a); // 对cp取地址则错误
reinterpret_cast<char *&>(a) = cp;// 正确,
(5)指向数组的引用保留了数组的尺寸信息,而指针则不保留(这个性质有时在数组名作为函数实参进行调用函数时使用)
(6)可以声明函数的引用:
ex: int f( double );
int (* const pf) (double) = f; // pf是指向函数f()的常量指针
int (&rf) (double) = f; // rf是函数f()的引用
可以把引用或函数本身(隐式)转换成指向函数的指针,再使用反引用语法。
6、对常量(性)的认识误区
(1)字母常量没有地址,永远不可以作左值
(2)int ci; const int *ip2 = &ci 的const只描述了通过ip2对ci的操作限制,而不是对ci的一般操作的限制。也就是说ci = 5可以,而*ip2 = 5不可以。
7、C语言的精妙之处
(1)如果?表达式中的两个选择结果都是左值,则该表达式本身就是个左值。
(2)case语句的标签必须是整形常量性的表达式(必须在编译时就要算出case中值)
8、区分可访问性和可见性
(1)要提供一个放置各种适当前置声明的专用头文件
(2)把class的接口与其实现分离,从而要达到真正的数据隐藏之境,而其不二法门则是运用桥接设计模式
(3)使用桥接,任何对于Class C实现的修改,只要不改变Class C的接口,影响就会被牢牢地钳制在一个单独的实现文件。