58 C++ 現代C++編程藝術7-模板友元

C++ 現代C++編程藝術7-模板友元

文章目錄

  • C++ 現代C++編程藝術7-模板友元
    • 一、基礎應用場景 🧩
      • 1. 模板類聲明友元函數
      • 2. 普通類聲明模板函數為友元
    • 二、模板類互訪場景 ??
      • 1. 同類模板互訪(一對一)
      • 2. 異類模板互訪(多對多)
    • 三、高級用法:綁定特定實例 🚀
    • 四、常見陷阱與規避方案 ??
    • 五、現代C++(C++11~20)
      • 1. 友元類型別名(C++11)
      • 2. 模板友元簡化語法(C++17)

核心定義:模板友元(Template Friend)是C++中允許模板類或函數成為另一個類(模板類或普通類)友元的機制,用于突破封裝限制,實現跨模板的私有成員訪問。

一、基礎應用場景 🧩

1. 模板類聲明友元函數

#include <iostream>template<typename T>
class Container {T m_data;  // 私有成員 
public:// 聲明模板函數為友元(允許訪問私有成員)template<typename U>friend void peek(const Container<U>& c);Container(T dat):m_data(dat){}~Container() = default;
};template<typename U>
void peek(const Container<U>& c) {std::cout << "訪問私有數據: " << c.m_data;  // ? 合法
}int main() {Container<int> A(2);peek(A);//訪問私有數據: 2return 0;
}

2. 普通類聲明模板函數為友元

#include <iostream>class Container {
int m_data;  // 私有成員 
public:// 聲明模板函數為友元(允許訪問私有成員)
template<typename U>
friend void peek(U a,  Container& c);Container(int dat):m_data(dat){}
~Container() = default;
};template<typename U>
void peek(U a,  Container& c) 
{
std::cout << "私有數據: " << c.m_data;  
c.m_data = a;// ? 合法
std::cout << " 改為 " << c.m_data;  
}int main() {Container A(2);
peek<int >(3, A);//私有數據: 2 改為 3return 0;
}

二、模板類互訪場景 ??

1. 同類模板互訪(一對一)

#include <iostream>template<typename T>
class BoxA {T secret_;public:BoxA(T val = T{}) : secret_(val) {}  // ? 默認初始化// 聲明同類型模板類為友元// friend class BoxA<U>;  // ? 錯誤!需前置聲明// 正確寫法:template<typename U> friend class BoxA;  // ? 所有BoxA實例均為友元 template<typename U>void print(const BoxA<U>& other) {  // ? 通過友元訪問other的私有成員 std::cout << "跨類型訪問: " << other.secret_  << std::endl;}
};int main() {BoxA<int> a(42);BoxA<double> b(3.14);a.print(b);  // ? 跨類型訪問: 3.14b.print(a);  // ? 跨類型訪問: 42return 0;
}

2. 異類模板互訪(多對多)

#include <iostream>// 前置聲明 
template<typename T> class classA;
template<typename U> class classB;template<typename T>
class classA {T raw_data_;
public:classA(T val = T{}) : raw_data_(val) {}  // ? 默認初始化~classA() = default;// 聲明特定模板類為友元template<typename U> friend class classB;template<typename U>void inspect(const classB<U>& b) const {std::cout << "classA 訪問 classB 的secret_:" << b.secret_  << std::endl;}};template<typename U>
class classB {U secret_;  // 添加私有成員 
public:explicit classB(U val = U{}) : secret_(val) {}  // ? 默認初始化~classB() = default;// 聲明classA為友元以實現雙向訪問 template<typename T> friend class classA;template<typename T>void display(classA<T>& a) {std::cout << "classB 訪問 classA 的raw_data_:" << a.raw_data_  << std::endl;}
};int main() {classA<int> a_int(100);classA<double> a_double(3.1415);classB<std::string> b_str("SECRET-2025");classB<char> b_char('X');// 多對多互訪演示 b_str.display(a_int);      // classB<string> → classA<int>b_char.display(a_double);  // classB<char> → classA<double>a_int.inspect(b_str);      // classA<int> → classB<string>a_double.inspect(b_char);  // classA<double> → classB<char>return 0;
}

🌟 關鍵技術突破:

  1. 雙向模板友元機制

    • classAclassB互相聲明為模板友元
    • 突破傳統單向友元限制,實現環形訪問權限
  2. 泛型成員函數模板

    • display/inspect方法使用嵌套模板參數
    • 支持任意classA<T>classB<U>組合訪問
  3. 類型安全增強

    • explicit構造函數阻止隱式轉換

三、高級用法:綁定特定實例 🚀

#include <iostream>// 前置聲明 
template<typename T> class Worker;
template<typename T, typename U> void collaborate(Worker<T>&, Worker<U>&);template<typename T>
class Worker {T task_;// 僅綁定同類型實例的協作函數friend void collaborate<>(Worker<T>&, Worker<T>&);  // ? 注意<>public:explicit Worker(T task) : task_(task) {} };// 模板函數實現 
template<typename T, typename U>
void collaborate(Worker<T>& a, Worker<U>& b) {std::cout << "協作結果: " << a.task_  + b.task_  << "\n";// ? 當T==U時可編譯,否則報錯(實現精確控制)
}int main() {Worker<int> A(8);Worker<int> B(4);collaborate(A,B);// ? 合法但無意義調用(需約束類型)Worker<std::string> writer("Hello");Worker<std::string> editor("World");collaborate(writer, editor);        // 輸出拼接字符串 "HelloWorld"// ?? 觸發編譯錯誤的場景演示(注釋保留)// Worker<double> analyst(3.14);// collaborate(A, analyst);  // ? 錯誤:無法訪問task_return 0;
}

核心機制解析 🔍

  1. 精確友元控制原理

    • 通過friend void collaborate<>(Worker<T>&, Worker<T>&)
    • 僅當模板參數T == U時,collaborate函數才能訪問task_
    • 實現類型安全的協作限制
  2. 編譯期錯誤觸發邏輯

    template<typename T, typename U>
    void collaborate(Worker<T>& a, Worker<U>& b) {// 當T≠U時,此處訪問a.task_和b.task_會觸發:// error: 'task_' is private within this context 
    }
    

典型應用場景 📊

場景數據類型組合協作邏輯
程序員協作Worker<int> + Worker<int>合并代碼行數
文本編輯Worker<std::string> ×2字符串拼接
數值計算Worker<Matrix> ×2矩陣相加
資源調度Worker<MemoryBlock> ×2內存塊合并

四、常見陷阱與規避方案 ??

  1. 循環依賴問題

    // 錯誤:相互友元導致無限依賴 
    template<typename T> class A { friend class B<T>; };
    template<typename T> class B { friend class A<T>; };  // ? 編譯失敗 // 方案:使用前置聲明 + 單向友元
    template<typename T> class B;  // 前置聲明
    template<typename T> class A { friend class B<T>; };  // ?
    
  2. 過度暴露風險

    - friend class AllTemplate<T>;  // 危險:所有實例化均可訪問
    + friend class SpecificPartner<T>;  // 安全:僅特定模板可訪問
    
  3. 鏈接錯誤處理

    // 聲明與實現分離時需顯式實例化
    template class Monitor<int>;  // 強制實例化避免鏈接錯誤
    

五、現代C++(C++11~20)

1. 友元類型別名(C++11)

template<typename T>
using DataInspector = Inspector<T, AuditTag>;class FinancialRecord {friend DataInspector<double>;  // ? 僅授權審計專用的檢查器
};

2. 模板友元簡化語法(C++17)

template<typename T>
class SecureVault {// 無需前置聲明直接定義友元模板函數 friend void emergency_reset<>(SecureVault<T>&);  // ? <C++17 
};

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

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

相關文章

Undertow —— JBOSS 的社區版,redhat 下場維護的開源項目,頂頂好用的 Java web server

Undertow JBoss Community Undertow Undertow is a flexible performant web server written in java, providing both blocking and non-blocking API’s based on NIO. Undertow 是一個用 Java 編寫的靈活高性能 Web 服務器&#xff0c;提供基于 NIO 的阻塞和非阻塞 API。…

【AI智能體】Dify 搭建業務單據差異核對助手實戰詳解

目錄 一、前言 二、Dify介紹 2.1 Dify 是什么 2.2 Dify 核心特性 2.2.1 Dify特點 2.2.2 Dify 多模型支持 2.2.3 Dify 適應場景 2.2.4 基于Dify 搭建發票識別應用優勢 三、Dify 搭建業務單據核對助手實戰過程 3.1 前置準備 3.1.1 安裝必要的插件 3.2 完整操作步驟 3…

Centos編譯安裝Python3.10

gcc編譯源碼包 下載python源碼包并解壓 wget https://www.python.org/ftp/python/3.10.18/Python-3.10.18.tgz tar -xf Python-3.10.18.tgz cd Python-3.10.18系統編譯依賴環境安裝 sudo yum install zlib-devel ncurses-devel gdbm-devel nss-devel openssl-devel readline-de…

Maya 3D建模 導入參考圖、鎖定參考圖

1 導入參考圖切換到 前視圖 或者 側視圖 導入 &#xff08;根據參考圖片類別去選擇&#xff09;方法1&#xff1a;視圖--圖像平面--導入圖像方法2&#xff1a;直接點 圖像平面 備注&#xff1a;誤操作導致看不到 解決辦法&#xff1a;顯示--視口 找對應的2 鎖定參考圖目的&…

基于單片機智能加濕器/空氣加濕器

傳送門 &#x1f449;&#x1f449;&#x1f449;&#x1f449;其他作品題目速選一覽表 &#x1f449;&#x1f449;&#x1f449;&#x1f449;其他作品題目功能速覽 概述 基于單片機的智能加濕器通過集成溫濕度傳感器、控制模塊和霧化裝置&#xff0c;實現環境濕度的自…

SNDR:高精度ADC系統的綜合性能標尺

SNDR&#xff1a;高精度ADC系統的綜合性能標尺 一、SNDR的本質定義與理論基礎 信噪失真比(Signal-to-Noise-and-Distortion Ratio) 是評估ADC系統綜合性能的核心指標&#xff0c;定義為信號功率與噪聲及失真功率之和的比值&#xff1a; SNDRdB10log?10(PsignalPnoisePdistorti…

2025年滲透測試面試題總結-31(題目+回答)

安全領域各種資源&#xff0c;學習文檔&#xff0c;以及工具分享、前沿信息分享、POC、EXP分享。不定期分享各種好玩的項目及好用的工具&#xff0c;歡迎關注。 目錄 一、代碼審計核心思路&#xff08;261&#xff09; 二、MySQL Getshell前提&#xff08;262&#xff09; …

[創業之路-560]:機械、電氣、自控、電子、軟件、信息、通信、大數據、人工智能,上述技術演進過程

上述關鍵詞反映的技術演進過程可梳理為一條從機械執行到智能決策的遞進式發展主線&#xff0c;各技術領域在不同階段相互滲透、共同推動機器人技術從功能替代向認知革命躍遷。以下是具體演進邏輯與趨勢分析&#xff1a;一、技術演進的三階段遞進機械主導階段&#xff08;工業革…

芋道前端項目部署后刷新 404 的解決辦法(Nginx 配置教程)

很多同學在把 芋道前端項目 部署到服務器后&#xff0c;會遇到一個奇怪的問題&#xff1a; &#x1f449; 項目首頁能正常訪問&#xff0c;但一旦在瀏覽器里手動刷新某個頁面&#xff0c;就會報 404 Not Found 錯誤。 這到底是為什么呢&#xff1f;又該怎么解決呢&#xff1f;下…

更適合后端寶寶的前端三件套之HTML

文章目錄&#x1f4d5;1. HTML基礎??1.1 什么是HTML??1.2 認識HTML標簽??1.3 HTML文件基本結構??1.4 標簽層次結構&#x1f4d5;2. HTML常見標簽??2.1 標題標簽??2.2 段落標簽??2.3 換行標簽??2.4 圖片標簽??2.5 超鏈接標簽??2.6 表格標簽&#x1f4d5;3. …

【JVM內存結構系列】四、不同垃圾回收器與堆內存的適配關系:從分代GC到Region GC

在JVM內存體系中&#xff0c;堆內存的“分代結構”與“對象流轉規則”是通用基礎&#xff0c;但垃圾回收器&#xff08;GC&#xff09;是決定堆內存實際表現的核心變量——不同GC為實現“低延遲”“高吞吐量”等目標&#xff0c;會對堆的劃分方式、對象管理邏輯、參數配置規則進…

Zemax光學設計輸出3D

輸出立體數據文件&#xff08;IGES/STEP/SAT/STL 格式&#xff09;的參數設置界面&#xff0c;各參數含義如下&#xff1a;1. 起始面/終止面&#xff1a;設定要輸出立體數據對應的光學表面范圍&#xff0c;從第 0 個表面到第 9 個表面 &#xff0c;限定參與輸出的光學結構表面區…

模塊測試與低功耗模式全攻略

一、模塊測試流程在測試一個模塊時&#xff0c;建議遵循以下步驟&#xff1a;基本測試&#xff1a;測試該模塊的寄存器讀寫功能是否正常。可以向每個寄存器寫入 0x5A5A 和 0xA5A5&#xff0c;這兩種模式可以覆蓋對寄存器寫入 0 和 1 的情況。進階測試&#xff1a;在基本測試通過…

機器學習實驗三、使用決策樹算法預測泰坦尼克號幸存者

實驗目的1. 掌握特征工程&#xff0c;會進行特征提取與特征選擇&#xff0c;會進行缺失值填充。2. 建立決策樹模型&#xff0c;解決實際問題。3. 會對模型進行調試&#xff0c;能夠繪制并保存決策樹。實驗環境Python 3.7.0&#xff0c;Sklearn &#xff0c;PyCharm實驗原理1、特…

從全棧開發到微服務架構:一次真實的Java面試實錄

從全棧開發到微服務架構&#xff1a;一次真實的Java面試實錄 面試官與應聘者介紹 面試官&#xff1a;李明&#xff0c;某互聯網大廠技術負責人&#xff0c;擅長Java后端、微服務及云原生架構。 應聘者&#xff1a;張偉&#xff0c;28歲&#xff0c;碩士學歷&#xff0c;擁有5年…

新的 Gmail 網絡釣魚攻擊利用 AI 提示注入來逃避檢測

網絡釣魚一直以來都是為了欺騙人們。但在這次活動中&#xff0c;攻擊者不僅瞄準用戶&#xff0c;還試圖操縱基于人工智能的防御系統。 這是我上周記錄的Gmail 網絡釣魚鏈的演變。那次攻擊活動依賴于緊迫性和重定向&#xff0c;但這次引入了隱藏的 AI 提示&#xff0c;旨在混淆…

Restful風格設計

文章目錄什么是Restful風格&#xff1f;RESTful API設計最佳實踐1. URL設計原則2. HTTP狀態碼的正確使用3. 統一的響應格式實際案例&#xff1a;用戶管理系統API總結什么是Restful風格&#xff1f; 我的理解是&#xff1a;Restful是一種基于HTTP協議的架構設計風格&#xff0c…

深入 Glide 圖像變換:自定義效果、GIF處理與組合變換

在 Android 開發中&#xff0c;Glide 的強大不僅在于其高效的加載和緩存能力&#xff0c;更在于其無與倫比的可擴展性&#xff0c;尤其是在圖像處理層面。當內置的 fitCenter() 和 circleCrop() 無法滿足你的設計需求時&#xff0c;自定義 Transformation 便是你的終極武器。本…

數據挖掘 4.8 評估泛化能力

4.8 Estimating Generalization 4.8 評估泛化能力 評估模型的泛化能力如何合理評估模型的泛化能力指導原則 (Guidelines)存在的問題 (Issues)K-fold 交叉驗證&#xff08;Cross-Validation)留一交叉驗證&#xff08;Leave One Out CV&#xff09;(LOOCV)Stratification 分層訓練…

46.【.NET8 實戰--孢子記賬--從單體到微服務--轉向微服務】--擴展功能--集成網關--網關集成日志

本篇文章&#xff0c;我們一起在網關中集成日志功能&#xff0c;我們要在網關中記錄下游微服務出現的異常信息、請求信息以及響應信息。在微服務架構中&#xff0c;網關作為系統的入口&#xff0c;承擔著非常重要的職責。通過在網關層面集成日志功能&#xff0c;我們可以更好地…