C++基础面试题 | 什么是C++的移动语义和完美转发?

2024-08-27 01:28

本文主要是介绍C++基础面试题 | 什么是C++的移动语义和完美转发?,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

若待上林花似锦,出门俱是看花人。 - 《城东早春》(杨巨源)

2024.8.26

回答重点

移动语义(Move Semantics)和完美转发(Perfect Forwarding)都是C++11 引入的新特性。

  1. 移动语义实现了从传统的拷贝资源到转移将亡对象资源,减少了拷贝,提高了效率。
  2. 而完美转发实现了一个可以接收任意类型参数的函数,并通过forward将原始的参数类型转发给最匹配的下层函数,最大限度地提高效率。

移动语义(Move Semantics)

移动语义是C++11中引入的一种机制,它允许对象的资源(如内存、文件句柄等)从一个对象转移到另一个对象,而不需要复制资源**(从拷贝构造函数的拷贝资源到移动构造的转移资源)**。它可以避免不必要的复制操作,从而提高程序的效率。

移动语义的核心是两个特殊的成员函数:

  1. 移动构造函数:接受一个右值引用作为参数,用于接收一个即将被销毁的对象,从而获取其资源。
  2. 移动赋值运算符:同样接受一个右值引用,用于将一个即将被销毁的对象的资源转移给当前对象。

例如,如果你有一个包含动态分配内存的类,你可以这样实现移动构造函数和移动赋值运算符:

class A {
public:A(int size) : size_(size) {data_ = new int[size];}A(){}A(const A& a) {size_ = a.size_;data_ = new int[size_];cout << "copy " << endl;}A(A&& a) {this->data_ = a.data_;a.data_ = nullptr;cout << "move " << endl;}~A() {if (data_ != nullptr) {delete[] data_;}}int *data_;int size_;
};
int main() {A a(10);A b = a;A c = std::move(a); // 调用移动构造函数return 0;
}

如果不使用move实现移动语义,使用将亡值构造新对象时内部的资源没有得到充分利用,拷贝代价极大。

std::vector<string> vecs;
...
std::vector<string> vecm = std::move(vecs); // 免去很多拷贝

完美转发(Perfect Forwarding)

完美转发是模板编程中的一个概念,它允许我们将函数的参数以它们原始的值类别(lvalue或rvalue)传递给另一个函数。这意味着我们可以在模板函数中接收一个参数,然后以它被传递给我们的方式传递给另一个函数,无论它是通过值、引用还是移动语义。

完美转发通常与std::forward一起使用。std::forward是一个函数模板,它接受一个通用引用(也称为转发引用),并将其转发给另一个函数。

例如,一个使用完美转发的模板函数可能看起来像这样:

#include<iostream>
using namespace std;
void PrintV(int &t) { // 接收被转发的左值对象cout << "lvalue" << endl;
}void PrintV(int &&t) { // 接收被转发的右值对象cout << "rvalue" << endl;
}// 定义一个模板函数 能够接收所有类型对象(左值和右值都会被传递至此)
template<typename T>
void Test(T &&t) {PrintV(std::forward<T>(t));// 变量t退化为左值// forward进行完美转发:如果原本的参数是右值,转化为右值传递到下一层// 如果原本的参数是左值,不做操作,正常传递到下一层。 下层函数printV根据参数决定调用的函数
}int main() {Test(1); // rvalueint a = 1;Test(a); // lvalueTest(std::forward<int>(a)); // rvalueTest(std::forward<int &>(a)); // lvalueTest(std::forward<int &&>(a)); // rvaluereturn 0;
}

这两个特性一起使用,可以极大地提高C++程序的效率和灵活性。

这篇关于C++基础面试题 | 什么是C++的移动语义和完美转发?的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1110238

相关文章

【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提供个模板形参的名

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

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)

【C++高阶】C++类型转换全攻略:深入理解并高效应用

📝个人主页🌹:Eternity._ ⏩收录专栏⏪:C++ “ 登神长阶 ” 🤡往期回顾🤡:C++ 智能指针 🌹🌹期待您的关注 🌹🌹 ❀C++的类型转换 📒1. C语言中的类型转换📚2. C++强制类型转换⛰️static_cast🌞reinterpret_cast⭐const_cast🍁dynamic_cast 📜3. C++强制类型转换的原因📝

【Linux 从基础到进阶】Ansible自动化运维工具使用

Ansible自动化运维工具使用 Ansible 是一款开源的自动化运维工具,采用无代理架构(agentless),基于 SSH 连接进行管理,具有简单易用、灵活强大、可扩展性高等特点。它广泛用于服务器管理、应用部署、配置管理等任务。本文将介绍 Ansible 的安装、基本使用方法及一些实际运维场景中的应用,旨在帮助运维人员快速上手并熟练运用 Ansible。 1. Ansible的核心概念

C++——stack、queue的实现及deque的介绍

目录 1.stack与queue的实现 1.1stack的实现  1.2 queue的实现 2.重温vector、list、stack、queue的介绍 2.1 STL标准库中stack和queue的底层结构  3.deque的简单介绍 3.1为什么选择deque作为stack和queue的底层默认容器  3.2 STL中对stack与queue的模拟实现 ①stack模拟实现