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++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

C++中初始化二维数组的几种常见方法

《C++中初始化二维数组的几种常见方法》本文详细介绍了在C++中初始化二维数组的不同方式,包括静态初始化、循环、全部为零、部分初始化、std::array和std::vector,以及std::vec... 目录1. 静态初始化2. 使用循环初始化3. 全部初始化为零4. 部分初始化5. 使用 std::a

C++ vector的常见用法超详细讲解

《C++vector的常见用法超详细讲解》:本文主要介绍C++vector的常见用法,包括C++中vector容器的定义、初始化方法、访问元素、常用函数及其时间复杂度,通过代码介绍的非常详细,... 目录1、vector的定义2、vector常用初始化方法1、使编程用花括号直接赋值2、使用圆括号赋值3、ve

如何高效移除C++关联容器中的元素

《如何高效移除C++关联容器中的元素》关联容器和顺序容器有着很大不同,关联容器中的元素是按照关键字来保存和访问的,而顺序容器中的元素是按它们在容器中的位置来顺序保存和访问的,本文介绍了如何高效移除C+... 目录一、简介二、移除给定位置的元素三、移除与特定键值等价的元素四、移除满足特android定条件的元

Python获取C++中返回的char*字段的两种思路

《Python获取C++中返回的char*字段的两种思路》有时候需要获取C++函数中返回来的不定长的char*字符串,本文小编为大家找到了两种解决问题的思路,感兴趣的小伙伴可以跟随小编一起学习一下... 有时候需要获取C++函数中返回来的不定长的char*字符串,目前我找到两种解决问题的思路,具体实现如下:

C++ Sort函数使用场景分析

《C++Sort函数使用场景分析》sort函数是algorithm库下的一个函数,sort函数是不稳定的,即大小相同的元素在排序后相对顺序可能发生改变,如果某些场景需要保持相同元素间的相对顺序,可使... 目录C++ Sort函数详解一、sort函数调用的两种方式二、sort函数使用场景三、sort函数排序

Java调用C++动态库超详细步骤讲解(附源码)

《Java调用C++动态库超详细步骤讲解(附源码)》C语言因其高效和接近硬件的特性,时常会被用在性能要求较高或者需要直接操作硬件的场合,:本文主要介绍Java调用C++动态库的相关资料,文中通过代... 目录一、直接调用C++库第一步:动态库生成(vs2017+qt5.12.10)第二步:Java调用C++

C/C++错误信息处理的常见方法及函数

《C/C++错误信息处理的常见方法及函数》C/C++是两种广泛使用的编程语言,特别是在系统编程、嵌入式开发以及高性能计算领域,:本文主要介绍C/C++错误信息处理的常见方法及函数,文中通过代码介绍... 目录前言1. errno 和 perror()示例:2. strerror()示例:3. perror(

C++变换迭代器使用方法小结

《C++变换迭代器使用方法小结》本文主要介绍了C++变换迭代器使用方法小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录1、源码2、代码解析代码解析:transform_iterator1. transform_iterat

详解C++中类的大小决定因数

《详解C++中类的大小决定因数》类的大小受多个因素影响,主要包括成员变量、对齐方式、继承关系、虚函数表等,下面就来介绍一下,具有一定的参考价值,感兴趣的可以了解一下... 目录1. 非静态数据成员示例:2. 数据对齐(Padding)示例:3. 虚函数(vtable 指针)示例:4. 继承普通继承虚继承5.