读书笔记:《More Effective C++》

More Effective C++

Basics

reference & pointer

  • reference 必定有值,pointer 可以为空
  • reference 声明时必须定义,必须初始化
  • reference 无需测试有效性,pointer 必须测试是否为 null
  • reference 可以更改指向对象的值,但是无法指向其他对象,pointer 可以指向其他对象

cast

  • c 转型:(type)expr
  • c++ 转型:cast(expr)
    • static:just like c 转型,无类型检查
    • dynamic:base-》derived,有类型检查
    • reinterpret:二进制重新解释,受制于编译期平台
    • const:amend costness or volatileness

polymorphically array

  • 使用base array pointer处理derived array,会导致未定义行为

default constructor

  • 贸然提供default constructor需要在运行时测试是否值有效
  • 缺乏default constructor会带来束缚
    • 无法创建对象数组,只能创建指针数组
    • 可以使用 placement new 创建对象数组,显式使用析构函数的 placement delete删除
      • 使用 delete 删除 placement new 创建的对象会导致未定义行为

Operators

implicit conversion

  • 单参数构造函数(或其余参数有默认值)
  • 隐式类型转换操作符
  • 隐式类型转换导致难以发现的问题:
    • 构造函数标记为 explicit
    • 定义显式类型转换:asType()
    • 只能进行一次隐式转换,多次类型转换不会有问题

increment/decrement & prefix/postfix

  • 【++i,累加后取出】:clazz& operator++()
    • 返回引用,允许++++i
  • 【i++,取出后累加】:const clazz operator++(clazz)
    • 额外参数区分,参数不得使用
    • 返回const,不允许i++++
    • 为了使得逻辑一致,用increment prefix来实现increment postfix

override &&、||、,

  • 重载&&、||导致问题:
    • &&、||,条件满足/不满足会提前终止
    • 重载后,左右式作为函数参数都会被执行
  • 重载,导致问题:
    • 逗号表达式,先求解左式,再求解右式,值为右式
    • 重载后,函数无法保证执行顺序

new, operator new, placement new, delete, operator delete, []

  • new:无法重载,operator new + constructor + placement new
  • operator new:重载分配内存的行为
    • void * operator new(size_t)
  • placement new:重载在指定内存上构建对象
    • void * operator new(size_t, void* location)
  • delete:destructor + operator delete
    • placement new 分配的内存无法通过 operator delete 删除
      • 显式析构 + 分配内存对应的方式删除
        • new->delete
        • malloc->free

Exceptions

  • 程序运行出错时设置状态变量或返回错误码,无法保证异常被程序处理,程序可能会异常运行;而抛出异常如果未进行捕获,程序便会立刻终止。

delete in destructor

  • 将资源分配在对象内,在析构函数中删除内存
  • 析构函数必然会被调用,防止因为异常产生未删除内存

exceptions in constructor

  • 构造函数异常,部分构造的对象不会自动析构
    • 将所有成员变量视为RAII对象,使用shared_ptr

exceptions in destructor

  • 使用uncaught_exception判断析构函数时候正有一个exception在作用中
  • 正在处理异常的析构函数抛出未捕获的异常,会导致程序终止
  • 不应该让析构函数抛出异常,析构函数应该是异常安全的
stack unwinding

C++异常处理,栈展开机制:

  1. 异常被抛出
  2. 当前作用域,搜索异常处理程序catch块
  3. 未搜索到,销毁该作用域内的局部对象
    1. 调用析构函数出现异常,程序终止,std::terminate
  4. 退出当前作用域,继续搜索上一级作用域
  5. 未处理异常,std::terminate,std::abort,生成核心转储文件

catch exceptions param pass

  • 处理异常和函数传参的区别
    • 异常永远会被复制,因为异常处理程序超出函数调用作用域
      • 按值捕获,复制两次,传参一次,复制一次
      • 引用捕获,复制一次
    • 不发生隐式类型转换,仅能发生继承类型转换
    • 异常按catch的顺序处理,没有最优类型匹配
      • 子类异常应该写在基类异常的前面,否则不会被处理

catch exceptions by reference

  • catch by pointer:无法判断是否应该删除资源
  • catch by value:子类异常由基类捕获会导致对象切割,仅剩下基类行为
  • catch by reference:推荐使用,且仅复制一次

exception specifications

  • 在定义中指定函数可能抛出的异常
    • 未指定则表明可能抛出任何异常
    • 指定noexcept表明不会抛出任何异常
  • 抛出未定义异常会导致unexpected函数调用
    • unexpected的默认行为是调用terminate
    • 可以使用set_unexpected自定义处理行为
  • 可以提供额外的异常定义说明,但是违反说明的异常会直接导致程序终止

exception handling cost

  • 异常处理会使得程序效率下降5%~10%

Efficiency

80-20 rule

  • 80%的资源被20%的代码占用,不仅需要大幅优化代码,更重要的是找到那20%的瓶颈
  • 不根据经验和直觉,而是使用profiler来分析那20%代码

lazy evaluation

  • 如果计算不是绝对必要,使用lazy evaluation替代eager evaluation

over-eager evaluation

  • cache:缓存已计算的值
  • prefetch:预先从磁盘读取可能访问的值,每次读取磁盘多于请求量的值

avoid temporary

  • 临时对象≠局部对象,临时对象是没有名字的栈对象
    • 函数传参,隐式类型转换
      • by value:隐式转换为临时对象
      • by const reference:隐式转换为临时对象
      • by non-const reference:隐式转化无法被触发,因为对临时对象的修改无法反馈会引用对象,因此编译报错
    • 函数返回值
      • 使用+=替代+
      • 返回值优化RVO

return value optimize

  • 直接返回构造函数,避免临时对象的生成

overload to avoid implicit type conversions

  • 使用函数重载来避免没有函数重载时的隐式类型转换

supply += and +

  • 操作符+=往往比+有更高的效率
  • 重载操作符的时候提供+=版本

third party library

  • 相同功能的第三方库,在性能方面,可能会有不同的设计,和不同的效率偏重

inherit cost

  • virtual functions:
    • 每个class包含一个虚函数表
    • 每个对象包含一个虚函数表的指针
    • virtual 函数无法 inline,因为 inline 需要在编译期复制函数调用,而 virtual 需要运行期才能确定调用函数
  • multiple inheritance & virtual base classes
    • 菱形继承需要virtual base class,额外的指针开销
  • runtime type identification,RTTI
    • typeid获取class相应的type_info,需要额外的指针指向type_info
    • 可以在虚函数表中添加指向type_info的指针

Techniques

virtual constructor & non-member function

  • virtual constructor
    • factory function 根据参数生成不同的子类
  • virtual copy constructor
    • 每个子类添加虚函数clone调用拷贝构造函数
  • virtual non-member function
    • 每个子类添加虚函数实现具体操作
    • non-member function inline化,调用相应虚函数

limit class instance count

  • 关键资源有限,其对应的类示例数目也有限
    • 零或一:单例模式
    • n:在构造函数中添加static示例计数,使用make函数显式构造,防止隐式构造增加计数

fore constructor in/out of heap

  • in heap:对象有自杀能力,delete删除内存
  • out of heap:对象不会造成内存泄漏
  • 重载:
    • static void * operator new(size_t)
    • static void operator delete(void *)

smart pointers

  • 智能指针无法在子类和基类之间转换,需要借助于隐式类型转换操作符

reference count

  • 多数对象共享少数的实值,实值的产生和删除成本高,使用引用计数来记录共享对象的个数,在引用计数为零时删除

proxy class

  • 表示一个观念上不存在的对象
    • 实现[][]重载,proxy对象为Array1D

multi virtual function

  • 根据多个子类的类型判断调用的函数
    • func(base*, base*)
      • 虚函数+RTTI,运行时类型辨识
      • 虚函数重载
      • 自行实现虚函数表

Miscellany

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/191030.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

POSTGRESQL中如何利用SQL语句快速的进行同环比?

1. 引言 在数据驱动的时代,了解销售、收入或任何业务指标的同比和环比情况对企业决策至关重要。本文将深入介绍如何利用 PostgreSQL 和 SQL 语句快速、准确地进行这两种重要分析。 2. 数据准备 为了演示,假设我们有一张 sales 表,存储了销…

【PyTorch】线性回归

文章目录 1. 代码实现1.1 一元线性回归模型的训练 2. 代码解读2.1. tensorboardX2.1.1. tensorboardX的安装2.1.2. tensorboardX的使用 1. 代码实现 波士顿房价数据集下载 1.1 一元线性回归模型的训练 import numpy as np import torch import torch.nn as nn from torch.ut…

[足式机器人]Part4 南科大高等机器人控制课 Ch00 课程简介

本文仅供学习使用 本文参考: B站:CLEAR_LAB 南科大高等机器人控制课 Ch00 课程简介 1. What is this course about?2. Tentative Schedule暂定时间表 1. What is this course about? Develop a solid foundation in robot modeling and control to co…

深度学习:什么是知识蒸馏(Knowledge Distillation)

1 概况 1.1 定义 知识蒸馏(Knowledge Distillation)是一种深度学习技术,旨在将一个复杂模型(通常称为“教师模型”)的知识转移到一个更简单、更小的模型(称为“学生模型”)中。这一技术由Hint…

亚马逊首席技术官2024年科技预测

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗?订阅我们的简报,深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同,从行业内部的深度分析和实用指南中受益。不要错过这个机会,成为AI领…

二叉树遍历及应用

文章目录 前言构建二叉树前序遍历中序遍历后序遍历二叉树的结点个数二叉树的叶节点个数二叉树的高度二叉树第K层结点个数 前言 二叉树的遍历及应用主要是运用了递归、分治的思想。在这一篇文章,小编将介绍二叉树的前序遍历、中序遍历、后序遍历,求二叉树…

Fiddler抓包工具之fiddler设置手机端抓包

fiddler设置手机端抓包 安卓手机抓包 第一步:配置电脑和安卓的相关设置 1、手机和fiddler位于同一个局域网内;首先从fiddler处获取到ip地址和端口号: ,点击online,最后一行就是ip地址 2、路径:Tools》O…

【ASP.NET CORE】数据迁移 codefirst

已经写好实体类,使用add-migration生成数据迁移语句,注意如果项目中有多个dbcontext需要使用 -context 名称,指定下需要使用的dbcontext add-Migration Address -context mvcsqlcontext运行后会生成两个文件 2. 使用Update-Database语句更…

#Css篇:实现一个元素水平和垂直居中实现左右固定,中间自身适应布局 左侧固定 右侧自适应

实现一个元素水平和垂直居中 元素示例&#xff1a; <div class"container"><div class"centered-element">居中的内容</div> </div>flex布局 .container {display: flex;justify-content: center;align-items: center;height: …

Web自动化测试详解

做测试的同学们都了解&#xff0c;做Web自动化&#xff0c;我们主要用Selenium或者是QTP。 有的人可能就会说&#xff0c;我没这个Java基础&#xff0c;没有Selenium基础&#xff0c;能行吗&#xff1f;测试虽然属于计算机行业&#xff0c;但其实并不需要太深入的编程知识&…

力扣129. 求根节点到叶节点数字之和

递归 思路&#xff1a; 递归子问题&#xff1a; sum 左子树值 右子树值子树值 上一级值 * 10 当前节点值 int sum prevSum * 10 root->val; calcSum(root->left, sum) calcSum(root->right, sum); 终止条件&#xff1a; 如果节点为 nullptr&#xff0c;则值为…

C++学习之路(十六)C++ 用Qt5实现一个工具箱(为屏幕颜色提取功能增加一个点击复制的功能)- 示例代码拆分讲解

上篇文章&#xff0c;我们用 Qt5 实现了在小工具箱中添加了《颜色代码转换和屏幕颜色提取功能》功能。今天我们把屏幕颜色提取的功能再扩展一下&#xff0c;让它可以点击复制吧。下面我们就来看看如何来规划开发这样的小功能并且添加到我们的工具箱中吧。 老规矩&#xff0c;先…

【无标题】RTI-DDS实现C/S通信

RTI-DDS&#xff08;Real-Time Innovations Data Distribution Service&#xff09;是一款实时消息传递中间件&#xff0c;用于实现分布式系统中的实时数据传输和通信。使用RTI-DDS&#xff0c;可以实现Client-Server&#xff08;CS&#xff09;通信模式&#xff0c;该协议在车…

leetcode 12.2 每日一题 拼车(近双百解法,绝对通俗易懂)

1094. 拼车 车上最初有 capacity 个空座位。车 只能 向一个方向行驶&#xff08;也就是说&#xff0c;不允许掉头或改变方向&#xff09; 给定整数 capacity 和一个数组 trips , trip[i] [numPassengersi, fromi, toi] 表示第 i 次旅行有 numPassengersi 乘客&#xff0c; 接…

编译ubuntu kernel

下载kernel源码 sudo apt-get install linux-source 下载完成后&#xff0c;kernel源码位于/usr/src/ 。 编译ubuntu kernel make oldconfig make menuconfig make make modules_install make install

数字图像处理(实践篇)十三 数据增强之给图像添加噪声!

目录 一 涉及的函数 二 实践 一 涉及的函数 skimage.util.random_noise( ) skimage.util.random_noise(image, modegaussian, seedNone, clipTrue, **kwargs) 函数的功能&#xff1a;为浮点型图片添加各种随机噪声。 输入&#xff1a; ①image&#xff1a;输入图像&…

【C/C++笔试练习】公有派生、构造函数内不执行多态、抽象类和纯虚函数、多态中的缺省值、虚函数的描述、纯虚函数的声明、查找输入整数二进制中1的个数、手套

文章目录 C/C笔试练习选择部分&#xff08;1&#xff09;公有派生&#xff08;2&#xff09;构造函数内不执行多态&#xff08;3&#xff09;抽象类和纯虚函数&#xff08;4&#xff09;多态中的缺省值&#xff08;5&#xff09;程序分析&#xff08;6&#xff09;重载和隐藏&a…

【开箱即用】前后端同时开源!周末和AI用Go语言共同研发了一款笔记留言小程序!

大家好&#xff0c;我是豆小匠。 真的是当你在怀疑AI会不会取代人类的时候&#xff0c;别人已经用AI工具加速几倍的生产速度了… 周末体验了和AI共同开发的感受&#xff0c;小项目真的可以一人全干了… 本次实验使用的AI工具有两个&#xff1a;1. GitHub Copilot&#xff08;…

ubuntu 更换国内镜像

备份 cd /etc/aptcp sources.list sources.list.bakup修改源为清华源 sed -i s/archive.ubuntu.com/mirrors.aliyun.com/g sources.list更新软件源 apt-get update其他源如下&#xff1a; mirrors.ustc.edu.cn 中科大 mirrors.163.com 163 mirrors.aliyun.com 阿里云

前端文本省略号后面添加复制文字

前端文本省略号后面添加复制文字 1、效果图 2、代码展示 <div class"link-content-wrap" click"copyLinkText"><div class"link-content">{{ shareResult.url || }} </div><span class"show-ellipsis" click&…