当前位置:首页 > 科技  > 软件

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

来源: 责编: 时间:2023-12-20 17:46:21 314观看
导读一、类模板特化1.特化的实现你可以为特定类型提供类模板的替代实现。例如,你可能认为 const char 类型(C 风格字符串)的 Grid 行为没有意义。Grid<const char> 将在 vector<vector<optional<const char*>>> 中存储其元素

一、类模板特化

1.特化的实现

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

JYR28资讯网——每日最新资讯28at.com

模板的替代实现称为模板特化。你可能会发现其语法初看有些奇怪。当你编写类模板特化时,你必须指定这是模板,并且你正在为特定类型编写模板的版本。以下是 Grid 的 const charJYR28资讯网——每日最新资讯28at.com

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* 和字符串。此时一个明显的问题是,为什么这个类仍然是模板。即,以下语法有什么用途?JYR28资讯网——每日最新资讯28at.com

template <> class Grid<const char*>

这种语法告诉编译器,这个类是 Grid 类的 const char* 特化。假设你没有使用这种语法,而是尝试编写如下代码:JYR28资讯网——每日最新资讯28at.com

class Grid

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

2.主模块接口文件

主模块接口文件简单地导入并导出两个模块接口分区:JYR28资讯网——每日最新资讯28at.com

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

特化可以按照以下方式进行测试:JYR28资讯网——每日最新资讯28at.com

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*>。事实上,你可以编写一个与原始类完全不相关的完全不同的类。当然,这会滥用模板特化功能,如果没有充分理由,你不应该这样做。JYR28资讯网——每日最新资讯28at.com

下面是 const char* 特化的方法实现。与模板定义中不同,你不需要在每个方法定义前重复 template<> 语法。JYR28资讯网——每日最新资讯28at.com

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));}

二、从类模板派生

派生自类模板

您可以从类模板继承。如果派生类从模板本身继承,它也必须是一个模板。另外,您可以从类模板的特定实例继承,在这种情况下,您的派生类不需要是一个模板。JYR28资讯网——每日最新资讯28at.com

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

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 实例化。JYR28资讯网——每日最新资讯28at.com

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

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 模板:JYR28资讯网——每日最新资讯28at.com

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

注意:当然,如果您想重写 Grid 中的方法,您必须在 Grid 类模板中将它们标记为虚拟的。JYR28资讯网——每日最新资讯28at.com

JYR28资讯网——每日最新资讯28at.com

三、继承与特化

特性
JYR28资讯网——每日最新资讯28at.com

继承
JYR28资讯网——每日最新资讯28at.com

特化
JYR28资讯网——每日最新资讯28at.com

代码复用?
JYR28资讯网——每日最新资讯28at.com

是:派生类包含所有基类的数据成员和方法。
JYR28资讯网——每日最新资讯28at.com

否:您必须在特化中重写所有所需代码。
JYR28资讯网——每日最新资讯28at.com

名称复用?
JYR28资讯网——每日最新资讯28at.com

否:派生类名称必须与基类名称不同。
JYR28资讯网——每日最新资讯28at.com

是:特化必须与原始模板具有相同的名称。
JYR28资讯网——每日最新资讯28at.com

支持多态性?
JYR28资讯网——每日最新资讯28at.com

是:派生类的对象可以代替基类的对象。
JYR28资讯网——每日最新资讯28at.com

否:每个类型的模板实例化都是不同的类型。
JYR28资讯网——每日最新资讯28at.com

注意:使用继承来扩展实现和实现多态性。使用特化来为特定类型定制实现。JYR28资讯网——每日最新资讯28at.com

JYR28资讯网——每日最新资讯28at.com

本文链接:http://www.28at.com/showinfo-26-50748-0.htmlC++类模板特化与继承使用说明书,新手也能get

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: 微服务集成中的三个常见缺陷,以及如何避免它们

下一篇: 尤雨溪:Vue 3 开发中的经验和教训

标签:
  • 热门焦点
  • 小米平板5 Pro 12.4简评:多专多能 兼顾影音娱乐的大屏利器

    疫情带来了网课,网课盘活了安卓平板,安卓平板市场虽然中途停滞了几年,但好的一点就是停滞的这几年行业又有了新的发展方向,例如超窄边框、高刷新率、多摄镜头组合等,这就让安卓
  • vivo TWS Air开箱体验:真轻 臻好听

    在vivo S15系列新机的发布会上,vivo的最新款真无线蓝牙耳机vivo TWS Air也一同发布,本次就这款耳机新品给大家带来一个简单的分享。外包装盒上,vivo TWS Air保持了vivo自家产
  • 服务存储设计模式:Cache-Aside模式

    Cache-Aside模式一种常用的缓存方式,通常是把数据从主存储加载到KV缓存中,加速后续的访问。在存在重复度的场景,Cache-Aside可以提升服务性能,降低底层存储的压力,缺点是缓存和底
  • 之家push系统迭代之路

    前言在这个信息爆炸的互联网时代,能够及时准确获取信息是当今社会要解决的关键问题之一。随着之家用户体量和内容规模的不断增大,传统的靠"主动拉"获取信息的方式已不能满足用
  • 得物宠物生意「狂飙」,发力“它经济”

    作者|花花小萌主近日,得物宣布正式上线宠物鉴别,通过得物App内的&ldquo;在线鉴别&rdquo;,可找到鉴别宠物的选项。通过上传自家宠物的部位细节,就能收获拥有专业资质认证的得物鉴
  • 消费结构调整丨巨头低价博弈,拼多多还卷得动吗?

    来源:征探财经作者:陈香羽随着流量红利的退潮,电商的存量博弈越来越明显。曾经主攻中高端与品质的淘宝天猫、京东重拾&ldquo;低价&rdquo;口号。而过去与他们错位竞争的拼多多,靠
  • 苹果、三星、惠普等暂停向印度出口笔记本和平板电脑

    集微网消息,据彭博社报道,在8月3日印度突然禁止在没有许可证的情况下向印度进口电脑/平板及显示器等产品后,苹果、三星电子和惠普等大公司暂停向印度
  • 三星电子Q2营收60万亿韩元 存储业务营收同比仍下滑超过50%

    7月27日消息,据外媒报道,从三星电子所发布的财报来看,他们主要利润来源的存储芯片业务在今年二季度仍不乐观,营收同比仍在大幅下滑,所在的设备解决方案
  • Windows 11发布,微软一改往常对老机型开放的态度

    距离 Windows 11 发布已经过去一周,在过去一周里,很多数码爱好者围绕其对 Android 应用的支持、对老机型的升级问题展开了激烈讨论。与以往不同的是,在这次大
Top