C++ CRTP

C++ CRTP(奇異遞歸模板模式)


CRTP 是什么?

一句話總結:CRTP 就是讓子類把自己作為模板參數傳遞給父類

聽起來有點繞,直接上代碼就明白了:

template <typename Derived>
class Base {// ...
};class Derived : public Base<Derived> {// ...
};

Derived 繼承自 Base<Derived>,也就是說,子類把自己“遞歸”地傳給了父類。這就是“奇異遞歸”的由來。


CRTP 有啥用?

其實 CRTP 最常用的場景有三個:

  1. 靜態多態:不用虛函數也能實現類似多態的效果,而且沒有虛表,效率高。
  2. 代碼復用:基類寫通用邏輯,子類只需要實現自己的部分。
  3. 每個子類獨立的靜態成員:比如計數器,每個子類都有自己的靜態變量。

CRTP 的原理

CRTP 的核心原理其實很簡單,就是利用了 C++ 模板的“編譯期展開”特性,讓基類在編譯時就能知道派生類的類型。

1. static_cast 的作用

在 CRTP 里,基類通常會用 static_cast<Derived*>(this) 把自己轉換成派生類指針,然后調用派生類的方法。這樣,雖然代碼寫在基類里,但實際調用的是派生類的實現。

比如:

void interface() {static_cast<Derived*>(this)->implementation();
}

這行代碼在編譯期就能確定 Derived 的類型,所以沒有虛表,也沒有運行時開銷。

2. 編譯期多態的本質

CRTP 實現的是“靜態多態”,也就是多態的分發發生在編譯期,而不是運行時。模板展開時,基類里的 static_cast<Derived*> 會被替換成具體的派生類類型,所有調用都在編譯時就確定了。

3. 代碼復用和靜態接口約束

  • 代碼復用:基類可以寫通用的邏輯,比如日志、計數、接口包裝等,具體實現交給派生類。這樣不同的派生類可以復用同一套基類邏輯。
  • 靜態接口約束:如果派生類沒有實現基類里要調用的方法(比如 implementation()),編譯時就會報錯。這其實是一種“編譯期接口檢查”,比傳統的虛函數更早發現問題。

一個簡單的例子

假如我有一堆不同的動物,每種動物都能“說話”,但我又不想用虛函數(比如對性能有要求),CRTP 就能派上用場:

#include <iostream>template <typename Derived>
class Animal {
public:void speak() {static_cast<Derived*>(this)->speak_impl();}
};class Dog : public Animal<Dog> {
public:void speak_impl() {std::cout << "汪汪!" << std::endl;}
};class Cat : public Animal<Cat> {
public:void speak_impl() {std::cout << "喵喵!" << std::endl;}
};int main() {Dog d;Cat c;d.speak(); // 汪汪!c.speak(); // 喵喵!return 0;
}

這里的 Animal 基類里有個 speak(),但真正的實現是在子類里。通過 static_cast<Derived*>(this),基類可以“靜態”地調用子類的方法。這樣既有多態的效果,又沒有虛函數的開銷。


CRTP 實例

1. 每個子類獨立計數

有時候我想統計每種類型各自創建了多少對象,CRTP 也能輕松搞定:

template <typename Derived>
class Counter {
public:static int count;Counter() { ++count; }
};
template <typename Derived>
int Counter<Derived>::count = 0;class Apple : public Counter<Apple> {};
class Banana : public Counter<Banana> {};int main() {Apple a1, a2;Banana b1;std::cout << Apple::count << std::endl;  // 輸出2std::cout << Banana::count << std::endl; // 輸出1
}

每個子類都有自己的靜態成員變量,互不影響。

2 . 日志

#include <iostream>
#include <string>// CRTP 日志基類
template <typename Derived>
class LoggerBase {
public:void runWithLog(const std::string& opName) {std::cout << "[LOG] 開始操作: " << opName << std::endl;static_cast<Derived*>(this)->run();  // 調用派生類的 run()std::cout << "[LOG] 結束操作: " << opName << std::endl;}
};// 業務類A
class MyAlgorithm : public LoggerBase<MyAlgorithm> {
public:void run() {std::cout << "算法A正在運行..." << std::endl;}
};// 業務類B
class MyService : public LoggerBase<MyService> {
public:void run() {std::cout << "服務B正在處理..." << std::endl;}
};int main() {MyAlgorithm algo;MyService svc;algo.runWithLog("算法A任務");svc.runWithLog("服務B任務");return 0;
}

CRTP 和虛函數的區別

  • 虛函數:運行時多態,有虛表指針,靈活但有點性能損耗。
  • CRTP:編譯期多態,沒有虛表,效率高,但只能在編譯期確定類型。


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

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

相關文章

21.映射字典的值

有時候你會希望保留字典的鍵不變,但將每個鍵對應的值應用一個函數進行轉換,比如提取字段、做數學運算、格式化等。 ? 基本用法 你可以使用 dict.items() 搭配字典推導式或生成器表達式來實現。 def map_values(obj, fn):return dict((k, fn(v)

【算法】貪心算法:擺動序列C++

文章目錄前言題目解析算法原理代碼示例策略證明前言 題目的鏈接&#xff0c;大家可以先試著去做一下再來看一下思路。376. 擺動序列 - 力扣&#xff08;LeetCode&#xff09; 題目解析 將題目有用的信息劃出來&#xff0c;結合示例認真閱讀&#xff0c;去理解題目。 我們的擺…

【DOCKER】-6 docker的資源限制與監控

文章目錄1、docker的資源限制1.1 容器資源限制的介紹1.2 OOM1.3 容器的內存限制1.3.1 內存限制的相關選項1.4 容器的CPU限制介紹2、docker的監控插件2.1 cadvisor2.2 portainer1、docker的資源限制 1.1 容器資源限制的介紹 默認情況下&#xff0c;容器沒有資源的使用限制&…

gcc 源碼分析--gimple 關鍵數據結構

gimple 操作碼&#xff0c;支持這些&#xff1a;DEFGSCODE(GIMPLE_symbol, printable name, GSS_symbol). */ DEFGSCODE(GIMPLE_ERROR_MARK, "gimple_error_mark", GSS_BASE) DEFGSCODE(GIMPLE_COND, "gimple_cond", GSS_WITH_OPS) DEFGSCODE(GIMPLE_DEBU…

TDengine GREATEST 和 LEAST 函數用戶手冊

TDengine GREATEST 和 LEAST 函數用戶手冊 1. 需求背景 1.1 問題描述 在實際生產過程中&#xff0c;客戶經常需要計算三相電流、電壓的最大值和最小值。傳統的實現方式需要使用復雜的 CASE WHEN 語句&#xff0c;例如&#xff1a; -- 傳統方式&#xff1a;計算三相電流最大…

Redis 與數據庫不一致問題及解決方案

一、不一致的原因分析 1. 緩存更新策略不當 先更新數據庫后刪除緩存:刪除緩存失敗會導致不一致 先刪除緩存后更新數據庫:并發請求可能導致不一致 緩存穿透:大量請求直接打到數據庫,繞過緩存 2. 并發操作問題 讀寫并發:讀請求獲取舊緩存時,寫請求更新了數據庫但未更新緩存…

iOS 加固工具使用經驗與 App 安全交付流程的實戰分享

在實際開發中&#xff0c;iOS App不僅要安全&#xff0c;還要能被穩定、快速、無誤地交付。這在外包、B端項目、渠道分發、企業自用系統等場景中尤為常見。 然而&#xff0c;許多開發者在引入加固工具后會遇到以下困擾&#xff1a; 混淆后App運行異常、不穩定&#xff1b;資源路…

Windows 下 Visual Studio 開發 C++ 項目的部署流程

在Windows環境中使用Visual Studio&#xff08;以下簡稱VS&#xff09;開發C項目時&#xff0c;“部署”是確保程序能在目標設備上正常運行的關鍵環節。部署的核心目標是&#xff1a;將編譯生成的可執行文件&#xff08;.exe&#xff09;、依賴的動態鏈接庫&#xff08;.dll&am…

yolo8+聲紋識別(實時字幕)

現在已經完成了人臉識別跟蹤 ?&#xff0c;接下來要&#xff1a; ? 加入「聲紋識別&#xff08;說話人識別&#xff09;」功能&#xff0c;識別誰在講話&#xff0c;并在視頻中“這個人”的名字旁邊加上「正在講話」。 這屬于多模態識別&#xff08;視覺 音頻&#xff09;&a…

DH(Denavit–Hartenberg)矩陣

DH 矩陣&#xff08;Denavit-Hartenberg 矩陣&#xff09;是 1955 年由 Denavit 和 Hartenberg 提出的一種機器人運動學建模方法&#xff0c;用于描述機器人連桿和關節之間的關系。該方法通過在機器人每個連桿上建立坐標系&#xff0c;并用 44 的齊次變換矩陣&#xff08;DH 矩…

Vim的magic模式

在 Vim 中&#xff0c;magic 模式用于控制正則表達式中特殊字符的解析方式。它決定了哪些字符需要轉義才能發揮特殊作用&#xff0c;從而影響搜索和替換命令的寫法。以下是詳細介紹&#xff1a; 一、三種 magic 模式 Vim 提供三種 magic 模式&#xff0c;通過在正則表達式前添加…

Git 使用技巧與原理(一)—— 基礎操作

1、起步 1.1 版本控制 版本控制是一種記錄一個或若干文件內容變化&#xff0c;以便將來查閱特定版本修訂情況的系統。 版本控制系統&#xff08;VCS&#xff0c;Version Control System&#xff09;通常可以分為三類&#xff1a; 本地版本控制系統&#xff1a;大多都是采用某…

軟件測試之自動化測試

目錄 1.什么是自動化測試 2.web?動化測試 2.1驅動 WebDriverManager 3. Selenium 3.1selenium驅動瀏覽器的?作原理 4.常用函數 4.1元素的定位 4.1.1cssSelector選擇器 4.2.2xpath 4.2操作測試對象 4.3窗? 4.4等待 4.5瀏覽器導航 4.6彈窗 4.7文件上傳 4.8設置…

sqlserver遷移日志文件和數據文件

sqlserver安裝后沒有指定日志存儲路徑或者還原庫指定的日志存儲位置不理想想要更改&#xff0c;都可以按照這種方式來更換&#xff1b;1.前提準備&#xff1a;數據庫的備份bak文件2.查看自己當前數據庫的日志文件和數據文件存儲路徑是否理想選中當前數據庫&#xff0c;右鍵屬性…

MFC UI表格制作從專家到入門

文章目錄CListCtrl常見問題增強版CGridCtrl&#xff08;第三方&#xff09;第三方庫ReoGridCListCtrl 默認情況下&#xff0c;CListCtrl不支持直接編輯單元格&#xff0c;需通過消息處理實現。 1.添加控件到資源視圖 在對話框資源編輯器中拖入List Control控件&#xff0c;設…

數字后端APR innovus sroute到底是如何選取寬度來鋪power rail的?

吾愛IC社區新一期IC訓練營將于7月初開班&#xff08;07.06號晚上第一次直播課&#xff09;&#xff01;社區所有IC后端訓練營課程均為直播課&#xff01;全網唯一一家敢開后端直播課的&#xff08;口碑不好招生一定存在困難&#xff0c;自然就無法開直播課&#xff09;&#xf…

LVS集群技術

LVS&#xff08;Linux Virtual Server&#xff09;是一種基于Linux內核的高性能、高可用性服務器集群技術&#xff0c;它通過負載均衡將客戶端請求分發到多臺后端真實服務器&#xff0c;實現 scalability 和 fault tolerance。LVS工作在傳輸層&#xff08;OSI Layer 4&#xff…

git項目,有idea文件夾,怎么去掉

要從Git項目中排除.idea文件夾&#xff08;IntelliJ IDEA的配置文件目錄&#xff09;&#xff0c;可以通過以下步驟操作&#xff1a; 1. 添加.gitignore規則 在項目根目錄創建或編輯.gitignore文件&#xff0c;添加以下內容&#xff1a; .idea/2. 從Git緩存中刪除已跟蹤的.idea…

springboot+swagger2文檔從swagger-bootstrap-ui更換為knife4j及文檔接口參數不顯示問題

背景 已有springboot項目,且使用的是swagger2+swagger-bootstrap-ui的版本 1.pom依賴如下 <!-- Swagger接口管理工具 --><dependency><groupId>io.springfox</groupId><artifactId>springfox-swagger2</artifactId><version>2.9…

mysql數據庫表只能查詢,對于插入、更新、刪除操作一直卡住,直到報錯Lost connection to MySQL server during query

診斷步驟1. 查看阻塞進程SELECT * FROM performance_schema.metadata_locks WHERE LOCK_STATUS PENDING;SELECT * FROM sys.schema_table_lock_waits;2. 查看當前活動事務SELECT * FROM information_schema.INNODB_TRX;3. 查看進程列表SHOW PROCESSLIST;通過SELECT * FROM in…