您当前的位置:首页 > 电脑百科 > 程序开发 > 语言 > C/C++/C#

C++类模板特化与继承使用说明书,新手也能get

时间:2023-12-21 13:18:38  来源:微信公众号  作者:coding日记

一、类模板特化

1.特化的实现

你可以为特定类型提供类模板的替代实现。例如,你可能认为 const char 类型(C 风格字符串)的 Grid 行为没有意义。Grid<const char> 将在 vector<vector<optional<const char*>>> 中存储其元素。拷贝构造函数和赋值运算符将执行这些 const char 指针类型的浅拷贝。对于 const char,进行深拷贝字符串可能更有意义。最简单的解决方案是为 const char 编写一个专门的实现,将它们转换为 C++ 字符串,并存储在 vector<vector<optional<string>>> 中。

C++类模板特化与继承使用说明书,新手也能get

模板的替代实现称为模板特化。你可能会发现其语法初看有些奇怪。当你编写类模板特化时,你必须指定这是模板,并且你正在为特定类型编写模板的版本。以下是 Grid 的 const char

export module grid:string;

// 当使用模板特化时,原始模板也必须可见。
import :mAIn;

export template <>
class Grid<const char*> {
public:
    explicit Grid(size_t width = DefaultWidth, size_t height = DefaultHeight);
    virtual ~Grid() = default;
    // 明确默认拷贝构造函数和赋值运算符。
    Grid(const Grid& src) = default;
    Grid& operator=(const Grid& rhs) = default;
    // 明确默认移动构造函数和赋值运算符。
    Grid(Grid&& src) = default;
    Grid& operator=(Grid&& rhs) = default;

    std::optional<std::string>& at(size_t x, size_t y);
    const std::optional<std::string>& at(size_t x, size_t y) const;

    size_t getHeight() const { return m_height; }
    size_t getWidth() const { return m_width; }

    static const size_t DefaultWidth { 10 };
    static const size_t DefaultHeight { 10 };

private:
    void verifyCoordinate(size_t x, size_t y) const;
    std::vector<std::vector<std::optional<std::string>>> m_cells;
    size_t m_width { 0 }, m_height { 0 };
};

注意,在特化中你不使用任何类型变量,例如 T,你直接使用 const char* 和字符串。此时一个明显的问题是,为什么这个类仍然是模板。即,以下语法有什么用途?

template <> class Grid<const char*>

这种语法告诉编译器,这个类是 Grid 类的 const char* 特化。假设你没有使用这种语法,而是尝试编写如下代码:

class Grid

编译器不会允许你这样做,因为已经存在一个名为 Grid 的类模板(原始类模板)。只有通过特化,你才能重用这个名称。特化的主要好处是它们对用户来说可以是不可见的。当用户创建 int 或 SpreadsheetCells 的 Grid 时,编译器会从原始 Grid 模板生成代码。当用户创建 const char* 的 Grid 时,编译器使用 const char* 特化。这一切都可以在“幕后”进行。

2.主模块接口文件

主模块接口文件简单地导入并导出两个模块接口分区:

export module grid;
export import :main;
export import :string;

特化可以按照以下方式进行测试:

Grid<int> myIntGrid; // 使用原始 Grid 模板。
Grid<const char*> stringGrid1 { 2, 2 }; // 使用 const char* 特化。
const char* dummy { "dummy" };
stringGrid1.at(0, 0) = "hello";
stringGrid1.at(0, 1) = dummy;
stringGrid1.at(1, 0) = dummy;
stringGrid1.at(1, 1) = "there";
Grid<const char*> stringGrid2 { stringGrid1 };

当你特化一个模板时,你不会“继承”任何代码;特化不像派生。你必须重写类的整个实现。没有要求你提供具有相同名称或行为的方法。例如,const char* 的 Grid 特化实现了 at() 方法,返回 optional<string>,而不是 optional<const char*>。事实上,你可以编写一个与原始类完全不相关的完全不同的类。当然,这会滥用模板特化功能,如果没有充分理由,你不应该这样做。

下面是 const char* 特化的方法实现。与模板定义中不同,你不需要在每个方法定义前重复 template<> 语法。

Grid<const char*>::Grid(size_t width, size_t height)
    : m_width { width }, m_height { height } {
    m_cells.resize(m_width);
    for (auto& column : m_cells) {
        column.resize(m_height);
    }
}

void Grid<const char*>::verifyCoordinate(size_t x, size_t y) const {
    if (x >= m_width) {
        throw std::out_of_range { std::format("{} must be less than {}.", x, m_width) };
    }
    if (y >= m_height) {
        throw std::out_of_range { std::format("{} must be less than {}.", y, m_height) };
    }
}

const std::optional<std::string>& Grid<const char*>::at(size_t x, size_t y) const {
    verifyCoordinate(x, y);
    return m_cells[x][y];
}

std::optional<std::string>& Grid<const char*>::at(size_t x, size_t y) {
    return const_cast<std::optional<std::string>&>(std::as_const(*this).at(x, y));
}

二、从类模板派生

派生自类模板

您可以从类模板继承。如果派生类从模板本身继承,它也必须是一个模板。另外,您可以从类模板的特定实例继承,在这种情况下,您的派生类不需要是一个模板。

作为前者的一个例子,假设您决定通用的 Grid 类没有提供足够的功能来用作游戏棋盘。具体来说,您希望为游戏棋盘添加一个 move() 方法,将棋子从棋盘上的一个位置移动到另一个位置。以下是 GameBoard 模板的类定义:

import grid;

export template <typename T>
class GameBoard : public Grid<T> {
public:
    explicit GameBoard(size_t width = Grid<T>::DefaultWidth, size_t height = Grid<T>::DefaultHeight);
    void move(size_t xSrc, size_t ySrc, size_t xDest, size_t yDest);
};

这个 GameBoard 模板派生自 Grid 模板,从而继承了所有其功能。您不需要重写 at()、getHeight() 或任何其他方法。您也不需要添加拷贝构造函数、operator= 或析构函数,因为您在 GameBoard 中没有任何动态分配的内存。继承语法看起来很正常,除了基类是 Grid<T>,而不是 Grid。这种语法的原因是 GameBoard 模板并不真正从通用的 Grid 模板派生。相反,GameBoard 模板的每个实例化都派生自相同类型的 Grid 实例化。

例如,如果您使用 ChessPiece 类型实例化一个 GameBoard,那么编译器也会为 Grid<ChessPiece> 生成代码。: public Grid<T> 语法表示这个类继承自对于 T 类型参数有意义的任何 Grid 实例化。请注意,尽管一些编译器不强制执行,但 C++ 名称查找规则要求您使用 this 指针或 Grid<T>:: 来引用基类模板中的数据成员和方法。因此,我们使用 Grid<T>::DefaultWidth 而不是仅仅使用 DefaultWidth。以下是构造函数和 move() 方法的实现:

template <typename T>
GameBoard<T>::GameBoard(size_t width, size_t height) : Grid<T> { width, height } { }

template <typename T>
void GameBoard<T>::move(size_t xSrc, size_t ySrc, size_t xDest, size_t yDest) {
    Grid<T>::at(xDest, yDest) = std::move(Grid<T>::at(xSrc, ySrc));
    Grid<T>::at(xSrc, ySrc).reset(); // 重置源单元
    // 或者:
    // this->at(xDest, yDest) = std::move(this->at(xSrc, ySrc));
    // this->at(xSrc, ySrc).reset();
}

您可以按以下方式使用 GameBoard 模板:

GameBoard<ChessPiece> chessboard { 8, 8 };
ChessPiece pawn;
chessBoard.at(0, 0) = pawn;
chessBoard.move(0, 0, 0, 1);

注意:当然,如果您想重写 Grid 中的方法,您必须在 Grid 类模板中将它们标记为虚拟的。

三、继承与特化

特性

继承

特化

代码复用?

是:派生类包含所有基类的数据成员和方法。

否:您必须在特化中重写所有所需代码。

名称复用?

否:派生类名称必须与基类名称不同。

是:特化必须与原始模板具有相同的名称。

支持多态性?

是:派生类的对象可以代替基类的对象。

否:每个类型的模板实例化都是不同的类型。

注意:使用继承来扩展实现和实现多态性。使用特化来为特定类型定制实现。



Tags:C++   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,不构成投资建议。投资者据此操作,风险自担。如有任何标注错误或版权侵犯请与我们联系,我们将及时更正、删除。
▌相关推荐
C++常见避坑指南
C++ 从入门到放弃?本文主要总结了在C++开发或review过程中常见易出错点做了归纳总结,希望借此能增进大家对C++的了解,减少编程出错,提升工作效率,也可以作为C++开发的避坑攻略。...【详细内容】
2024-04-03  Search: C++  点击:(4)  评论:(0)  加入收藏
C++ 之父反驳白宫警告:自诞生第一天起,C++ 的目标就一直是提高安全性
整理 | 郑丽媛上个月,美国白宫国家网络主任办公室(ONCD)在一份主题为《回到基础构件:通往安全软件之路》的 19 页 PDF 报告中,呼吁开发人员停止使用容易出现内存安全漏洞的编程语...【详细内容】
2024-03-25  Search: C++  点击:(4)  评论:(0)  加入收藏
八个 C++ 开源项目,帮助初学者进阶成长
通过参与或阅读开源项目的源代码,你可以获得丰富的实践机会。实际的项目代码比简单的教程更具挑战性,可以帮助你深入理解 C++ 的各种概念和技术。1.ThreadPool一个简单的 C++1...【详细内容】
2024-03-22  Search: C++  点击:(21)  评论:(0)  加入收藏
C++多线程编程:解锁性能与并发的奥秘
今天我们将深入探讨C++中的多线程编程,揭示多线程如何解锁性能潜力,提高程序的并发性能。什么是多线程?在计算机科学中,多线程是指一个进程(程序的执行实例)中的多个线程同时执行...【详细内容】
2024-02-03  Search: C++  点击:(68)  评论:(0)  加入收藏
C++代码优化攻略
今天我们将深入探讨C++性能优化的世界。在当今软件开发的浪潮中,高性能的代码是必不可少的。无论是开发桌面应用、移动应用,还是嵌入式系统,性能都是关键。1. 选择合适的数据结...【详细内容】
2024-01-26  Search: C++  点击:(112)  评论:(0)  加入收藏
C++质数检测器的设计与实现​
质数,作为数学中的一个基本概念,一直以其独特的性质吸引着众多研究者和爱好者。质数是指大于1的自然数中,除了1和它本身以外不再有其他因数的数。在实际应用中,质数检测也扮演着...【详细内容】
2024-01-15  Search: C++  点击:(110)  评论:(0)  加入收藏
指针变量在C/C++中的内存占用
在编程领域,尤其是C和C++这类底层语言中,指针是一个核心概念,它允许程序直接操作内存地址。然而,关于指针本身在内存中占用的空间大小,却常常让初学者感到困惑。本文将深入探讨这...【详细内容】
2024-01-09  Search: C++  点击:(94)  评论:(0)  加入收藏
C++的面向对象编程:深入解析与理解
当我们谈论C++时,面向对象编程(OOP)是一个无法回避的话题。那么,C++的面向对象究竟是什么?为什么它如此重要?本文将从基本概念到实际应用,为您详细解析C++中的面向对象编程。一、面...【详细内容】
2024-01-03  Search: C++  点击:(95)  评论:(0)  加入收藏
有什么好用的C/C++源代码混淆工具?
开始使用ipaguard前言iOS加固保护是直接针对ios ipa二进制文件的保护技术,可以对iOS APP中的可执行文件进行深度混淆、加密。使用任何工具都无法逆向、破解还原源文件。对APP...【详细内容】
2023-12-29  Search: C++  点击:(117)  评论:(0)  加入收藏
C++中new与malloc:内存分配机制深度解析
本文旨在深入探讨C++中new和malloc两种内存分配机制的区别。通过对比它们在内存分配、初始化、错误处理、调用构造函数/析构函数、类型转换和使用便捷性等方面的不同,我们将...【详细内容】
2023-12-27  Search: C++  点击:(126)  评论:(0)  加入收藏
▌简易百科推荐
C++常见避坑指南
C++ 从入门到放弃?本文主要总结了在C++开发或review过程中常见易出错点做了归纳总结,希望借此能增进大家对C++的了解,减少编程出错,提升工作效率,也可以作为C++开发的避坑攻略。...【详细内容】
2024-04-03  腾讯技术工程    Tags:C++   点击:(4)  评论:(0)  加入收藏
C++ 之父反驳白宫警告:自诞生第一天起,C++ 的目标就一直是提高安全性
整理 | 郑丽媛上个月,美国白宫国家网络主任办公室(ONCD)在一份主题为《回到基础构件:通往安全软件之路》的 19 页 PDF 报告中,呼吁开发人员停止使用容易出现内存安全漏洞的编程语...【详细内容】
2024-03-25    CSDN  Tags:C++   点击:(4)  评论:(0)  加入收藏
八个 C++ 开源项目,帮助初学者进阶成长
通过参与或阅读开源项目的源代码,你可以获得丰富的实践机会。实际的项目代码比简单的教程更具挑战性,可以帮助你深入理解 C++ 的各种概念和技术。1.ThreadPool一个简单的 C++1...【详细内容】
2024-03-22  AI让生活更美好  微信公众号  Tags:C++   点击:(21)  评论:(0)  加入收藏
C# 中15个值得收藏的开源项目推荐
在开源的世界里,C# 编程语言也占有一席之地。这些开源项目涵盖了多个领域,从框架、库到工具,它们为C#开发者提供了丰富的资源和工具,帮助他们更高效地开发、测试和部署应用程序...【详细内容】
2024-03-20  程序员编程日记  微信公众号  Tags:C#   点击:(29)  评论:(0)  加入收藏
C#异步编程:Task.Run vs. async-await,掌握基础与高级用法
概述:C#中的异步编程有两主要方式:Task.Run用于在后台线程执行同步操作,而async-await更适用于清晰表达异步流程。基础用法展示了它们的简单应用,高级用法则演示了它们的结合使...【详细内容】
2024-03-09  架构师老卢  今日头条  Tags:C#   点击:(22)  评论:(0)  加入收藏
C++多线程编程:解锁性能与并发的奥秘
今天我们将深入探讨C++中的多线程编程,揭示多线程如何解锁性能潜力,提高程序的并发性能。什么是多线程?在计算机科学中,多线程是指一个进程(程序的执行实例)中的多个线程同时执行...【详细内容】
2024-02-03     AI让生活更美好  Tags:C++   点击:(68)  评论:(0)  加入收藏
C++代码优化攻略
今天我们将深入探讨C++性能优化的世界。在当今软件开发的浪潮中,高性能的代码是必不可少的。无论是开发桌面应用、移动应用,还是嵌入式系统,性能都是关键。1. 选择合适的数据结...【详细内容】
2024-01-26  AI让生活更美好  微信公众号  Tags:C++   点击:(112)  评论:(0)  加入收藏
C# 线程本地存储为什么线程间值不一样
为什么用 ThreadStatic 标记的字段,只有第一个线程拿到了初始值,其他线程都是默认值,让我能不能帮他解答一下,尼玛,我也不是神仙什么都懂,既然问了,那我试着帮他解答一下,也给后面类...【详细内容】
2024-01-26  一线码农聊技术  微信公众号  Tags:C#   点击:(66)  评论:(0)  加入收藏
C++质数检测器的设计与实现​
质数,作为数学中的一个基本概念,一直以其独特的性质吸引着众多研究者和爱好者。质数是指大于1的自然数中,除了1和它本身以外不再有其他因数的数。在实际应用中,质数检测也扮演着...【详细内容】
2024-01-15  鲨鱼编程  微信公众号  Tags:C++   点击:(110)  评论:(0)  加入收藏
C# 登顶!超越Java或非空想
整理丨诺亚出品 | 51CTO技术栈(微信号:blog51cto)近日,TIOBE编程社区公布年度编程语言,此次摘得这一桂冠的是C#。这也是C#在TIOBE二十多年评选历史中首次赢得这一年度大奖。C#虽...【详细内容】
2024-01-15    51CTO  Tags:C#   点击:(112)  评论:(0)  加入收藏
站内最新
站内热门
站内头条