C++增强笔记
这个是我在学习
C++
语言中所记录的笔记,有可能会存在错误和遗漏,并且我有一点点C语言基础, 会大量的提及C语言与C++的不同,从而造成笔记晦涩; 另外C++的学习是一个长期且艰难的过程,因此本文进行了切分;
1 C++对象模型(深入版)
- 对象模型是一种描述计算机程序中的对象及其相互关系的方式。在面向对象编程中,每个对象都有一个类型、属性和方法。对象之间可以相互通信,从而实现特定的任务。
- 在对象模型中,对象类是描述对象的通用模板。每个类都有一组属性和方法,它们定义了该类的行为。当我们创建一个对象时,我们使用该类来生成该对象,并且该对象将具备该类所定义的属性和方法。
- 对象之间可以通过消息传递进行通信。消息包含了一个方法调用请求,接收者也就是被调用的对象会执行该方法,并将结果返回给发送者。这种方式实现了数据和操作的封装,使得程序的实现更加模块化和灵活。
- 总之,对象模型是一种以对象为基本单位的编程范式,通过对对象的抽象,实现了高度复用性、可扩展性和可维护性的程序设计。
1.1 关于对象(Object Lessons)
在C++中农, Point3d可以 使用独立的抽象数据类型(abstract data types ADT)
class Point3d {
public:
Point3d(float x = 0.0, float y = 0.0, float z = 0.0) : _x(x), _y(y), _z(z) {}
float x() const { return _x; }
float y() const { return _y; }
float z() const { return _z; }
private:
float _x;
float _y;
float _z;
};
inline std::ostream& operator << (std::ostream &os, const Point3d &pt)
{
os << '{' << pt.x() << pt.y() << pt.z() << '}';
return os;
}
双层或者三层class体系
class Point {
public:
Point(float x = 0.0) : _x(x) {}
float x() const { return _x; }
void x(float val = 0.0) { _x = val; }
protected:
float _x;
};
class Point2d : public Point {
public:
Point2d(float x = 0.0, float y = 0.0) : Point(x), _y(y) {}
float y() const { return _y; }
void y(float val = 0.0) { _y = val; }
protected:
float _y;
};
class Point3d : public Point2d {
public:
Point3d(float x = 0.0, float y = 0.0, float z = 0.0) : Point2d(x, y), _z(z) {}
float z() const { return _z; }
void z(float val = 0.0) { _z = val; }
protected:
float _z;
};
inline std::ostream& operator << (std::ostream &os, const Point3d &pt)
{
os << '{' << pt.x() << pt.y() << pt.z() << '}';
return os;
}
更近一步,不管哪一种形式,他们都可以被参数化,也可以坐标参数化
template<class T>
class Point3d {
public:
Point3d(T x = 0.0, T y = 0.0, T z = 0.0) : _x(x), _y(y), _z(z) {}
T x() const { return _x; }
T y() const { return _y; }
T z() const { return _z; }
private:
T _x;
T _y;
T _z;
};
template<class T>
inline std::ostream& operator << (std::ostream &os, const Point3d<T> &pt)
{
os << '{' << pt.x() << pt.y() << pt.z() << '}';
return os;
}
也可以是坐标类型和坐标数目都参数化
template<class T, int dim>
class Point {
public:
Point();
Point(T coords[dim]) {
for (int index = 0; index < dim; index++) {
_coords[index] = coords[index];
}
}
T& operator[](int index) {
assert(index < dim && index >= 0);
return _coords[index];
}
T operator[](int index) const {
assert(index < dim && index >= 0);
return _coords[index];
}
private:
T _coords[dim];
};
template<class T, int dim>
inline std::ostream& operator << (std::ostream &os, const Point<T, dim> &pt)
{
os << '{';
for (int i = 0; i <= dim - 1; i++) {
os << pt[i] << ";";
}
os << '}';
return os;
}
加上封装后的布局成本
- virtual function机制
- virtual base class机制
1.1.1 C++对象模型
在C++中,有两种class data member: static
和nonstatic
有三种class function member: static
,nonstatic
与virtual
已知下面的class point的声明:
class Point {
public:
Point(float val);
virtual ~Point();
float x() const;
static int PointCount();
protected:
virtual ostream& print(ostream &os) const;
float _x;
static int _point_count;
}
那么这个class point在机器中将会怎么样表现呢?也就是说我们如何模型(modeling)出各种data member与function member呢?
C++ 对象模型(The C++ Object Model )
C++之父的最终设想
Stroustrup 当初设计 ( 当前亦仍占有优势) 的 C++ 对象模型是从简单对象 模 型派生而来的 ,并对内存空间和存取时间做了优化 .在此模型中,Nonstatic data members 被配置于每一 个 class object 之内 ,static data members 则被存放在所有 的 class object 之外. Static 和 nonstatic function members 也被放在所有的 class object 之外. Virtual functions 则以两个步骤支持之 :
- 每一 个 class 产 生 出一 堆 始 向 virtual functions 的指 针 ,放 在 表 格 之 中.这 个表格被称为 virtuaI table (vtbl)
- 每 一 个 class object 被添加一 个指针,指向相关 的 virtual table . 通 常 这个指针被 称为 vptr。
vptr的设定 ( setting ) 和重 置 ( resetting ) 都 由 每 一 个
class
的constructor
,destructor
和copy assignment
运算符自动完成,每一个class
所关联 的type_info object
( 用以支持 runtime type identification, RTTI) 经由virtualtable被指出来,通 常是放在表格的第一个slot
处 .
图 1.3 说明 C++ 对象模型如何应用于前面所说的 Point class 身上。这个模型的主要优点在于它的空间和存取时间的效率 ;主要缺点则是 ,如果应用程序代码本身未曾改变,但所用到的 class objects 的 nonstatic data members 有所修改( 可能是增加、移除或更改 ), 那么那些应用程序代码 同样得重新编译 .关于这 点,前述的双表格模型就提供了较大的 弹性 ,因为它多提供了一 层间接性 。不过,它也因此付出空间和执行效率两方面的代价就是了。
加上继承(Adding Inheritance)
C++支持的单一继承
class Libray_materials {...};
class Book : public Libray_materials {...};
class Rental_book : public Book {...};
C++也支持多重继承
// 原本iostream的实现方式
class iostream:
public istream,
public ostream {...};
甚至,继承关系也可以指定为虚拟(virtaul,也就是共享的意思)
class istream : virtual public ios {...};
class ostream : virtual public ios {...};
graph TD ios --> ostream ios --> istream ostream --> iostream istream --> iostream
在虚拟继承的情况下, base class 不管在继承串链中被派生 ( derived ) 多少次 , 永远只会存在一个实体 ( 称为 subobject ) 。例如 iostream 之中就只有 virtual ios base class 的一个实体。
自C++ 2.0 起才新导入的 virtual base class,需要一些间接的 base class 表现 方法 。Virtual base class 的原始模型是在 class object 中为每一个有关联的 virtaul base class
加上一个指针 .其它演化出来的模型则若不是 导入 一个 virtual base,class table ,就是扩充原已存在的 virtual table ,以便维护每 一个 virtual base class 的位置。
对象模型如何影响程序(How to Object Model Effects Programs)
不同的对象模型 ,会导致 “现有的程序代码必须修改” 以及 “ 必须加入新 的程序代码” 两个结果 。
例如下面这个 函数 ,其中 class X 定义了一个 copy constructor ,一个 virtual destructor ,和一个 virtual function foo
X foorbar()
{
X xx;
x *px = new X;
// foo是一个virtaul function
xx.foo();
px->foo();
delete px;
delete xx;
}
这个函数有可能在内部转换为
void foobar(X &_result)
{
// 构造_result
// 用result用来取代 local xx
_result.X::X();
// 扩展 X *px = new X;
px = _new(sizeof(x));
if (px != 0)
px->X::X();
// 扩展xx.foo(),但不使用virtual机制
// 以result取代xx.foo(&result);
// 使用virtual机制扩展px->foo( )
(*px->vtbl[2])(px)
// 扩展 delet e px ;
if (px != 0) {
( *px->vtbl [1])(px); // destructor
_delete (px) ;
}
// 不需使用 named return statement
// 不需要摧毁 local object xx
return ;
}
1.1.2 对象的差异(An Object Distinction)
C++程序设计模型直接支持三种programming paradiagms(程序设计典范);
- 程序模型(prcedural model),就像C一样,
C++
当然也支持它,字符串的处理就是一个例子,我们可以使用字符数组以及str*
函数集(定义在标准的C函数库中)
char boy [ ] =”Da nny” ;
char *p_son;
...
p_son = new char[strlen(boy) + 1];
strcpy(p_son, boy) ;
...
if (!strcmp(p_son, boy))
take_to_disneyland(boy);
- 抽象数据类型模型(abstract data type model, ADT). 该模型所谓的“抽象”是和一组表达式(public接口)一起提供, 而其运算定义
仍然隐而未明. 例如下面的
String class
:
String girl == "Anna";
String daughter;
...
// String::operator=();
daughter = girl;
...
// St ring ::ope rator== ( ) ;
if (girl == daughter)
take_t o_disneyland (girl);
- 面向对象模型(object-oriented model)。在此模型中有一些彼此相关的类型,通过一个抽象的base class(用以提供共通接口)被封装起来。 Library_ materials class 就是一个例子,真正的 subtypes 例如 Book、Video、Compac仁Disc 、Puppet、Laptop等等都可以从那里派生而来:
void
check_in(Libr ar y_ma te主主als *pmat)
{
if (pmat->late())
pmat->fine() ; pma t ->check in ( ) ;
if ( Lend er *plend = pma t -> reser ved ( ) ) pma t->notif y ( plend ) ;
}
纯粹以一种 paradigm 写程序 ,有助于整体行为的 良好稳固.然而如果混 合 了不同的 paradigms,就可能会带来让人惊 吓的后果 ,特别是在没有谨慎处理的情 况下 。最常见的疏忽发生在当你以一 个 base class 的具体实体如:
Library_materails thinql;
来完成某种多态(polymorphism)局面时:
graph TD Library_materails --> Book
class Book : public Library_materials { ... };
Book book ;
// 喔欧:thingl不是一个Book !
// book被裁切(sliced )了.
// 不过 thingl仍保有一个 Library_materials
thingl = book ;
// 喔欧:调用的是Library_rnaterials::check_in()
thingl.check_in();
而不是通过 base class 的 pointer 或 reference 来完成多态局面:
// OK :现在 thing2 参考到 book
Library_materials & thing2 = boo k ;
// OK :现在引发的是 Book : :check in ( )
thing2 .check_in ( ) ;
虽然你可以直接或间接处理继承体系中的一个 base class object ,但只有通过 pointer 或 reference 的间接处理 ,才支持OO程序设计所需的多态性质.上个例子中的 thing2 的定义和运用,是OO paradigm 中一个良好的例证 thingl 的定 义和运用则逸出了OO的习惯 :它反映 的是一个 ADT paradigm 的良好行为,thingl的行为是好是坏 ,视程序员 的意图而定 .在此范例 中,它的行为非常有可 能不是你要的!
在OO paradigm 之中 ,程序员需要处理 一个未知实体 ,它的类型虽然有所界 定,却有无穷可能.这组类型受限于其继承体系 ,然而该体系理论上没有深度和 广度的限制.原则上 ,被指定 的 obje ct 的真实类型在每一个特定 执行点之前 ,是 无法解析的.在 C++ 中,只有通过 pointers 和 references 的操作才能够完成 4 相反地 ,在 ADT paradigm 中程序员处理的是一 个拥有固定而单一类型的实体 , 它在编译时期就已经完全定义好了 .举个例子 ,下面这组声明:
//描述 obj ect s :不确定类型 Libra r_materials *px = retrieve_ some_ma ter主al {} ; Libr ar_ma ter ials & rx = *px ;
//描述已知物: 不可能有令 人惊讶的结果产生 Libra r_ma teria l s dx 步 px ;
你绝对没有办法确定地说出 px 或 rx 到底指向何种类型的 objects ,你只能 够说它要不就是 Library_materials object ,要不就是后者的一个子类型 ( subtype ) 。 不过 ,我们倒是可以确定 ,dx 只能是 Libraη materials class 的一个 object o 本 节稍后 ,我会讨论为什么这样的行为虽然或许未如你所预 期,却是良好的行为 e
虽然 “对于 object 的多态操作” 要求此 object 必须可以经由一个 pointer 或 reference 来存取 ,然而 C++ 中的 pointer 或 reference 的处理却不是多态的必要 结果.想想下面的情况 :
//没有多态 ( 译注z 因为操作对象不是 class obj ect ) int *pi;
//没有语言所支持的多态 ( 译注:因为操作对象不是 class obj ect ) void *pvi;
II ok : class x 视为一 个 ba se class ( 译注:可以有多态的效果) x *px;
在 C忡,多态只存在于一个个的 public class 体系中.举个例子 ,px 可能指 向自我类型的一个 object ,或指向以 public 派生而来的一个类型 ( 请不要把不良 的转型操作考虑在内) o Nonpublic 的派生行为以及类型为 void * 的指针可以说 是多态 ,但它们并没有被语言明白地支持,也就是说它们必须由程序员通过明白 的转型操作来管理 ( 你或许可以说它们并不是多态对象的一线选手 ).
C++ 以下列方法支持多态 :
?.经由一组隐 含 的转化 操作 .例如把 一 个 derived class 指针转化 为 个指 向其 public base type 的指针 :
shape *ps = new circle() ;
- 经由 virtual function 机制 :
ps ->rotate ( ) ;
- 经由 dynamic_cast 和 type id 运算符 :
if ( circle 女pc = dynamic_ca st< circle* > ( ps ) )
多态的主要用途是经 由一个共 同的接口来影响类型的封装 ,这个接口通常被 定义在 一 个抽象 的 base class 中.例如Library_materials class 就为 Book 、 阿deo 、Puppet 等 subtype 定义了一个接口.这个共享接口 是以 virtual function 机 制引发的 ,它可以在执行期根据 obje ct 的真正类型解析出到底是哪一个函数实体 被调用 。经由这样的操作 :
Librar y_ma ter ial->check ou t ( ) ;
我们的代民可以避免由于 “借助某 一特定 library 的 materials,而导致变动无常 。 这表只使得 当类型有所增加 、修改 、或删减时,我们的程序代码不 需改变’\ 而且也使 个新灼 L.;易r_ary_materials subtype 的供应者不需要重新写出 “对继承体 系中的所有类型都共渍 ” 白行为和操作 .
考虑一下这样的码,
v 0j_ d rotate ( )( d2.tum, con ,ζ X *pointer , t ’ 1st X & r ef erence )
//在执行期之前 ,无法决定到底词用哪一个 rotate ( ) 实体 ( *pointer ) .r otate ( ) ; r ef erence . rotate ( ) ;
//下面这个操作总是调用 X : :rotate ( ) datum .rotate ( ) ;
ma in ( ) {
Z z ; / / Z 是 x 的一个子类型
rotate ( z , & z , z ) ; return O ; ….-. .11
经由 pointer 和 reference 完成的两个 “ 函数调用操作” 会被动态完 成 !此例 中它们都调用 Z::rotateO o 经由 datum 完成的 “ 函数调用操作” 则可能 ( 或可能 不) 经由 virtual 机制 .不过 ,它反正总是调用 X·:rotateO 就是了.( 这就是所 谓的 “ 编译素养” 问题 :不管经由 datum 所调用的 virtual function 采不采 用 virtual 机制 ,从语意来说 ,结果都是相同的D 4.2 节对此有更详细的讨论 )
需要多少内存才能够表现 一个 class obje ct?一般而言要有:
·其 non static data members 的总和大 小:
·加 上 任 何 由于 alignment ( 译注) 的需求而填补 ( padding ) 上去的空 间 ( 可 能存在 于 members 之 间 ,也可 能存在于集合体边界 ) .
译注:al ignment 就是将数值调整到某数的倍数.在 32 位计算机上 ,通 常 alignment 为 4 bytes ( 32 位 ) ,以使 bus 的 “运输量” 达到最高效率 。
·加上 为 了支持 virtual 而由内部产 生 的任何额外负担 ( overhead ) •
一个指针2 ,不管它指向哪一种数据类型 ,指针本身所需的内存大小是固定 的.举个例子 ,下面有一 个 ZooAnimal 声明z
class ZooAnimal { publ ic : ZooAnimal ( ) ; virtual 句 Z ooAnima l ( ) ;
1.2 构造函数语意学(The Semantics of constructors)
1.2.1 Default Constructor的建构操作
1.2.2 Copy Constructor的建构操作
1.2.3 程序转换语意学(Program Transformation Semantics)
1.2.4 成员们的初始化队伍(Member Initialization List)
1.3 Data语意学(The Semantics of Data)
1.3.1 Data Member的绑定
1.3.2 Data Member的布局
1.3.3 Data Member的存取
1.3.4 “继承"与Data Member
1.3.5 对象的效率
1.3.6 指向Data Member的指针
1.4 Function语意学(The Semantics of Function)
1.4.1 纯虚函数的存在(Presence of a Pu re Virtual Function)
1.4.2 虚拟规格的存在 (Presence of a Virtual Specification )
1.4.3 虚拟规格中const 的存在
1.4.4 重新考虑class的声明
1.5 构造、析构、拷贝语意学
1.6 执行期语意学
1.7 站在对象模型的类端
2 构造函数
2.1 普通构造(默认构造函数)
在C++中,构造函数是一种特殊的成员函数,其名称与类名称相同,用于初始化实例化的对象的成员变量。构造函数没有返回类型声明,并且通常被声明为公共成员函数。当创建一个对象时,构造函数被自动调用。
普通构造函数(也称为默认构造函数)是不带参数的构造函数。它可以通过一个空的参数列表来定义,如下所示:
cpp
class MyClass {
public:
MyClass() {
// 构造函数代码
}
};
这个类的构造函数不需要任何参数,并且在创建对象时会自动调用。我们可以在构造函数中添加代码来初始化类的各种成员变量,例如:
class MyClass {
public:
MyClass() {
num = 0;
str = "";
}
private:
int num;
string str;
};
在上面的例子中,我们设置了两个私有成员变量num和str的初始值。
此外,如果您想初始化一个常量成员变量,您需要使用构造函数的成员初始化列表,如下所示:
class MyClass {
public:
MyClass() : constVar(10) {
// 构造函数代码
}
private:
const int constVar;
};
在上面的例子中,我们使用构造函数的成员初始化列表来初始化constVar常量成员变量。
总之,构造函数是用于初始化对象的重要部分。在设计类时,需要考虑到对象的初始化方式,以确保所有成员变量都被正确地初始化。
2.2 拷贝构造
在C++中,拷贝构造函数是一种特殊的构造函数,用于将一个对象复制到另一个对象。它通常用于按值传递参数或以值返回对象的情况下,从而创建一个新的独立对象。拷贝构造函数也可以用于初始化一个对象数组或在创建对象时调用默认构造函数。
拷贝构造函数有以下格式:
class MyClass {
public:
MyClass(const MyClass& other) {
// 拷贝构造函数代码
}
};
在上面的代码中,参数other是对同类对象的引用。我们可以使用other来访问它的成员变量,并将其复制到当前对象中。
以下是一个简单的拷贝构造函数示例:
class MyClass {
public:
int num;
// 普通构造函数
MyClass(int n) : num(n) {}
// 拷贝构造函数
MyClass(const MyClass& other) : num(other.num) {}
};
int main() {
MyClass obj1(10);
MyClass obj2 = obj1; // 调用拷贝构造函数
cout << obj1.num << endl; // 输出 10
cout << obj2.num << endl; // 输出 10
return 0;
}
在上面的代码中,我们定义了一个MyClass类,它具有一个整数类型的成员变量num。我们定义了一个普通构造函数和一个拷贝构造函数来初始化num成员变量。在主函数中,我们创建了一个名为obj1的MyClass对象,并将其值设置为10。然后,我们使用obj1初始化了另一个MyClass对象obj2,这将调用拷贝构造函数。最后,我们输出obj1和obj2的num成员变量,结果均为10。
需要注意的是,默认情况下,C++会提供一个默认的拷贝构造函数,该函数执行浅拷贝。如果要实现深拷贝,则需要自定义拷贝构造函数来复制指向动态分配内存的指针或其他资源。
2.3 移动构造
在C++11之后,引入了移动语义的概念,也就是通过将资源所有权从一个对象转移到另一个对象,从而提高程序性能。移动构造函数是一种特殊的构造函数,用于实现移动语义。它通常用于将临时对象的值移动到新对象中。
移动构造函数有以下格式:
class MyClass {
public:
MyClass(MyClass&& other) noexcept {
// 移动构造函数代码
}
};
在上面的代码中,参数other是对同类对象的右值引用。我们可以使用std::move(other)来访问其成员变量,并将其移动到当前对象中。需要注意的是,在移动构造函数中必须使用noexcept关键字进行标记,以确保不会抛出异常。
以下是一个简单的移动构造函数示例:
class MyClass {
public:
int* data;
size_t size;
// 普通构造函数
MyClass(size_t s) : data(new int[s]), size(s) {}
// 移动构造函数
MyClass(MyClass&& other) noexcept : data(other.data), size(other.size) {
other.data = nullptr;
other.size = 0;
}
// 析构函数
~MyClass() {
delete[] data;
}
};
int main() {
MyClass obj1(10);
MyClass obj2(std::move(obj1)); // 调用移动构造函数
cout << obj2.size << endl; // 输出 10
cout << obj1.size << endl; // 输出 0
return 0;
}
在上面的代码中,我们定义了一个MyClass类,它具有一个指向int类型的动态数组data和一个size_t类型的size成员变量。我们定义了一个普通构造函数来初始化data和size成员变量,并使用delete[]释放data内存。
然后,我们定义了一个移动构造函数,它将data和size成员变量从临时对象other中移动到当前对象中。在移动完之后,我们将临时对象other的data指针设置为nullptr,以确保不会删除已经释放的内存。
在主函数中,我们创建了一个名为obj1的MyClass对象,并将其大小设置为10。然后,我们使用std::move(obj1)将值移动到新的MyClass对象obj2中,这将调用移动构造函数。最后,我们输出obj1和obj2的size成员变量,结果分别为0和10。
需要注意的是,移动构造函数只能用于右值引用参数,不能用于左值引用或常量引用参数。
2.4 委托构造
在C++11之后,引入了委托构造函数的概念。委托构造函数是一种特殊的构造函数,可以在一个构造函数中调用另一个构造函数来完成对象的初始化。这使得代码更加简洁,减少了冗余代码的编写。
委托构造函数有以下格式:
class MyClass {
public:
MyClass(int num, string str) : num(num), str(str) {}
MyClass() : MyClass(0, "") {}
};
在上面的代码中,我们定义了一个MyClass类,具有两个私有成员变量num和str。我们定义了一个带参数的构造函数,以及一个不带参数的构造函数。在不带参数的构造函数中,我们通过调用带参数的构造函数并传递默认值来实现对象的初始化。这就是委托构造函数的应用。
需要注意的是,如果要在委托构造函数中使用成员初始化列表,则必须在委托构造函数调用前执行。例如:
class MyClass {
public:
MyClass(int num, string str) : num(num), str(str) {}
MyClass() : MyClass(0, "") {
// 委托构造函数调用后,还可以添加其他构造函数代码
}
};
在上面的代码中,我们先调用委托构造函数,然后可以在构造函数中添加其他代码。
总之,委托构造函数是一种非常有用的C++11语言特性,可以使代码更加简洁和可读。
3 移动语意
移动语义是C++11中引入的一种新特性,通过将资源的所有权从一个对象转移到另一个对象来提高程序的性能
3.1 左值与右值
移动语意的前提: 右值引用
定义可以认为是赋值表达式的左右边;
- 左值 (lvalue): 处于赋值表达式左边
- 右值 (rvalue):处于赋值表达式右边
但是上面的定义也太粗糙了吧!
int a = 0; // a 是左值
int b = 0; // b 是左值
int c = a + b; // c 是左值,但是a与b发生了一次右值转换
那么我们写一个例子,强行右值作为左值
int foo()
{
return 0;
}
int main()
{
foo() = 2;
return 0;
}
已启动生成…
1>------ 已启动生成: 项目: obj, 配置: Debug x64 ------
1>obj.cpp
1>C:\Program Files\Microsoft Visual Studio\2022\Community\MSBuild\Microsoft\VC\v170\Microsoft.CppCommon.targets(693,5): error MSB6006: “CL.exe”已退出,代码为 2。
1>D:\work\test\obj\obj.cpp(8,14): error C2106: “=”: 左操作数必须为左值
1>已完成生成项目“obj.vcxproj”的操作 - 失败。
========== 生成: 成功 0 个,失败 1 个,最新 0 个,跳过 0 个 ==========
我们可以清楚的明白,函数不可以赋值,但是下面的例子呢?
int& foo()
{
static int a = 10;
return a;
}
int main()
{
foo() = 2;
return 0;
}
但是此时我们编译成功了?
我们稍微读一下代码就可以明白,
我们并不是对函数进行赋值,而是对函数的返回值进行赋值
3.2 可修改的左值
下面看一个例子
const int a = 1;
a = 2;
那a
是左值还是右值,很明显,左值;
为什么呢?
于是定义需要继续精化。不是所有的左值都可以被赋值。可赋值的左值被称为 可修改左值 (modifiable lvalues) 。C99标准定义可修改左值为:
[…] 可修改左值是特殊的左值,不含有数组类型、不完整类型、const 修饰的类型。如果它是
struct
或union
,它的成员都(递归地)不应含有 const 修饰的类型。
(未完待续。。。)
3.3 完美转发
提到完美转发,就有必要先说一下,什么是转发,什么样的转发才称得上是完美转发。
在 C++ 中,转发指的就是函数之间的参数传递(例如函数 f1
接收了一个参数 a
,而后又将此参数 a
传递给了其函数体内调用的另一个函数 f2
)。
而完美转发指的就是在函数之间传递参数的过程中,参数在传递后的属性保持不变(如左值仍是左值,右值仍是右值,const
修饰也会保留)。
3.4 移动语意
#include <iostream>
#include <vector>
using namespace std;
template <class T>
void print_vector(vector<T> &vec)
{
for (auto v : vec) {
cout << "data: " << v << endl;
}
}
int main()
{
vector<int> A = {1, 2, 3, 4};
vector<int> B;
cout << "A size: " << A.size() << "addr :" << &A << endl;
print_vector(A);
cout << "B size: " << B.size() << "addr :" << &B << endl;
print_vector(B);
B = move(A);
cout << "移动之后" << endl;
cout << "A size: " << A.size() << "addr :" << &A << endl;
print_vector(A);
cout << "B size: " << B.size() << "addr :" << &B << endl;
print_vector(B);
return 0;
}
4 智能指针
智能指针是一种 C++ 的语言特性,用于管理动态分配内存的生命周期。它可以自动跟踪一个对象被多少个指针引用,并在不需要时释放对象占用的内存。 C++ 智能指针的实现一般采用 RAII(Resource Acquisition Is Initialization)技术,即资源获取即初始化。当一个智能指针对象被创建时,它会自动申请分配所需的内存空间,并将所指对象的指针保存在自己的成员变量中。当这个智能指针对象被销毁时,它会自动释放其所占用的内存空间。
4.1 std::unique_ptr
std::unique_ptr 是 C++ 11 标准中提供的一种智能指针,用于管理独占式所有权的资源。 它采用了 RAII 技术,可以自动管理资源的生命周期。
std::unique_ptr 的最大特点是“独占性”,即同一时间内只能有一个 std::unique_ptr 指向一个资源(内存块、文件句柄等)。当 std::unique_ptr 对象被销毁时,它所管理的资源也会被自动释放,从而避免了内存泄漏和资源泄露等问题。
int main() {
// 创建一个指向 int 类型的 unique_ptr
std::unique_ptr<int> p1(new int(10));
std::cout << *p1 << std::endl; // 输出 10
// 将 p1 转移给 p2,并释放 p1 原本所管理的内存
std::unique_ptr<int> p2 = std::move(p1);
std::cout << *p2 << std::endl; // 输出 10
std::cout << *p1 << std::endl; // 错误,p1 已不再指向有效内存,此时直接就会崩溃
// 使用 make_unique 创建一个指向数组的 unique_ptr
std::unique_ptr<int[]> p3 = std::make_unique<int[]>(5);
for (int i = 0; i < 5; ++i) {
p3[i] = i;
}
for (int i = 0; i < 5; ++i) {
std::cout << p3[i] << " ";
}
std::cout << std::endl; // 输出 0 1 2 3 4
return 0;
}
4.2 std::shared_ptr
shared_ptr
是C++11中的一种智能指针类型,可以用于管理动态分配的对象,以防止内存泄漏。
使用shared_ptr
时需要注意以下几点:
shared_ptr
管理的对象必须是通过new
关键字动态分配出来的。- 如果多个
shared_ptr
指向同一个对象,那么该对象的引用计数会增加,当所有shared_ptr
都销毁时,对象的引用计数会减少。当引用计数为0时,对象会被自动删除。这样保证了动态分配的对象不会发生内存泄漏。
class MyClass {
public:
MyClass() { std::cout << "MyClass constructor" << std::endl; }
~MyClass() { std::cout << "MyClass destructor" << std::endl; }
void sayHello() { std::cout << "Hello, world!" << std::endl; }
};
int main() {
std::shared_ptr<MyClass> ptr1(new MyClass);
{
std::shared_ptr<MyClass> ptr2 = ptr1; // 引用计数+1
std::shared_ptr<MyClass> ptr3(ptr1); // 引用计数+1
ptr2->sayHello();
} // 引用计数-2
ptr1->sayHello();
return 0;
}
4.3 std::weak_ptr
weak_ptr
也是C++11中的一个智能指针类型,它可以被用来协助shared_ptr
进行管理动态分配的对象。与shared_ptr
不同的是,weak_ptr
不会增加对象的引用计数,因此不会对对象的生命周期产生影响。
weak_ptr
通常用于需要访问shared_ptr
所管理的对象,但又不希望影响该对象生命周期的情况,例如:
- 避免循环引用:当存在多个对象相互引用时,使用
weak_ptr
可以避免出现循环引用导致对象无法正确释放的问题。 - 延迟初始化:当对象的创建成本较高时,可以使用
weak_ptr
延迟初始化对象,只有在需要访问对象时才创建对象。这样可以提高程序的性能。
下面是一个使用weak_ptr
的例子:
#include <iostream>
#include <memory>
class MyClass {
public:
MyClass() { std::cout << "MyClass constructor" << std::endl; }
~MyClass() { std::cout << "MyClass destructor" << std::endl; }
void sayHello() { std::cout << "Hello, world!" << std::endl; }
};
int main() {
std::weak_ptr<MyClass> ptr1;
{
std::shared_ptr<MyClass> ptr2(new MyClass);
ptr1 = ptr2; // 使用weak_ptr保存shared_ptr
std::cout << "ptr2.use_count(): " << ptr2.use_count() << std::endl; // 输出1
}
if (!ptr1.expired()) { // 判断是否已经被释放
std::shared_ptr<MyClass> ptr3 = ptr1.lock(); // 获取shared_ptr
ptr3->sayHello();
std::cout << "ptr3.use_count(): " << ptr3.use_count() << std::endl; // 输出1
} else {
std::cout << "shared_ptr has been released." << std::endl;
}
return 0;
}
在上面的代码中,首先定义了一个名为MyClass
的类,在main()
函数中,使用weak_ptr
对象ptr1
保存了一个尚未创建的shared_ptr
。接着,定义了一个名为ptr2
的shared_ptr
,并将其作为参数传递给了ptr1
,此时该对象的引用计数为1。
在ptr2
生命周期结束后,判断ptr1
所指向的对象是否已经被释放,如果没有被释放,则使用lock()
函数获取ptr1
所指向的shared_ptr
,然后调用sayHello()
方法输出一条信息。
需要注意的是,由于这里是通过lock()
函数获取shared_ptr
,因此要确保在使用shared_ptr
对象之前,要对expired()
进行检查,以防止在lock()
函数执行期间,shared_ptr
对象已经被释放导致问题发生。
总之,weak_ptr
可以有效地解决循环引用和延迟初始化等问题,同时还可以提供更为灵活的内存管理方式。
5 强制类型转换
5.1 C风格的强制转换
顾名思义,就是C强制类型转换
int main()
{
int i = 10;
float j = (float)i;
}
5.2 const_cast
在 C++ 中,const_cast 是一种用于去除类型的 const 属性的强制类型转换。它可以用来将 const 变量转换为非 const 变量,或者将指向 const 对象的指针转换为指向非 const 对象的指针。
const_cast<type>(expression)
其中,type 表示欲转换的目标类型,expression 表示要进行转换的表达式。需要注意的是,const_cast 只能用于去除 const 属性,如果尝试使用它来添加 const 属性或者将一种类型转换成另一种类型,则会导致未定义的行为。
一个常见的用途是通过 const_cast 去除 const 属性,以便修改对象的值。例如:
const int i = 10;
int& r = const_cast<int&>(i); // 去除 i 的 const 属性
r = 20; // 修改 r 的值
需要注意的是,虽然 const_cast 可以让你去除 const 属性,但这并不意味着你可以随意修改原始对象的值。如果原始对象是 const 类型,则其值仍然不能被修改。
5.3 static_cast
在 C++ 中,static_cast 是一种用于进行静态类型转换的强制类型转换。它可以将一种类型转换为另一种类型,但是只能进行安全的转换,即编译器能够在编译时确定类型转换是有效的。
static_cast<type>(expression)
其中,type 表示欲转换的目标类型,expression 表示要进行转换的表达式。
下面是 static_cast 常见的几种用法:
将一种算术类型转换为另一种算术类型。 例如,将一个 int 类型的变量转换成一个 float 类型的变量:
int i = 10;
float f = static_cast<float>(i);
将一个指针类型转换为另一个指针类型。 例如,将一个基类指针转换为派生类指针:
class Base {};
class Derived : public Base {};
Base* b = new Derived;
Derived* d = static_cast<Derived*>(b);
需要注意的是,在进行指针类型转换时,static_cast 并不能保证其结果是有效的。因此,在进行指针类型转换时应该格外谨慎,并尽可能避免使用 static_cast 进行指针类型的转换。
将一个对象转换成一个与之相关的类型。 例如,将一个结构体转换成一个 union:
struct S { int i; };
union U { int i; float f; };
S s = {10};
U u = static_cast<U>(s);
需要注意的是,static_cast 并不能用于执行动态类型转换。如果要进行动态类型转换,则应该使用 dynamic_cast。
5.4 dynamic_cast
在 C++ 中,dynamic_cast 是一种用于进行动态类型转换的强制类型转换。它可以将一个指向基类的指针或引用转换为指向派生类的指针或引用,同时还提供了类型安全检查,避免了类型转换时出现错误。
dynamic_cast
其中,type 表示欲转换的目标类型,expression 表示要进行转换的表达式。需要注意的是,dynamic_cast 只能用于含有虚函数的类层次结构中,否则编译时会出错。
下面是 dynamic_cast 常见的几种用法:
将指向基类的指针或引用转换为指向派生类的指针或引用。 例如,将一个基类指针转换为派生类指针:
class Base { public: virtual ~Base() {} };
class Derived : public Base {};
Base* b = new Derived; Derived* d = dynamic_cast<Derived*>(b);
需要注意的是,如果将一个指向基类的指针或引用转换为指向派生类的指针或引用失败,则 dynamic_cast 返回 nullptr(对于指针)或抛出 std::bad_cast 异常(对于引用)。
在类之间进行安全的向下转型。 例如,将一个基类指针或引用转换为指向某个派生类的指针或引用,以便访问派生类的成员函数或成员变量。
class Base {
public:
virtual ~Base() {}
};
class Derived : public Base {
public:
void derivedFunc() {}
};
Base* b = new Derived;
Derived* d = dynamic_cast<Derived*>(b);
if (d != nullptr) {
d->derivedFunc();
}
需要注意的是,dynamic_cast 的使用应该尽可能地避免。它通常表明了不良的设计,因为它破坏了抽象基类的概念,并且会导致代码的可维护性下降。
5.5 reinterpret_cast
在 C++ 中,reinterpret_cast 是一种用于进行底层类型转换的强制类型转换。它可以将一个指针或引用类型的值转换为另一种不同类型的指针或引用类型的值,而且转换是不安全的,因为它会取消类型之间的任何类型检查。
reinterpret_cast<type>(expression)
其中,type 表示欲转换的目标类型,expression 表示要进行转换的表达式。需要注意的是,reinterpret_cast 可以在指针、引用、整数与指针之间进行转换,但是如果对不存在的类型使用 reinterpret_cast,则可能导致未定义的行为。
下面是 reinterpret_cast 常见的几种用法:
将一个指向某个类型的指针转换为指向另一种类型的指针。 例如,将一个 int 类型的指针转换成 char 类型的指针:
int i = 10;
char* p = reinterpret_cast<char*>(&i);
需要注意的是,由于 reinterpret_cast 取消了类型之间的任何类型检查,因此在进行指针类型转换时应该格外谨慎,并尽可能避免使用 reinterpret_cast 进行指针类型的转换。
将一个指针转换成一个整数类型。 例如,将一个指向 int 类型的指针转换为一个 unsigned long 类型的整数:
int i = 10;
unsigned long n = reinterpret_cast<unsigned long>(&i);
需要注意的是,由于指针的大小和整数的大小可能不同,因此在进行指针与整数之间的转换时应该格外谨慎。