友元类
适用情况
- 紧密协作:
- 当两个类需要紧密协作,且需要共享内部实现细节时
- 例如,操作类需要访问数据类的内部数据来实现复杂的功能
- 封装复杂操作:
- 当某些复杂操作不能或不应成为数据类的成员函数时
- 通过友元类可以将这些操作封装在独立的类中,从而简化数据类的设计
- 提高性能:
- 在某些情况下,通过友元类直接访问内部数据可以提高操作的性能,而不是通过公共接口进行间接访问
- 实现特定的设计模式:
- 在某些设计模式中,例如访问者模式,友元类可以用来访问对象的私有数据,以实现特定的行为
示例
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 |
class Matrix; class MatrixOperations { public: static Matrix add(const Matrix& m1, const Matrix& m2); static void print(const Matrix& m); }; class Matrix { private: std::vector<std::vector<int>> data; public: Matrix(int rows, int cols) : data(rows, std::vector<int>(cols)) {} // 将 MatrixOperations 声明为友元类 friend class MatrixOperations; }; |
友元函数
适用情况
- 非成员函数需要访问类的内部数据:
- 当一个非成员函数需要访问类的私有或保护成员时,可以将其声明为友元函数
- 例如,独立的计算函数需要访问对象的内部数据
- 辅助功能:
- 当一个辅助函数需要对类进行特定操作,但不应该成为类的成员函数时
- 例如,用于调试或日志记录的函数
- 提高性能:
- 当需要直接访问内部数据以提高操作性能,而不是通过公共接口进行间接访问时
- 特定算法实现:
- 在某些算法实现中,友元函数可以用来访问和修改对象的内部状态,以实现高效的操作
示例
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 |
class Box { private: double length; double width; double height; public: Box(double l, double w, double h) : length(l), width(w), height(h) {} // 将 calculateVolume 声明为友元函数 friend double calculateVolume(const Box& b); }; // 友元函数可以访问 Box 的私有数据成员 double calculateVolume(const Box& b) { return b.length * b.width * b.height; } |
联合体
概述
- 联合体(
union
)是C++
中的一种数据结构- 它允许你在同一存储空间内存储不同类型的数据,但只能同时存储其中的一种类型
- 联合体的成员共享同一个内存空间,这意味着在任何时间点,联合体只能存储一个成员的值
- 联合体在某些特定场景下非常有用,特别是当你希望节省内存或需要处理多种类型的数据时
作用
- 节省内存:
- 由于联合体的所有成员共享同一块内存,它在需要存储不同类型的数据但同时只需要一个的情况下,可以节省内存空间
- 例如,处理多种数据格式的网络数据包或协议消息时,可以使用联合体来节省内存
- 多用途数据结构:
- 联合体可以用来创建多用途的数据结构,这些结构能够根据具体情况存储不同类型的数据
- 例如,一个联合体可以存储一个整数或一个浮点数,具体取决于上下文需求
- 简化数据管理:
- 当你处理的变量可能是几种不同类型之一时,联合体可以简化数据管理
- 通过使用联合体,你可以避免使用多种变量来处理相同的数据逻辑
示例
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 |
#include <iostream> #include <cstring> union Data { int intValue; float floatValue; char charValue[20]; }; int main() { Data data; data.intValue = 10; std::cout << "Integer: " << data.intValue << std::endl; data.floatValue = 220.5; std::cout << "Float: " << data.floatValue << std::endl; strcpy(data.charValue, "Hello, World!"); std::cout << "String: " << data.charValue << std::endl; return 0; } |
注意事项
- 只能存储一个值:
- 联合体在任意时间点只能存储一个值,因此你需要小心处理不同类型的数据
- 存储新值会覆盖之前存储的值
- 数据有效性:
- 在使用联合体时,必须清楚当前存储的数据类型,以避免读取错误的值
- 这通常通过额外的标志变量或类型标签来管理
- 构造函数和析构函数:
- 联合体的成员不能有构造函数或析构函数,因为联合体的所有成员共享同一块内存
- 如果需要在联合体中使用复杂类型,可以考虑使用带有联合体的结构体,并在结构体中管理这些类型的构造和析构
带有匿名联合体的结构体
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 |
#include <iostream> #include <cstring> struct Data { enum Type { INT, FLOAT, CHAR } type; union { int intValue; float floatValue; char charValue[20]; }; Data(int value) : type(INT), intValue(value) {} Data(float value) : type(FLOAT), floatValue(value) {} Data(const char* value) : type(CHAR) { strcpy(charValue, value); } void print() const { switch (type) { case INT: std::cout << "Integer: " << intValue << std::endl; break; case FLOAT: std::cout << "Float: " << floatValue << std::endl; break; case CHAR: std::cout << "String: " << charValue << std::endl; break; } } }; int main() { Data d1(10); Data d2(220.5f); Data d3("Hello, World!"); d1.print(); d2.print(); d3.print(); return 0; } |
内联函数
概述
- 内联函数(
inline function
)是C++
中的一种函数 - 它允许编译器在调用函数时将函数的代码直接插入到调用点,从而避免函数调用的开销
- 使用内联函数可以提高程序的执行效率,特别是在频繁调用的小函数中
作用
- 消除函数调用开销:
- 在传统的函数调用中,程序需要执行一些额外的操作,如压栈、跳转和返回等
- 对于频繁调用的小函数,这些操作的开销可能会显著影响性能
- 内联函数通过将函数体直接插入调用点,消除了这些额外的操作
- 提高性能:
- 通过内联函数,程序可以在不牺牲可读性和模块化的情况下提高性能
- 特别是在时间敏感的应用中,如游戏编程、实时系统等,内联函数可以显著提高性能
- 编译器优化:
- 内联函数为编译器提供了更多的优化机会
- 编译器可以在插入内联函数代码后进行进一步的优化,如消除冗余代码、常量折叠等
内联与宏
- 内联与宏都可以实现代码替换,比较如下:
- 类型安全:
- 内联函数是类型安全的,而宏只是简单的文本替换,没有类型检查
- 调试信息:
- 内联函数可以生成调试信息,有助于调试,而宏展开后很难调试
- 作用域控制:
- 内联函数受作用域控制,而宏在整个翻译单元中有效,容易引起命名冲突
注意
- 适用于小函数:
- 内联函数适用于小且频繁调用的函数
- 对于大函数,内联会导致代码膨胀,从而可能降低程序的性能
- 编译器的决定权:
inline
关键字只是对编译器的建议,编译器可以选择忽略内联请求- 如果编译器认为内联不合适(例如函数体太大),它可以选择不内联该函数
- 递归函数:
- 递归函数不适合内联,因为每次调用都需要新的函数调用环境
- 尽管可以将递归函数声明为内联,但编译器通常不会内联递归函数
- 代码膨胀:
- 过度使用内联函数会导致代码膨胀,增加可执行文件的大小,并可能导致指令缓存未命中,从而降低程序性能
内联本质
- 内联函数的本质是在编译阶段,编译器将函数调用处用函数体的代码直接替换(展开),从而避免了函数调用的开销
- 这个过程发生在编译期间,具体步骤如下:
- 源代码解析:编译器首先解析源代码,识别出内联函数
- 内联展开:在生成汇编指令之前,编译器会将内联函数的代码插入到每一个调用点
这意味着,函数体的代码会直接嵌入到调用函数的代码中,而不是进行常规的函数调用(即跳转和返回)
示例
- 全局内联
1 2 3 4 5 6 7 8 9 10 11 |
#include <iostream> inline int add(int a, int b) { return a + b; } int main() { int result = add(5, 3); std::cout << "Result: " << result << std::endl; return 0; } |
1 2 3 4 5 6 |
; int result = add(5, 3); mov eax, 5 ; 将5加载到寄存器eax add eax, 3 ; 将3加到寄存器eax的值 ; return result; mov [result], eax; 将eax的值(即8)存储到result |
- 类内内联
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 |
#include <iostream> class Math { public: inline int multiply(int a, int b) { return a * b; } int divide(int a, int b) { return a / b; } }; int main() { Math math; std::cout << "Multiply: " << math.multiply(4, 5) << std::endl; std::cout << "Divide: " << math.divide(20, 5) << std::endl; return 0; } |
static
概述
- 可以用来修饰变量和函数,具有不同的含义和作用
静态局部变量
- 用法:
static
可以用来修饰局部变量,使其在函数调用之间保持其值不变
- 作用:
- 静态局部变量只在第一次调用函数时初始化,以后每次调用函数时都会保留上一次的值
- 生命周期:
- 静态局部变量的生命周期是整个程序运行期间,但它的作用域仅限于定义它的函数
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 |
#include <iostream> void counter() { static int count = 0; // 静态局部变量 count++; std::cout << "Count: " << count << std::endl; } int main() { counter(); counter(); counter(); return 0; } Count: 1 Count: 2 Count: 3 |
静态成员变量
- 用法:
static
可以用来修饰类的成员变量,使其成为静态成员变量
- 作用:
- 静态成员变量在所有对象之间共享,一个类的所有对象对该静态成员变量只有一个实例
- 生命周期:
- 静态成员变量在程序启动时初始化,并在程序结束时销毁
- 初始化:
- 静态成员变量需要在类外进行初始化
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 |
#include <iostream> class MyClass { public: static int count; // 静态成员变量 MyClass() { count++; } }; int MyClass::count = 0; // 类外初始化 int main() { MyClass obj1; MyClass obj2; std::cout << "Count: " << MyClass::count << std::endl; return 0; } Count: 2 |
静态成员函数
- 用法:
static
可以用来修饰类的成员函数,使其成为静态成员函数
- 作用:
- 静态成员函数可以在不创建对象的情况下调用,它们不能访问类的非静态成员变量和非静态成员函数,但可以访问静态成员变量和静态成员函数
- 调用:
- 静态成员函数可以通过类名直接调用
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 |
#include <iostream> class MyClass { public: static int count; static void displayCount() { // 静态成员函数 std::cout << "Count: " << count << std::endl; } }; int MyClass::count = 0; int main() { MyClass::displayCount(); MyClass obj1; MyClass::count++; MyClass::displayCount(); return 0; } Count: 0 Count: 1 |
静态全局变量
- 用法:
static
可以用来修饰全局变量,使其成为静态全局变量
- 作用:
- 静态全局变量的作用域仅限于定义它的文件,而不是整个程序。这有助于防止命名冲突
- 生命周期:
- 静态全局变量在程序启动时初始化,并在程序结束时销毁
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 |
#include <iostream> static int globalVar = 0; // 静态全局变量 void increment() { globalVar++; std::cout << "Global Var: " << globalVar << std::endl; } int main() { increment(); increment(); return 0; } Global Var: 1 Global Var: 2 |
静态全局函数(文件作用域)
- 用法:
static
可以用来修饰全局函数,使其成为静态函数
- 作用:
- 静态函数的作用域仅限于定义它的文件,这有助于防止命名冲突
- 生命周期:
- 静态函数在程序启动时加载,并在程序结束时卸载
问题
- 线程入口函数里面的静态局部变量,这个静态局部变量会不会在多个线程中保存副本?
- 在
C++
中,静态局部变量在函数中只会有一个实例,并且在所有调用中共享这个实例 - 即使这个函数被多个线程调用,所有线程也会共享同一个静态局部变量
- 静态局部变量的生命周期在程序运行期间内是唯一的,不会为每个线程创建单独的副本
- 在
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 |
#include <iostream> #include <thread> #include <mutex> std::mutex coutMutex; void threadFunction() { static int staticVar = 0; // 静态局部变量 ++staticVar; std::lock_guard<std::mutex> guard(coutMutex); std::cout << "Thread ID: " << std::this_thread::get_id() << ", Static Var: " << staticVar << std::endl; } int main() { std::thread t1(threadFunction); std::thread t2(threadFunction); std::thread t3(threadFunction); t1.join(); t2.join(); t3.join(); return 0; } |
- 怎么让每个线程都保存副本?
- 如果需要在每个线程中有自己独立的一份数据,可以使用线程本地存储(
Thread Local Storage
,TLS
) - 在
C++11
及之后的标准中,你可以使用thread_local
关键字来实现每个线程都有自己的独立变量实例
- 如果需要在每个线程中有自己独立的一份数据,可以使用线程本地存储(
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 |
#include <iostream> #include <thread> #include <mutex> std::mutex coutMutex; void threadFunction() { thread_local int localVar = 0; // 线程局部存储变量 ++localVar; std::lock_guard<std::mutex> guard(coutMutex); std::cout << "Thread ID: " << std::this_thread::get_id() << ", Local Var: " << localVar << std::endl; } int main() { std::thread t1(threadFunction); std::thread t2(threadFunction); std::thread t3(threadFunction); t1.join(); t2.join(); t3.join(); return 0; } |
thread_local
和static
可以同时使用吗- 在
C++
中,你不能同时使用thread_local
和static
关键字来修饰同一个变量 - 这是因为
thread_local
已经隐含了静态存储期,即变量在程序启动时分配,并在程序结束时销毁,而每个线程有其独立的实例 - 因此,
thread_local
本身已经包含了static
的语义,不需要再加static
关键字
- 在
指针
概述
- 指针的定义:
- 指针是一个变量,它的值是另一个变量的地址
- 指针的声明:
- 使用
*
号来声明指针类型。例如,int *p
表示一个指向整数类型的指针
- 使用
- 取地址操作符
&
:- 用于获取变量的地址。例如,
int a = 5; int *p = &a;
- 用于获取变量的地址。例如,
- 解引用操作符
*
:- 用于访问指针指向的地址处的值。例如,
*p
可以访问p
所指向的a
的值
- 用于访问指针指向的地址处的值。例如,
指针类型
- 指针类型与它所指向的数据类型相关,指针类型决定了解引用时的行为
- 即指针类型决定了解引用时读取的内存字节数以及如何解释这些字节
- 例如,
int*
指针解引用时会读取4个字节(在大多数平台上),并将这些字节解释为一个整数 - 而
double*
指针解引用时会读取8个字节,并将这些字节解释为一个双精度浮点数
解引用时的行为
- 不同类型的指针解引用时的表现不同,本质上是由于编译器根据指针的类型,从目标地址读取和解释内存的方式不同
- 编译器根据指针的类型确定需要读取多少内存字节,并且如何解释这些字节的内容
示例
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 |
#include <iostream> int main() { // 分配一块内存 char memory[8] = {0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08}; // 指向相同的内存地址 char *charPtr = memory; int *intPtr = reinterpret_cast<int*>(memory); double *doublePtr = reinterpret_cast<double*>(memory); // 解引用不同类型的指针 std::cout << "charPtr points to: " << static_cast<int>(*charPtr) << std::endl; // 1 byte std::cout << "intPtr points to: " << *intPtr << std::endl; // 4 bytes std::cout << "doublePtr points to: " << *doublePtr << std::endl; // 8 bytes return 0; } |
charPtr
作为char*
类型指针,解引用时读取1
个字节并解释为一个字符intPtr
作为int*
类型指针,解引用时读取4
个字节并解释为一个整数doublePtr
作为double*
类型指针,解引用时读取8
个字节并解释为一个双精度浮点数- 内存布局示意图:
charPtr
解引用时读取0x01
intPtr
解引用时读取0x01 0x02 0x03 0x04
并将其解释为一个整数(例如,小端序表示下的值为0x04030201
)doublePtr
解引用时读取所有8
个字节,并将其解释为一个双精度浮点数
1 2 |
地址: 0x00 0x01 0x02 0x03 0x04 0x05 0x06 0x07 内容: 0x01 0x02 0x03 0x04 0x05 0x06 0x07 0x08 |
- 汇编级表现:
- 从汇编级别看,不同类型指针解引用时,编译器生成的指令会不同,以读取正确数量的字节并正确解释这些字节
charPtr
解引用(读取1字节):mov al, [charPtr]
intPtr
解引用(读取4字节):mov eax, [intPtr]
doublePtr
解引用(读取8字节):movq xmm0, [doublePtr]
- 在
x86
架构的汇编语言中,al
寄存器是ax
寄存器的低8
位部分,而ax
寄存器又是eax
寄存器的低16
位部分
在64
位架构中,eax
寄存器是rax
寄存器的低32
位部分
备注
movq
- 这是一条汇编指令,用于将数据从内存移动到寄存器,或从寄存器移动到内存
movq
中的q
表示操作的是四字(quadword
),即64
位(8
字节)
xmm0
- 这是一个
128
位的寄存器,用于存储浮点数或SIMD
(单指令多数据)操作中的数据 - 在这个上下文中,我们只使用了
xmm0
的低64
位
- 这是一个
[doublePtr]
- 这是一个内存地址操作数,表示从
doublePtr
指针所指向的内存地址读取数据
- 这是一个内存地址操作数,表示从
浅拷贝和深拷贝
概述
- 在默认情况下,
C++
会提供一个浅拷贝的拷贝构造函数,但在需要深拷贝的情况下,需要用户自定义拷贝构造函数
对比
- 浅拷贝(Shallow Copy):
- 复制对象的所有成员变量,但不复制成员指针所指向的资源
- 结果是多个对象共享相同的资源
- 深拷贝(Deep Copy):
- 不仅复制对象的所有成员变量,还复制成员指针所指向的资源
- 每个对象都有自己的独立资源副本
默认拷贝构造函数
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 |
class Shallow { public: int* data; Shallow(int val) { data = new int(val); } // 默认的拷贝构造函数 Shallow(const Shallow& other) : data(other.data) { std::cout << "Shallow copy constructor called" << std::endl; } ~Shallow() { delete data; } }; int main() { Shallow obj1(10); Shallow obj2 = obj1; // 调用默认的拷贝构造函数 return 0; } |
自定义拷贝构造函数(深拷贝)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 |
class Deep { public: int* data; Deep(int val) { data = new int(val); } // 自定义的拷贝构造函数(深拷贝) Deep(const Deep& other) { data = new int(*other.data); // 分配新的内存并复制值 std::cout << "Deep copy constructor called" << std::endl; } ~Deep() { delete data; } }; int main() { Deep obj1(10); Deep obj2 = obj1; // 调用自定义的拷贝构造函数 std::cout << "obj1 data: " << *obj1.data << std::endl; std::cout << "obj2 data: " << *obj2.data << std::endl; return 0; } |
深拷贝的其他细节
- 拷贝赋值运算符:
- 当需要对现有对象赋值时,应该同时实现拷贝赋值运算符,以确保深拷贝行为
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 |
class Deep { public: int* data; Deep(int val) { data = new int(val); } // 自定义的拷贝构造函数(深拷贝) Deep(const Deep& other) { data = new int(*other.data); std::cout << "Deep copy constructor called" << std::endl; } // 自定义的拷贝赋值运算符 Deep& operator=(const Deep& other) { if (this == &other) { return *this; // 防止自赋值 } delete data; // 释放旧资源 data = new int(*other.data); // 分配新的内存并复制值 std::cout << "Deep copy assignment operator called" << std::endl; return *this; } ~Deep() { delete data; } }; int main() { Deep obj1(10); Deep obj2 = obj1; // 调用自定义的拷贝构造函数 Deep obj3(20); obj3 = obj1; // 调用自定义的拷贝赋值运算符 std::cout << "obj1 data: " << *obj1.data << std::endl; std::cout << "obj2 data: " << *obj2.data << std::endl; std::cout << "obj3 data: " << *obj3.data << std::endl; return 0; } |
- 移动语义:
- 在
C++11
及以后,可以实现移动构造函数和移动赋值运算符,以优化资源管理,避免不必要的拷贝操作
- 在
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 |
class Deep { public: int* data; Deep(int val) { data = new int(val); } // 自定义的拷贝构造函数(深拷贝) Deep(const Deep& other) { data = new int(*other.data); std::cout << "Deep copy constructor called" << std::endl; } // 自定义的移动构造函数 Deep(Deep&& other) noexcept : data(other.data) { other.data = nullptr; std::cout << "Move constructor called" << std::endl; } // 自定义的拷贝赋值运算符 Deep& operator=(const Deep& other) { if (this == &other) { return *this; // 防止自赋值 } delete data; // 释放旧资源 data = new int(*other.data); // 分配新的内存并复制值 std::cout << "Deep copy assignment operator called" << std::endl; return *this; } // 自定义的移动赋值运算符 Deep& operator=(Deep&& other) noexcept { if (this == &other) { return *this; // 防止自赋值 } delete data; // 释放旧资源 data = other.data; other.data = nullptr; std::cout << "Move assignment operator called" << std::endl; return *this; } ~Deep() { delete data; } }; int main() { Deep obj1(10); Deep obj2 = obj1; // 调用自定义的拷贝构造函数 Deep obj3(std::move(obj1)); // 调用自定义的移动构造函数 Deep obj4(20); obj4 = std::move(obj2); // 调用自定义的移动赋值运算符 return 0; } |
编译器怎么选择调用哪种构造函数
- 编译器会根据具体的上下文和传递的对象来决定使用哪种构造函数或赋值运算符
- 具体来说,当你用一个对象去创建一个新的对象时,调用的是拷贝构造函数还是移动构造函数,取决于传递的对象的类型(是左值还是右值)
构造函数选择规则
- 拷贝构造函数:
- 当传递的是一个左值对象(即具名对象或通过引用传递的对象)时,会调用拷贝构造函数
- 移动构造函数:
- 当传递的是一个右值对象(即临时对象或通过
std::move
显式转换为右值引用的对象)时,会调用移动构造函数
- 当传递的是一个右值对象(即临时对象或通过
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 |
#include <iostream> #include <utility> // for std::move class MyClass { public: int* data; // 默认构造函数 MyClass(int val) : data(new int(val)) { std::cout << "Constructor called" << std::endl; } // 拷贝构造函数 MyClass(const MyClass& other) : data(new int(*other.data)) { std::cout << "Copy constructor called" << std::endl; } // 移动构造函数 MyClass(MyClass&& other) noexcept : data(other.data) { other.data = nullptr; std::cout << "Move constructor called" << std::endl; } // 拷贝赋值运算符 MyClass& operator=(const MyClass& other) { if (this != &other) { delete data; data = new int(*other.data); std::cout << "Copy assignment operator called" << std::endl; } return *this; } // 移动赋值运算符 MyClass& operator=(MyClass&& other) noexcept { if (this != &other) { delete data; data = other.data; other.data = nullptr; std::cout << "Move assignment operator called" << std::endl; } return *this; } // 析构函数 ~MyClass() { delete data; std::cout << "Destructor called" << std::endl; } }; int main() { MyClass obj1(10); // 调用默认构造函数 MyClass obj2 = obj1; // 调用拷贝构造函数 MyClass obj3 = std::move(obj1); // 调用移动构造函数 MyClass obj4(20); // 调用默认构造函数 obj4 = obj2; // 调用拷贝赋值运算符 obj4 = std::move(obj3); // 调用移动赋值运算符 return 0; } |
本文为原创文章,版权归Aet所有,欢迎分享本文,转载请保留出处!
你可能也喜欢
- ♥ C++并发编程_同步并发(Condition_variable)05/21
- ♥ C++17_第一篇12/20
- ♥ C++编程规范101规则、准则与最佳实践 二01/07
- ♥ C++11_四种类型转换11/10
- ♥ Soui应用 动画二06/27
- ♥ C++14_第二篇06/21