完美轉發使用

完美轉發的幾個例子

例子 1:普通的完美轉發

首先,我們先來一個簡單的完美轉發的例子,展示如何使用 std::forward 來保持傳入參數的類型。

#include <iostream>
#include <utility>  // std::forwardvoid func(int& x) {std::cout << "Lvalue reference: " << x << std::endl;
}void func(int&& x) {std::cout << "Rvalue reference: " << x << std::endl;
}template <typename T>
void wrapper(T&& arg) {// 使用完美轉發func(std::forward<T>(arg));  // 根據傳入的類型精確地調用對應的函數
}int main() {int x = 10;wrapper(x);   // 左值傳遞wrapper(20);  // 右值傳遞
}

輸出:

Lvalue reference: 10
Rvalue reference: 20

在這個例子中:

  • wrapper(x) 傳入了左值,std::forward<T>(arg)arg 保持為左值,并轉發到 func(int& x)
  • wrapper(20) 傳入了右值,std::forward<T>(arg)arg 保持為右值,并轉發到 func(int&& x)

關鍵點:

  • std::forward<T>(arg) 確保了 arg 的類型(左值或右值)在轉發時不丟失。
例子 2:完美轉發與引用的結合

有時你可能希望通過完美轉發將引用類型的參數轉發給另一個函數。我們通過以下例子來演示這一點:

#include <iostream>
#include <utility>void process(int& x) {std::cout << "Processing left value: " << x << std::endl;
}void process(int&& x) {std::cout << "Processing right value: " << x << std::endl;
}template <typename T>
void handler(T&& arg) {process(std::forward<T>(arg));  // 完美轉發
}int main() {int a = 10;handler(a);        // 左值handler(20);       // 右值
}

輸出:

Processing left value: 10
Processing right value: 20
  • handler(a) 將左值 a 轉發給 process(int& x)
  • handler(20) 將右值 20 轉發給 process(int&& x)
例子 3:多個參數的完美轉發

完美轉發不僅適用于一個參數,還可以應用于多個參數。這是通過遞歸和 std::forward 的組合來實現的。

#include <iostream>
#include <utility>void process(int& a, double& b) {std::cout << "Processing left values: " << a << ", " << b << std::endl;
}void process(int&& a, double&& b) {std::cout << "Processing right values: " << a << ", " << b << std::endl;
}template <typename T, typename U>
void wrapper(T&& arg1, U&& arg2) {process(std::forward<T>(arg1), std::forward<U>(arg2));  // 完美轉發
}int main() {int x = 5;double y = 3.14;wrapper(x, y);      // 左值wrapper(10, 2.718);  // 右值
}

輸出:

Processing left values: 5, 3.14
Processing right values: 10, 2.718
  • wrapper(x, y) 轉發左值。
  • wrapper(10, 2.718) 轉發右值。
例子 4:通過完美轉發轉發容器

在處理容器時,完美轉發也很有用。以下是一個將容器對象通過完美轉發傳遞給函數的例子:

#include <iostream>
#include <vector>
#include <utility>void process(std::vector<int>& vec) {std::cout << "Lvalue reference to vector: ";for (auto v : vec) std::cout << v << " ";std::cout << std::endl;
}void process(std::vector<int>&& vec) {std::cout << "Rvalue reference to vector: ";for (auto v : vec) std::cout << v << " ";std::cout << std::endl;
}template <typename T>
void wrapper(T&& arg) {process(std::forward<T>(arg));  // 完美轉發
}int main() {std::vector<int> vec = {1, 2, 3};wrapper(vec);         // 左值wrapper(std::vector<int>{4, 5, 6});  // 右值
}

輸出:

Lvalue reference to vector: 1 2 3 
Rvalue reference to vector: 4 5 6
  • wrapper(vec) 轉發左值 vec
  • wrapper(std::vector<int>{4, 5, 6}) 轉發右值。

為什么命名為 std::forward

std::forward 的命名源于它的用途——“轉發”一個參數。這個名稱可以追溯到它的功能和它的語義:

  1. “Forward” 表示轉發std::forward 的主要目的是精確轉發一個參數,保持參數原本的值類別(左值或右值)。它“向前”轉發參數,就像把參數從一個函數“傳遞”到另一個函數。

  2. std::move 區別std::move 讓對象變成右值,而 std::forward 保證保持參數的原始類型(左值或右值)。std::move 的命名非常直觀,因為它的作用是“移動”資源。而 std::forward 的命名則代表“保持原樣,準確轉發”。

  3. 保證類型屬性不變std::forward 使用類型推導機制(模板參數 T)來決定傳遞給目標函數的參數是左值還是右值。這使得它能“轉發”參數并保持原始類型的屬性,不會做多余的修改。

總結
  • 完美轉發使用 std::forward 來確保參數傳遞時類型(左值或右值)保持不變。
  • std::forward 通過模板參數類型 T,結合條件判斷(左值或右值),確保正確地轉發參數。
  • “Forward” 這個命名意味著它是一個精確的“轉發”工具,它轉發的是一個函數的參數,并且保留了參數的原始類型屬性。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/diannao/70418.shtml
繁體地址,請注明出處:http://hk.pswp.cn/diannao/70418.shtml
英文地址,請注明出處:http://en.pswp.cn/diannao/70418.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

【Content-Type詳解、Postman中binary格式、json格式數據轉原始二進制流等】

Content-Type詳解、Postman中binary格式、json格式數據轉原始二進制流等 背景&#xff1a;postman中如何使用binary格式上傳文件 Content-TypeContent-Type的格式由三部分組成&#xff1a;以下是一些常見的Content-Type示例&#xff1a; Postman中 binary格式定義&#xff1a;用…

DeepSeek等大模型功能集成到WPS中的詳細步驟

記錄下將**DeepSeek功能集成到WPS中**的步驟&#xff0c;以備忘。 1. 下載并安裝OfficeAI插件 訪問OfficeAI插件下載地址&#xff1a;https://www.office-ai.cn/&#xff0c;下載插件&#xff08;目前只支持windows系統&#xff09;。 注意&#xff0c;有兩個插件&#xff0…

MATLAB學習之旅:從入門到基礎實踐

在當今科技飛速發展的時代,MATLAB作為一款強大的數學軟件,猶如一把神奇的鑰匙,能夠打開眾多領域的大門。無論是工程計算、數據分析,還是算法開發、可視化呈現,MATLAB都展現出了無與倫比的魅力。今天,就讓我們踏上這段奇妙的MATLAB學習之旅,從最基礎的部分開始,逐步探索…

在Ubutu18.04下搭建nfs服務器

在Ubutu18.04下搭建nfs服務器 主要參考這篇博客 Ubuntu18.04下安裝NFS詳細步驟_烏班圖安裝nfs-CSDN博客 1.安裝NFS服務&#xff1a; 服務器端&#xff1a; sudo apt install nfs-kernel-server

棧,優先級隊列,map,set

文章目錄 棧題目解析代碼 優先級隊列題解代碼 map題解代碼 set題解代碼 棧 題目解析 1.先把元素push進棧中&#xff0c;如果棧非空并且棧中的元素按順序和k相等就出棧&#xff0c;直到棧為空或者k ! sk.top() 代碼 #include<iostream> #include<stack> #include&l…

C++ Primer 類的靜態成員

歡迎閱讀我的 【CPrimer】專欄 專欄簡介&#xff1a;本專欄主要面向C初學者&#xff0c;解釋C的一些基本概念和基礎語言特性&#xff0c;涉及C標準庫的用法&#xff0c;面向對象特性&#xff0c;泛型特性高級用法。通過使用標準庫中定義的抽象設施&#xff0c;使你更加適應高級…

Java——super

在Java中&#xff0c;super關鍵字用于引用父類的成員&#xff08;屬性、方法或構造器&#xff09;。它在繼承關系中非常重要&#xff0c;主要用于以下幾種場景&#xff1a; 1. 調用父類的構造器 在子類的構造器中&#xff0c;可以使用super關鍵字調用父類的構造器。super()必須…

Unity 全局屏幕點擊特效

思路&#xff1a; 1、生成一個點擊特效實例&#xff0c;每點擊屏幕&#xff0c;就調整特效實例的位置并控制特效的顯隱狀態即可。 2、需要注意要保證在編輯器開發時或手機上運行時都要顯示點擊效果。 方案一 &#xff08;推薦&#xff09; using UnityEngine; using UnityEn…

什么是業務流程分類框架

業務流程分類框架是一個用于組織和系統化地分類業務流程的結構化方法。它旨在幫助企業理解、管理、分析和改進其運營流程。 可以把它想象成一個圖書館的圖書分類系統&#xff0c;幫助快速找到和理解不同類型的書籍。對于業務流程來說&#xff0c;分類框架幫助快速了解不同類型的…

基于springboot校園健康系統的設計與實現(源碼+文檔)

大家好我是風歌&#xff0c;今天要和大家聊的是一款基于springboot的園健康系統的設計與實現。項目源碼以及部署相關請聯系風歌&#xff0c;文末附上聯系信息 。 項目簡介&#xff1a; 基于springboot校園健康系統的設計與實現的主要使用者管理員具有最高的權限&#xff0c;通…

【Leetcode】平衡二叉樹

平衡二叉樹 題目 思路與代碼實現 常規解法&#xff1a; int max(int a,int b){return a>b?a:b;}int maxDepth(struct TreeNode* root) {if(rootNULL)return 0;return 1max(maxDepth(root->left),maxDepth(root->right)); }bool isBalanced(struct TreeNode* root)…

【AI實踐】阿里百煉文本對話Agent安卓版搭建

環境&#xff1a;安卓手機運行環境&#xff1b;WinsurfAI編程工具&#xff1b;阿里百煉提前創建Agent應用&#xff1b; 耗時&#xff1a;2小時&#xff1b; 1&#xff0c;新建安卓項目 完成文本輸入&#xff0c;并將輸入的文字顯示出來。 2&#xff0c;安裝SDK 參考文檔 安…

一文讀懂Docker之Docker Compose

目錄 一、Docker Compose簡介 二、Docker Compose的安裝和基本使用 1、Docker Compose的安裝 步驟一、下載docker-compose 步驟二、新增可執行權限 步驟三、查看是否安裝成功 2、Docker Compose的基本使用 (1)、docker-compose up (2)、docker-compose ps (3)、docke…

WordPress“更新失敗,響應不是有效的JSON響應”問題的修復

在使用WordPress搭建網站時&#xff0c;許多人在編輯或更新文章時&#xff0c;可能會遇到一個提示框&#xff0c;顯示“更新失敗&#xff0c;響應不是有效的JSON響應”。這個提示信息對于不了解技術細節的用戶來說&#xff0c;太難懂。其實&#xff0c;這個問題并不復雜&#x…

信息學奧賽一本通 1973 【16NOIP普及組】買鉛筆 | 洛谷 P1909 [NOIP 2016 普及組] 買鉛筆

【題目鏈接】 ybt 1973 【16NOIP普及組】買鉛筆 洛谷 P1909 [NOIP 2016 普及組] 買鉛筆 【題目考點】 1. 簡單數學 2. 數組 3. 向上取整 <cmath>中有函數double ceil(double x)&#xff0c;求x向上取整的值。 如果求正整數 ? a b ? \lceil \frac{a}{b} \rceil ?…

C++中的.*運算符

看運算符重載的時候&#xff0c;看到這一句 .* :: sizeof ?: . 注意以上5個運算符不能重載。 :: sizeof ?: . 這四個好理解&#xff0c;畢竟都學過&#xff0c;但.*是什么&#xff1f; 于是自己整理了一下 .* 是一種 C 中的運算符&#xff0c;稱為指針到成…

【JavaEE進階】MyBatis通過注解實現增刪改查

目錄 &#x1f343;前言 &#x1f340;打印日志 &#x1f334;傳遞參數 &#x1f38b;增(Insert) &#x1f6a9;返回主鍵 &#x1f384;刪(Delete) &#x1f332;改(Update) &#x1f333;查(Select) &#x1f6a9;起別名 &#x1f6a9;結果映射 &#x1f6a9;開啟駝…

【分布式理論14】分布式數據庫存儲:分表分庫、主從復制與數據擴容策略

文章目錄 一、分表分庫1. 數據分表的必要性與方式2. 數據分庫原則與優勢 二、主從復制1. 讀寫分離架構設計2. 數據復制方式3. MySQL實現主從復制4. MySQL主從復制實踐與高可用方案 三、數據擴容 隨著業務的不斷發展和數據量的增長&#xff0c;傳統的單機關系型數據庫已經逐漸不…

vxe-grid 通過配置式給單元格字段格式化樹結構數據,轉換樹結構節點

vxe-grid 通過配置式給單元格字段格式化樹結構數據&#xff0c;轉換樹結構節點 比如用戶自定義配置好的數據源&#xff0c;通過在列中配置好數據&#xff0c;全 json 方式直接返回給前端渲染&#xff0c;不需要寫任何格式化方法。 官網&#xff1a;https://vxetable.cn npm i…

延遲任務的11種實現方式(下)!!

接上文&#xff1a; Redisson的RDelayedQueue Redisson他是Redis的兒子&#xff08;Redis son&#xff09;&#xff0c;基于Redis實現了非常多的功能&#xff0c;其中最常使用的就是Redis分布式鎖的實現&#xff0c;但是除了實現Redis分布式鎖之外&#xff0c;它還實現了延遲…