《Effective C++ 3th》——让自己习惯C++

本文探讨C++编程的几个关键原则,包括将其视为语言联邦,提倡使用const、enum和inline代替#define,强调对象初始化的重要性,以及如何正确运用const确保代码安全性。


 
视c++为一个语言联邦
尽量以const,enum,inline替换#define
尽可能使用const
确定对象被使用前已先被初始化
 

视C++为一个语言联邦

C++: C with Classes。c++可以看成是C语言与OOP的结合体,并引入STL新特性:
在这里插入图片描述

  • C: C语言部分;
  • Object-Oriented C++: C++中OOP部分;
  • Template C++: C++中模板部分(泛型);
  • STL: C++标准库(泛型)。
     
     

尽量以const,enum,inline替换#define

以编译器替换预处理器。#define缺点:

  • 预处理阶段处理,错误提示不明确;
  • 盲目替换,导致存在多份;
  • 替换时展开导致语义错误

使用const,enum和inline替换的优点:

  • 编译阶段处理,明确的错误提示;
  • 单份代码。
///////////////////////////////////////
// 预处理器处理
// 预处理阶段 --> 盲目替换 --> 编译器
// - 预处理阶段处理,错误提示不明确
// - 盲目替换,导致存在多份
// - 替换时展开导致语义错误
#define ASPECT_RATIO 1.653

///////////////////////////////////////
// 编译器处理
// 编译器
// - 编译阶段处理,明确的错误提示
// - 单份代码
const double AspectRatio = 1.653;

class GamePlayer
{
private:
	enum { NumTurns = 5 }; // 类专属常量enum定义

private:
	static const int NumTurns; // 类专属常量const定义
};
const int GamePlayer::NumTurns = 5;

// inline模板定义
template<typename T>
inline T callWithMax(const T &a, const T &b)
{
	return (a > b ? a : b);
}

方法:

  1. 对于单纯常量,最好以const或者enum替换#define;
  2. 对于形似函数的宏,最好改用inline函数替换#define。
     
     

尽可能使用const

const定义不被改动的语义约束。特点:

  • const出现在 * 号左边,表示被指物是常量;const出现在 * 号右边,表示指针自身是常量;如果出现在 * 号两边,表示被指物和指针都是常量;
 char a = "Hello World!";
 const char *p = a; // a不可变
 char* const p = a; // p不可变
 const char* const p = a; // p和a都不可变
  • 如果被指物不可变,const写在类型前和写在类型后意义相同:
  void f1(const Widget *pw);
  void f2(Widget const *pw); // 意义相同
  • const成员函数:bitwise constness和“概念上的常量性”。
     
     

确定对象被使用前已先被初始化

解决使用与初始化的顺序问题:永远在使用对象之前将它初始化
区别 初始化赋值

ABEntry::ABEntry(const std::string &name, const std::string &address,
                const std::list<PhoneNumber> &phones)
{
    // 此处为赋值,并非初始化
    theName = name;
    theAddress = address;
    thePhones = phones;
    numTimesConsulted = 0;
}

ABEntry::ABEntry(const std::string &name, const std::string &address,
                const std::list<PhoneNumber> &phones)
    :theName(name),
	theAddress(address),
	thePhones(phones),
	numTimesConsulted(0) // 此处为值列初始化
{
}

解决使用和初始化的先后问题:

// 将每个non-local static对象搬到自己的专属函数内。
// 单例模式常见手法
class FileSystem {...};
FileSystem& tfs()
{
    static FileSystem fs;
    return fs;
}

class Directory {...};
Directory::Directory(params)
{
    ...
    std::size_t disks = tds().numDisks();
    ...
}

方法:

  1. 内置对象进行手工初始化,因为C++不保证初始化它们;
  2. 构造函数使用初值列初始化;初值列成员变量的排列次序应该与tam在class中声明次序相同;
  3. 以local static对象替换non-local static对象解决“跨单元的初始化次序”问题。
【SCI一区复现】基于配电网韧性提升的应急移动电源预配置和动态调度(下)—MPS动态调度(Matlab代码实现)内容概要:本文档围绕“基于配电网韧性提升的应急移动电源预配置和动态调度”主题,重点介绍MPS(Mobile Power Sources)动态调度的Matlab代码实现,是SCI一区论文复现的技术资料。内容涵盖在灾害或故障等极端场景下,如何通过优化算法对应急移动电源进行科学调度,以提升配电网在突发事件中的恢复能力与供电可靠性。文档强调采用先进的智能优化算法进行建模求解,并结合IEEE标准测试系统(如IEEE33节点)进行仿真验证,具有较强的学术前沿性和工程应用价值。; 适合人群:具备电力系统基础知识和Matlab编程能力,从事电力系统优化、配电网韧性、应急电源调度等相关领域研究的研究生、科研人员及工程技术人员。; 使用场景及目标:①用于复现高水平期刊(SCI一区、IEEE顶刊)中关于配电网韧性与移动电源调度的研究成果;②支撑科研项目中的模型构建与算法开发,提升配电网在故障后的快速恢复能力;③为电力系统应急调度策略提供仿真工具与技术参考。; 阅读建议:建议结合前篇“MPS预配置”内容系统学习,重点关注动态调度模型的数学建模、目标函数设计与Matlab代码实现细节,建议配合YALMIP等优化工具包进行仿真实验,并参考文中提供的网盘资源获取完整代码与数据。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值