自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

別再寫遞歸模板了!C++17 折疊表達(dá)式讓你告別模板地獄!

開發(fā)
現(xiàn)代 C++ 的核心思想就是讓復(fù)雜的事情變得簡單,讓危險的操作變得安全。折疊表達(dá)式就是最好的例子!

"啊啊啊!" 小王一頭栽在鍵盤上,發(fā)出哀嚎,"這個可變參數(shù)模板要寫吐了!" ??

老張正在享受他的下午茶時光,聽到動靜抬頭一看,不禁莞爾。"又在折騰什么呢,小伙子?"

模板地獄初體驗

"老張你看," 小王指著屏幕上密密麻麻的代碼,"就是想計算幾個數(shù)的和,寫得我頭暈眼花..."

// 基礎(chǔ)情況 - 只有一個參數(shù)時的處理 ??
template<typename T>
T sum(T v) {
    return v;  // 遞歸的終止條件
}

這是遞歸的基礎(chǔ)情況,就像爬樓梯要有第一級臺階一樣。。

接下來是遞歸的主體部分:

// 遞歸情況 - 處理多個參數(shù) ??
template<typename T, typename... Args>
T sum(T first, Args... args) {
    return first + sum(args...);  // 一層層往下遞歸 ????
}

這種寫法就像套娃一樣,一個函數(shù)調(diào)用套著另一個函數(shù)調(diào)用...

"哎呀," 老張喝了口咖啡,眼睛里閃著狡黠的光,"現(xiàn)在都2023年了,還在用這么老土的寫法啊?"

"啊?" 小王一臉茫然

"來來來,看看新時代的寫法!" 老張拉過鍵盤,手指飛快地敲擊著:

template<typename... Args>
auto sum(Args... args) {
    return (... + args);  // 一行解決戰(zhàn)斗! ??
}

"這...這也行?" 小王目瞪口呆,"這簡直就是魔法啊!" 

為什么折疊表達(dá)式更好? 

老張放下咖啡杯,開始細(xì)致地解釋: "讓我告訴你為什么新版本更優(yōu)秀:"

  • 代碼簡潔度 ?? "看看原來的版本,需要兩個模板函數(shù),而且還要寫遞歸。新版本只需要一個函數(shù),一行代碼就搞定!"
  • 編譯效率 ? "遞歸版本每處理一個參數(shù)都要生成一次函數(shù)調(diào)用,而折疊表達(dá)式在編譯期就能展開成一個扁平的表達(dá)式。比如:"
sum(1, 2, 3, 4)  
// 遞歸版本展開:
1 + sum(2, 3, 4)
1 + (2 + sum(3, 4))
1 + (2 + (3 + sum(4)))
1 + (2 + (3 + 4))

// 折疊表達(dá)式直接展開:
((1 + 2) + 3) + 4

運行時性能 ?? "遞歸版本每個遞歸調(diào)用都會產(chǎn)生函數(shù)調(diào)用開銷,而折疊表達(dá)式會被編譯器優(yōu)化成一組簡單的加法運算。"

小王若有所思地點點頭,"原來如此!不僅代碼更優(yōu)雅,性能也更好!"

"不止這些呢!" 老張興致勃勃地打開畫圖軟件,"折疊表達(dá)式就像疊千紙鶴,有四種基本手法..."

折疊表達(dá)式四種武功

"等等,老張!" 小王撓撓頭,"你說折疊表達(dá)式有四種手法,能具體講講嗎?" ??

"當(dāng)然!" 老張露出高深莫測的笑容,"我來給你演示一下:"

(1) 一元右折疊 (向右展開)

template<typename... Args>
void print_right(Args... args) {
    // 從右向左展開: a1 + (a2 + (a3 + a4)) ??
    (std::cout << ... << args)  // 從右向左展開
}

"就像疊紙飛機一樣," 老張解釋道, "從右邊開始一層層折疊!" ??

(2) 一元左折疊 (向左展開)

template<typename... Args>
void print_left(Args... args) {
    // 從左向右展開: ((a1 + a2) + a3) + a4 ??
    (該例子不恰當(dāng),以后會改??<< args << std::cout)  // 從左向右展開
}

"這次是從左邊開始折," 小王恍然大悟, "就像疊信封一樣!" ??

(3) 二元右折疊 (帶初始值)

template<typename... Args>
auto sum_right(Args... args) {
    return (args + ... + 100);  // 右邊帶初始值: a1 + (a2 + (a3 + 100)) ??
}

"哦!" 小王眼睛一亮, "這就像做蛋糕,最后要放個櫻桃在頂上!" ??

(4) 二元左折疊 (帶初始值)

template<typename... Args>
auto sum_left(Args... args) {
    return (100 + ... + args);  // 左邊帶初始值: ((100 + a1) + a2) + a3 ??
}

"對啦!" 老張點點頭, "就像搭積木,要先放個底座!" ???

"哦!明白了!" 小王眼睛一亮,"就像疊紙一樣,可以從左邊開始疊,也可以從右邊開始疊!" ??

"沒錯!" 老張點點頭,"而且?guī)С跏贾档陌姹靖踩?,就像疊紙前先打好底一樣!" ???

實戰(zhàn)修煉

"誒,小王," 老張眨眨眼睛 ??,"來個實戰(zhàn)練習(xí)怎么樣?"

"什么練習(xí)?" 小王立刻來了精神 ??

"寫個函數(shù),能一次性打印多個參數(shù),要用折疊表達(dá)式哦!" 老張露出狡黠的笑容 ??

小王思考片刻,眼睛一亮 ??:

template<typename... Args>
void print(Args... args) {
    (std::cout << ... << args) << "\n";  // 折疊魔法 ?
}

"哇!這也太簡單了吧!" 小王驚喜地喊道 ??

"對??!" 老張點點頭,"用起來更簡單:"

print("Hello", 42, 3.14, "World");  // 一行搞定 ??
// 輸出: Hello423.14World

"這比寫一堆重載函數(shù)爽多了!" 小王擊掌歡呼 ??

"沒錯," 老張笑著說,"這就是現(xiàn)代C++的魅力!" ?

注意事項小貼士

"誒,小王,折疊表達(dá)式雖好,但也有個坑要注意!" 老張突然嚴(yán)肅起來 ??

"什么坑啊?" 小王緊張地問 ??

"空參數(shù)包的問題!" 老張豎起食指 ??

template<typename... Args>
bool all(Args... args) {
    return (... && args);    // 安全 ?
    // return (... + args);  // 危險 ?
}

"哦!原來只有 &&、|| 和逗號運算符才能安全處理空參數(shù)包!" 小王恍然大悟 ??

"對頭!" 老張點點頭,"就像自動門雖然方便,但停止時還得靠人工開關(guān)一樣!" ??

折疊表達(dá)式的語法細(xì)節(jié)

"小王,來看看折疊表達(dá)式的四種基本形式!" 老張拿起馬克筆,在白板上畫起來 ??

// 第一種: 一元右折疊 - 像疊紙飛機一樣從右往左折 ??
(pack op ...)         
// 例如: (args + ...) 會展開成 a1 + (a2 + (a3 + a4))

"哦!這就像從右邊開始疊紙飛機!" 小王恍然大悟 ??

// 第二種: 一元左折疊 - 像疊信封一樣從左往右折 ??
(... op pack)         
// 例如: (... + args) 會展開成 ((a1 + a2) + a3) + a4

"對,再看看帶初始值的版本:" 老張繼續(xù)寫道:

// 第三種: 二元右折疊 - 最后再加個櫻桃 ??
(pack op ... op init) 
// 例如: (args + ... + 100) 變成 a1 + (a2 + (a3 + 100))

// 第四種: 二元左折疊 - 先放個底座再開始 ???
(init op ... op pack) 
// 例如: (100 + ... + args) 變成 ((100 + a1) + a2) + a3

"這里的 op 可以用很多運算符哦!" 老張解釋道,"我們把它們分類一下:" ??

// 1?? 算術(shù)運算符 - 做數(shù)學(xué)計算用
+, -, *, /, %

// 2?? 位運算符 - 處理二進制位
^, &, |, <<, >>

// 3?? 賦值運算符 - 存儲值用
=, +=, -=, *=, /=, %=, ^=, &=, |=, <<=, >>=

// 4?? 比較運算符 - 判斷大小關(guān)系
==, !=, <, >, <=, >=

// 5?? 邏輯運算符 - 處理真假值
&&, ||

// 6?? 其他特殊運算符
,(逗號), .*, ->*

"哇!原來可以用這么多運算符!" 小王驚嘆道 ??

"是的,不同的運算符可以實現(xiàn)不同的功能。" 老張笑著說,"就像廚師的各種刀工一樣,要用對工具!" ??

實用示例大放送

"來看幾個實際應(yīng)用吧!" 老張興致勃勃地說。

(1) 打印神器 ???

template<typename... Args>
void printer(Args&&... args) {
    (std::cout << ... << args) << '\n';  // 一元左折疊
}

"看這個!" 老張指著代碼說,"用一元左折疊實現(xiàn)打印,就像串糖葫蘆一樣,一個個打印出來!" ??

使用示例:

printer("你好", 42, "世界", 3.14);  // 輸出: 你好42世界3.14

(2) 類型極限探索者 ??

template<typename... Ts>
void print_limits() {
    ((std::cout << +std::numeric_limits<Ts>::max() << ' '), ...) << '\n';
}

"這個更有意思," 老張解釋道,"它能打印出不同類型的最大值。逗號運算符配合折疊表達(dá)式,就像魔術(shù)師變戲法一樣!" ??

使用示例:

print_limits<char, int, long>();  // 輸出: 127 2147483647 9223372036854775807

(3) Vector 快速填充器 ??

template<typename T, typename... Args>
void push_back_vec(std::vector<T>& v, Args&&... args) {
    // 先檢查類型是否匹配,就像檢查鑰匙能否開鎖 ??
    static_assert((std::is_constructible_v<T, Args&&> && ...)); 
    
    // 然后一個個放入vector,像往背包里裝東西 ??
    (v.push_back(std::forward<Args>(args)), ...);
}

"這個厲害了!" 小王眼前一亮,"不僅能批量添加元素,還能在編譯期檢查類型!"

使用示例:

std::vector<int> nums;
push_back_vec(nums, 1, 2, 3, 4, 5);  // 一次性添加多個數(shù)字 ?

"對啊," 老張笑著說,"現(xiàn)代C++就是這么優(yōu)雅,既安全又高效!" ??

"這...這簡直是魔法!" 小王目瞪口呆 ??

知識點總結(jié)

"誒,老張," 小王摸著下巴思考道,"今天學(xué)到的這個折疊表達(dá)式,能幫我總結(jié)一下它的精髓嗎?" ??

"當(dāng)然可以!" 老張放下咖啡杯,"我們來對比一下新舊方案:"

傳統(tǒng)寫法的痛點 ??:

  • 需要寫多個重載函數(shù) ??
  • 遞歸實現(xiàn)復(fù)雜且難維護 ??
  • 編譯生成大量函數(shù)調(diào)用 ??
  • 運行時性能有額外開銷 ??

折疊表達(dá)式的優(yōu)勢 ??:

  • 一個模板搞定所有情況 ?
  • 代碼簡潔優(yōu)雅,易于理解 ??
  • 編譯期展開,無遞歸開銷 ??
  • 運行時性能更優(yōu),直接內(nèi)聯(lián) ??

"哦!原來如此!" 小王恍然大悟,"感覺這就像是把復(fù)雜的積木搭建,變成了優(yōu)雅的折紙藝術(shù)!" ??

"沒錯!" 老張笑著說,"記住一點:現(xiàn)代C++的核心思想就是讓復(fù)雜的事情變得簡單,讓危險的操作變得安全。折疊表達(dá)式就是最好的例子!" ??

"太棒了!這下我可以告別模板地獄了!" 小王開心地說。

"學(xué)習(xí)新特性,就要敢于擁抱變化。" 老張拍拍小王的肩膀,"讓代碼既簡潔又高效,這才是現(xiàn)代C++的魅力所在!" ?

責(zé)任編輯:趙寧寧 來源: everystep
相關(guān)推薦

2024-04-23 08:26:56

C++折疊表達(dá)式編程

2024-12-19 11:30:00

C++17CTAD代碼

2024-12-27 12:00:00

C++17枚舉

2025-01-02 15:14:01

2020-06-04 09:18:52

CTOif-else代碼

2021-07-16 08:26:18

折疊表達(dá)式參數(shù)

2024-12-18 06:00:00

C++17C++

2024-12-13 15:50:00

C++編程代碼

2013-04-10 10:58:19

LambdaC#

2009-08-31 17:11:37

Lambda表達(dá)式

2021-03-02 07:33:13

開發(fā)C#字符

2019-04-16 13:30:05

表達(dá)式求值數(shù)據(jù)結(jié)構(gòu)算法

2024-03-25 13:46:12

C#Lambda編程

2020-06-15 08:12:51

try catch代碼處理器

2009-08-27 09:44:59

C# Lambda表達(dá)

2009-08-07 15:41:39

C#正規(guī)表達(dá)式

2009-10-12 10:11:08

Lambda表達(dá)式編寫

2024-12-24 12:10:00

代碼C++Lambda

2009-08-03 17:27:14

C#正則表達(dá)式

2009-08-26 16:17:23

C# Lambda表達(dá)
點贊
收藏

51CTO技術(shù)棧公眾號