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++中实现调试日志输出

《C++中实现调试日志输出》在C++编程中,调试日志对于定位问题和优化代码至关重要,本文将介绍几种常用的调试日志输出方法,并教你如何在日志中添加时间戳,希望对大家有所帮助... 目录1. 使用 #ifdef _DEBUG 宏2. 加入时间戳:精确到毫秒3.Windows 和 MFC 中的调试日志方法MFC

深入理解C++ 空类大小

《深入理解C++空类大小》本文主要介绍了C++空类大小,规定空类大小为1字节,主要是为了保证对象的唯一性和可区分性,满足数组元素地址连续的要求,下面就来了解一下... 目录1. 保证对象的唯一性和可区分性2. 满足数组元素地址连续的要求3. 与C++的对象模型和内存管理机制相适配查看类对象内存在C++中,规

在 VSCode 中配置 C++ 开发环境的详细教程

《在VSCode中配置C++开发环境的详细教程》本文详细介绍了如何在VisualStudioCode(VSCode)中配置C++开发环境,包括安装必要的工具、配置编译器、设置调试环境等步骤,通... 目录如何在 VSCode 中配置 C++ 开发环境:详细教程1. 什么是 VSCode?2. 安装 VSCo

C++11的函数包装器std::function使用示例

《C++11的函数包装器std::function使用示例》C++11引入的std::function是最常用的函数包装器,它可以存储任何可调用对象并提供统一的调用接口,以下是关于函数包装器的详细讲解... 目录一、std::function 的基本用法1. 基本语法二、如何使用 std::function

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

06 C++Lambda表达式

lambda表达式的定义 没有显式模版形参的lambda表达式 [捕获] 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 有显式模版形参的lambda表达式 [捕获] <模版形参> 模版约束 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 含义 捕获:包含零个或者多个捕获符的逗号分隔列表 模板形参:用于泛型lambda提供个模板形参的名

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)