Effective C++ 摘记(三)

2024-01-05 08:48
文章标签 c++ effective 摘记

本文主要是介绍Effective C++ 摘记(三),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!


(六)、继承与面向对象设计

三十二、确定public继承塑膜出is-a关系

如果子类为public继承基类, 

class D:public B

{

.....
}

表明,子类的对象也一定属于基类,则适用于基类的事情也适用于子类。


三十三、避免遮掩继承来的名称

基类的重载函数一旦在子类被重写后,其他的同名函数无法访问。

例如:

class Base{
private:int x;
public:void mf1();vodi mf1(int x);
};
class Derived: public Base{
private:int y;
public:void mf1(double );
};  
Derived d;
d.mf1(2.0);		//父类中的mf1(), mf1(int) 在子类中被覆盖,子类中只有 mf1(double) 可见。

可以在子类中提前注明 using Base::functions 来标明函数的作用范围。



或者使用转交函数来实现 选择继承基类的部分函数。


转交函数,即在子类的同名函数中调用父类的函数,从而避免子类的函数覆盖父类的函数的现象出现。

inline转交函数的另一个作用是为那些不支持using声明式的老旧编译器开辟一条新路,将继承而得的名称汇入 derived作用域内。


三十四、区分接口继承和实现继承

接口继承(只继承接口,实现由子类自己来写)和实现继承(继承父类的接口和实现,子类的函数用父类的实现)不同,在public继承下,derived classes总是继承base class的接口。

纯虚函数( pure virtual )指定接口继承(基类不对函数进行实现 声明为 virtual xxx = 0;,子类必须对函数进行实现);

虚函数( impure virtual ) 指定接口和默认实现(基类声明函数,同时实现函数的一个缺省版本,子类如果也对函数进行实现,则使用子类的函数,否则使用缺省函数)

一般函数( non-virtaul )指定接口和强制实现(子类必须不能够再是实现函数,否则会出现覆盖<也可以采用方式进行避免>)。


三十五、考虑虚函数以外的选择

类的public成员函数调用类的私有成员函数,私有虚函数在父类被调用的时候自动多态,基本保留何时调用的权力,子类拥有修改功能的权力;

function函数指针对象使得函数指针更加灵活;

古典策略模式:

使得不同的功能通过继承HealthCalcFunc改变。

三十六、绝不定义继承的非虚 ( non-virtual ) 函数

重修继承的非虚函数导致函数的访问由指向对象的指针或引用类型决定。

如果基类的public函数 F 为non-virtual,则子类public继承之后,重新实现了F, 则使用基类指针pB(指向某子类对象D)和子类指针pD(指向相同的子类对象D)来调用函数 F 其结果不同,因为F 不是虚函数,不会进行动态绑定。


三十七、绝不定义继承的默认参数值

基类指针指向子类对象,重载的虚函数的默认参数来自于基类;(因此,不要重新定义继承的虚函数的默认参数值)

将默认参数函数声明为普通成员函数,通过 non-virtual 函数调用私有的 virtual 虚函数即可。


三十八、用复合塑膜出has-a和实现关系

has-a:对象的包含关系;

实现:对象对另一个对象进行具体特化。

在应用域,复合意味着has-a(有一个), 在实现域,复合意味着is-implemented-in-terms-of(根据某物实现出)


三十九、审慎使用private继承

私有继承表达的是实现关系 is-implemented-in-terms-of ,子类使用父类提供的接口,但是不继承;

能用复合不用私有继承;

私有继承方式:


子类中的void onTick转换为private,防止客户误以为可以调用该函数。


复合方式:

这样Widget的子类就不会修改onTick函数了,将内部类移出,换做声明可以降低耦合;

private继承的空基类的大小实际为0,一般对象大小不能为0



四十、审慎使用多重继承

使用虚基类导致速度变慢;

多重继承中使用公有继承继承接口,私有继承完成实现关系。


(七)、模板与泛型编程

四十一、隐式接口与编译多态

class是显示接口——函数签名,运行多态——虚函数;

template是隐式接口——有效表达式,编译多态——模板具体化与函数重载解析。

四十二、typename双重含义

模板声明中与class没有任何区别;

嵌套从属类型的显式指定,不能出现在基类列表和初始化列表中;

四十三、处理模板化基类名称

继承模板化基类的名称不能像继承一样使用:通过this->名字修饰、using基类<T>::名字、或者基类<T>::名字一共三种修饰方式。第三种导致虚函数功能失效。

四十四、参数无关代码抽离模板

将与模板无关的非类型参数转移到类内;

尽量降低与模板无关的类型参数的膨胀度。

四十五、运用成员函数模板接受兼容类型

成员函数使用函数模板兼容更多类型;

函数模板声明后的copy构造和编译器生成的并不同,需要单独处理。

四十六、类型转换时为模板定义非成员函数

对于模板化的类要支持双操作运算符重载,首先必须是非成员函数,另外为了能让模板具体化必须将函数定在类体内部,因此只能将之声明为友元类型。(并非模板类内的友元函数必须类内定义)。

四十七、使用traits类表现类型信息

STL五大迭代器:

1.输入迭代器:向前,一次一步,只读一次,istream_iterator

2.输出迭代器:向前,一次一步,只写一次, ostream_iterator

3.前向迭代器:向前,一次一步,可读可写多次,单向列表。

4.双向迭代器:向前向后,一次一步,可读可写多次,listsetmap

5.随机迭代器:向前向后,一次多步,可读可写多次,vectordequestring

实现迭代器累加操作时候需要根据迭代器类型执行不同的操作方式,这种判断属于编译时期的判断,不应该使用if语句!

可以根据iterator_traits提供的类别标签区分迭代器类型,类别标签是空结构体类型,将标签作为函数参数,可以保证编译器能在编译时期对类型进行检查。

现在就可以把doAdvance封装起来自动完成编译期类型判断。

四十八、模板元编程

让某些事情变得容易可能,将某些工作从运行期转移到编译期;

分支——借由模板特化实现;

循环——借由递归完成;

优点:保证度量单位的正确、优化矩阵运算生成客户定制设计模式实现品;

避免了生成某些特殊类型不适合的代码。

(八)、定制newdelete

四十九、new-handler行为

set_new_handler指定内存分配失败时调用的函数。

五十、newdelete合理替换时机

改善性能,内存对齐,heap错误调试,收集heap信息。

五十一、newdelete固守常规

new含有无限循环分配内存,无法分配调用new-handler,处理0字节和超额申请;

delete处理null指针和超额申请。

五十二、写了placement new就要写placement delete

placement new在已有的缓冲区内申请对象;

不要掩盖已有的版本。

(九)、杂项

五十三、不要忽视警告

严肃对待警告信息;

不过度依赖警告信息。

五十四、熟悉TR1标准库

智能指针、Boost库。

五十五、熟悉Boost

社群、网站;

TR1组件实现品。


原文地址:http://www.cnblogs.com/fanzhidongyzby/archive/2012/11/18/2775603.html

这篇关于Effective C++ 摘记(三)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/572315

相关文章

C++11范围for初始化列表auto decltype详解

《C++11范围for初始化列表autodecltype详解》C++11引入auto类型推导、decltype类型推断、统一列表初始化、范围for循环及智能指针,提升代码简洁性、类型安全与资源管理效... 目录C++11新特性1. 自动类型推导auto1.1 基本语法2. decltype3. 列表初始化3

C++11右值引用与Lambda表达式的使用

《C++11右值引用与Lambda表达式的使用》C++11引入右值引用,实现移动语义提升性能,支持资源转移与完美转发;同时引入Lambda表达式,简化匿名函数定义,通过捕获列表和参数列表灵活处理变量... 目录C++11新特性右值引用和移动语义左值 / 右值常见的左值和右值移动语义移动构造函数移动复制运算符

C++中detach的作用、使用场景及注意事项

《C++中detach的作用、使用场景及注意事项》关于C++中的detach,它主要涉及多线程编程中的线程管理,理解detach的作用、使用场景以及注意事项,对于写出高效、安全的多线程程序至关重要,下... 目录一、什么是join()?它的作用是什么?类比一下:二、join()的作用总结三、join()怎么

C++中全局变量和局部变量的区别

《C++中全局变量和局部变量的区别》本文主要介绍了C++中全局变量和局部变量的区别,全局变量和局部变量在作用域和生命周期上有显著的区别,下面就来介绍一下,感兴趣的可以了解一下... 目录一、全局变量定义生命周期存储位置代码示例输出二、局部变量定义生命周期存储位置代码示例输出三、全局变量和局部变量的区别作用域

C++中assign函数的使用

《C++中assign函数的使用》在C++标准模板库中,std::list等容器都提供了assign成员函数,它比操作符更灵活,支持多种初始化方式,下面就来介绍一下assign的用法,具有一定的参考价... 目录​1.assign的基本功能​​语法​2. 具体用法示例​​​(1) 填充n个相同值​​(2)

c++ 类成员变量默认初始值的实现

《c++类成员变量默认初始值的实现》本文主要介绍了c++类成员变量默认初始值,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录C++类成员变量初始化c++类的变量的初始化在C++中,如果使用类成员变量时未给定其初始值,那么它将被

C++中NULL与nullptr的区别小结

《C++中NULL与nullptr的区别小结》本文介绍了C++编程中NULL与nullptr的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编... 目录C++98空值——NULLC++11空值——nullptr区别对比示例 C++98空值——NUL

C++ Log4cpp跨平台日志库的使用小结

《C++Log4cpp跨平台日志库的使用小结》Log4cpp是c++类库,本文详细介绍了C++日志库log4cpp的使用方法,及设置日志输出格式和优先级,具有一定的参考价值,感兴趣的可以了解一下... 目录一、介绍1. log4cpp的日志方式2.设置日志输出的格式3. 设置日志的输出优先级二、Window

从入门到精通C++11 <chrono> 库特性

《从入门到精通C++11<chrono>库特性》chrono库是C++11中一个非常强大和实用的库,它为时间处理提供了丰富的功能和类型安全的接口,通过本文的介绍,我们了解了chrono库的基本概念... 目录一、引言1.1 为什么需要<chrono>库1.2<chrono>库的基本概念二、时间段(Durat

C++20管道运算符的实现示例

《C++20管道运算符的实现示例》本文简要介绍C++20管道运算符的使用与实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录标准库的管道运算符使用自己实现类似的管道运算符我们不打算介绍太多,因为它实际属于c++20最为重要的