当前位置: 首页 > news >正文

跟我学c++中级篇——模板的继承

一、继承

面向对象编程有三个特点:封装、继承和多态。其中继承在其中起着承上启下的作用。一般来说,继承现在和组合的应用比较难区分,出于各种场景和目的,往往各有千秋。但目前主流的观点,一般是如果没有特殊情况,比如需要处理数据的访问权限、重新实现功能(多态),都是推荐使用组合的。继承一个是复杂另外一个不灵活。人类更擅长的是组件拼装,而不是重新造组件。同时,组合更适合开发设计原则,比如开闭原则、单一职责等等,好像只有一个里氏替换不合适。
但是在有些时候儿,继承还是必须要用的,那么今天重点谈一下,在模板中如何使用继承。

二、模板继承及方式

模板的继承一如普通类的继承,形式是基本相同的,可能有些实现上看上去比较复杂。在模板开发的继承中一般有四种形式的继承方式(以父子类来描述继承,不指明的为普通类)即:
1、父类为模板
类似于:

template <typename T> 
class Base {
};
class Derived:public Base<int>{
};

这里面有一个前文提到的CRTP模式:

template <class T>
class Base
{// 基类方法可以通过模板继承的方式来访问继承类的成员
};
class Derived : public Base<Derived>
{
};

2、子类为模板

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

3、父子类均为模板
此种情况又有两种形式:

//第一种实例化父类模板参数
template <typename T>
class Base {
};
template <typename T> 
class Derived:public Base<int>{
};
//第二种非实例化父类模板参数
class Base {
};
template <typename T> 
class Derived:public Base<T>{
};

4、父类的模板参数被继承

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

此处没有谈多继承,这个在编程时尽量避免。有兴趣可以自己搞一下。在父子类继承中,如果父类是模板,会出现在子类中访问父类变量无法直接访问的现象。需要使用父类的类名限制( Base ::basemember_;)或者直接在变量前加this进行访问。
原因是:继承的模板要进行两次编译,每一次只处理和本身的数据相关,也就是说只管自己的地盘,什么 父类模板参数啥的都暂时忽略;第二步,再处理上面没有处理的模板参数部分。所以此时直接访问父类继承过来的变量和函数会找不到报错,重要的要使用某种方式把这部分延期到第二步编译,那么就没有什么 问题了。方法就是上面的两种方式,这样编译器就明白这些不是本身模板的内容就放到第二步处理。
另外一个选择继承这里也没有说明,什么 意思呢?就是模板参数有N个,但父类模板只用其中的<N个,比如有子类有三个模板参数,父类只用了其中两个或者一个,其它的为子类自用。这种其实上面的组合方式,不再赘述,但是下面会给出例子。

三、例程:

下面看一个例程:

#include <iostream>
namespace TT {//========================父类为模板=========================================
template <typename T> class BaseT {
public:BaseT() = default;BaseT(int a) : d_(a) {std::cout << "call template Parent class BaseT" << std::endl;}~BaseT() {}protected:int d_ = 0;
};
class Derived : public BaseT<int> {
public:// Derived() = default;Derived(int a) : BaseT<int>(a) {std::cout << "call derived sub class no template!" << std::endl;std::cout << " Parent class Base d_:" << d_ << std::endl;}Derived() : BaseT<int>(10) {std::cout << "call derived sub class no template 10!" << std::endl;std::cout << " Parent class Base d_:" << d_ << std::endl;}~Derived() {}
};
//=========================子类为模板========================================
class Base {
public:Base() = default;Base(int a) : d_(a) { std::cout << "call  Parent class Base" << std::endl; }~Base() {}protected:int d_ = 0;
};
template <typename T> class DerivedT : public Base {
public:DerivedT(){};DerivedT(int a) : Base(a) {std::cout << "sub template class call Parent class Base" << std::endl;std::cout << " Parent class Base d_:" << d_ << std::endl;}~DerivedT() {}
};
//=======================父子均为模板但父类特化===============================
template <typename TT> class DerivedTTa : public BaseT<int> {
public:DerivedTTa() {std::cout << "sub a template class call init Parent template class BaseT"<< std::endl;}DerivedTTa(int a) : BaseT(a), d_(a) {std::cout << "sub a template class call init Parent template class BaseT"<< std::endl;std::cout << " Parent class Base d_:" << d_ << std::endl;}~DerivedTTa() {}protected:TT d_ = 0;
};
//=====================父子均为模板但父类未特化===============================
template <typename TT> class DerivedTTb : public BaseT<TT> {
public:DerivedTTb() {}DerivedTTb(TT a) : BaseT<TT>(a) {std::cout << "sub b template class call Parent template class BaseT"<< std::endl;std::cout << " Parent class BaseT d_:" << this->d_ << " " << BaseT<TT>::d_<< std::endl;}~DerivedTTb() {}
};
//=====================继承模板参数========================================
template <typename T> class DerivedP : public T {
public:DerivedP() {std::cout << "template class inherit template class Parameter" << std::endl;}DerivedP(int a) : T(a) {std::cout << "template class inherit template class Parameter" << std::endl;std::cout << "parameter a is:" << a << std::endl;}~DerivedP() {}
};
//====================选择继承============================================
template <typename T, typename N, typename P>
class DerivedPM : public BaseT<N> {
public:DerivedPM() {std::cout << "template class call template Mult class Parameter"<< std::endl;};DerivedPM(T t, N n, P p) : BaseT<N>(n), t_(t), n_(n), p_(p) {std::cout << "template class call template Mult class Parameter"<< std::endl;std::cout << "parameter t,n,p is:" << t << " " << n << " " << p<< std::endl;std::cout << "parameter d_ is:" << this->d_ << std::endl;}protected:T t_;N n_;P p_;
};
} int main() {TT::Derived td;TT::Derived td1(10);TT::Derived dd;TT::DerivedT<int> tdt;TT::DerivedT<int> tdt1(10);TT::DerivedTTa<int> tdta;TT::DerivedTTa<int> tdta1(10);TT::DerivedTTb<int> tdtb;TT::DerivedTTb<int> tdtb1(10);TT::DerivedP<TT::Base> tdp;TT::DerivedP<TT::BaseT<TT::Base>> tdpbb;TT::DerivedPM<int, int, int> tddpm;TT::DerivedPM<int, int, int> tddpm1(1, 2, 3);return 0;
}

例程不难,关键看模板继承模板中那个父类d_如何访问,有的为什么要加this(或Base:😃,有的为什么不加,模板一但特化,便和普通类一致,不需要this。仔细想一下,就会越来越明白。那个CRTP的在前文有专门的分析说明,此处不再举例。有兴趣可以 翻一下前面的例程

四、总结

模板不管用得多少 ,总会遇到一些很古怪的错误,但这些错误其实是编译器不友好,或者说模板比较复杂导致编译无法特别准确的提供一些信息造成的。所以重点还是要把模板的基础知识学清楚,一点点的推导出来,再配合相关的错误信息,就可以较快的解决问题。
网上还有先写非模板代码然后 再用模板代码改写的。这样也是一个不错的方法,虽然有点费事儿,但只要用在复杂的情况下就好了。重点在于不断的反复的把模板的知识对照着代码编写,熟悉后自然就渐渐明白道理,再写就逐渐顺手多了。

相关文章:

跟我学c++中级篇——模板的继承

一、继承 面向对象编程有三个特点&#xff1a;封装、继承和多态。其中继承在其中起着承上启下的作用。一般来说&#xff0c;继承现在和组合的应用比较难区分&#xff0c;出于各种场景和目的&#xff0c;往往各有千秋。但目前主流的观点&#xff0c;一般是如果没有特殊情况&…...

需求分析案例:消息配置中心

本文介绍了一个很常见的消息推送需求&#xff0c;在系统需要短信、微信、邮件之类的消息推送时&#xff0c;边界如何划分和如何设计技术方案。 1、需求 一个系统&#xff0c;一般会区分多个业务模块&#xff0c;并拆分成不同的业务系统&#xff0c;例如一个商城的架构如下&am…...

自动化测试——环境

一、搭建环境 1、安装Slenium pip install selenium 2、安装浏览器驱动-》查询浏览器版本-》下载对应版本驱动-》在path路径中配置&#xff08;浏览器更新需要重新下载&#xff09; pip install webdriver -helper&#xff08;自动化&#xff09;python3.9以上 pip install 安…...

短视频矩阵营销系统技术开发者开发笔记分享

一、开发短视频seo抖音矩阵系统需要遵循以下步骤&#xff1a; 1. 确定系统需求&#xff1a;根据客户的需求&#xff0c;确定系统的功能和特点&#xff0c;例如用户注册登录、视频上传、视频浏览、评论点赞等。 2. 设计系统架构&#xff1a;根据系统需求&#xff0c;设计系统的…...

vue2和vue3引用ueditor的区别

官方文档入口 UEditor Docs vue2使用方式 UE.vue组件 <template><div><script id"editor" type"text/plain"></script><Upload v-if"isupload" :config"{total:9}" :isupload"isupload" ret…...

【每日运维】RockyLinux8非容器化安装Mysql、Redis、RabitMQ单机环境

系统版本&#xff1a;RockyLinux 8.6 安装方式&#xff1a;非容器化单机部署 安装版本&#xff1a;mysql 8.0.32 redis 6.2.11 rabbitmq 3.11.11 elasticsearch 6.7.1 前置条件&#xff1a;时间同步、关闭selinux、主机名、主机解析host 环境说明&#xff1a;PC电脑VMware Work…...

第一次后端复习整理(JVM、Redis、反射)

1. JVM 文章仅为自身笔记 详情查看一篇文章掌握整个JVM&#xff0c;JVM超详细解析&#xff01;&#xff01;&#xff01; 1.1 什么是JVM jvm是Java虚拟机 1.2 Java文件的编译过程 程序员编写代码形成.java文件经过javac编译成.class文件再通过JVM的类加载器进入运行时数据…...

python的web学习(一)-初识django

文章目录 软件创建项目默认项目文件说明App的概念(应用)apps.py编写URL和视图函数对应关系【urls.py】编写视图函数【views.py】启动服务 软件 python下载 django下载 创建项目 django-admin startproject 文件名默认项目文件说明 项目名 manage.py(项目管理&#xff0c;启…...

JavaWeb+jsp+Tomcat的叮当书城项目

点击以下链接获取源码&#xff1a; https://download.csdn.net/download/qq_64505944/88123111?spm1001.2014.3001.5503 技术&#xff1a;ssm jsp JDK1.8 MySQL5.7 Tomcat8.3 源码数据库课程设计 功能&#xff1a;管理员与普通用户和超级管理员三个角色&#xff0c;管理员可…...

【嵌入式Linux系统开发】——系统移植概述

目录 &#x1f349;&#x1f349;一、什么是嵌入式系统 &#x1f349;&#x1f349;二、嵌入式系统操作 &#x1f349;&#x1f349;三、嵌入式Linux的特点 &#x1f349;&#x1f349;四、嵌入式系统的组成 1、硬件和软件 2、硬件层 3、中间层 4、软件层 5、 功能层与执…...

升讯威在线客服系统是如何实现对 IE8 完全完美支持的(怎样从 WebSocket 降级到 Http)【干货】

简介 升讯威在线客服与营销系统是基于 .net core / WPF 开发的一款在线客服软件&#xff0c;宗旨是&#xff1a; 开放、开源、共享。努力打造 .net 社区的一款优秀开源产品。 完整私有化包下载地址 &#x1f4be; https://kf.shengxunwei.com/freesite.zip 当前版本信息 发布…...

用VMware给运行在VMware上的CentOS7生成一个以SSH方式连接VMware上的CentOS7的运行在Windows上的命令行窗口

2023年7月27日&#xff0c;周四早上 目录 一个发现生成方法如果上面的方法连接失败&#xff0c;就采取这个方法 一个发现 今天早上无意间发现VMware可以生成一个以SSH方式连接着CentOS7的Windows命令行窗口&#xff0c; 这样做可以带来一定的便利性 &#xff1a; 方便复制、…...

C语言基础-3

1、函数 函数是C语言代码的基本组成部分&#xff0c;它是一个小的模块&#xff0c;整个程序由很多个功能独立的模块&#xff08;函数&#xff09;组成。这就是程序设计的基本分化方法。 main&#xff1a;C语言中所谓的主函数&#xff0c;主函数就是一种特别的函数。特别之处在于…...

Python 编程规范进阶(1) | 命名规范

养成良好的开发、编程习惯 跟着google开源项目走 https://github.com/google/styleguide 近期Target&#xff1a; 命名规范&#xff1b; Pythonic 积累 按照需求写需要的API&#xff1b; 写前先动脑子&#xff0c;比如画流程图&#xff0c;测试接口&#xff1b; Google 推荐的P…...

算法----二叉搜索树中第K小的元素

题目 二叉搜索树中第K小的元素 给定一个二叉搜索树的根节点 root &#xff0c;和一个整数 k &#xff0c;请你设计一个算法查找其中第 k 个最小元素&#xff08;从 1 开始计数&#xff09;。 示例 1&#xff1a; 输入&#xff1a;root [3,1,4,null,2], k 1 输出&#xff…...

阿里Java开发手册~安全规约

1. 【强制】隶属于用户个人的页面或者功能必须进行权限控制校验。 说明&#xff1a; 防止没有做水平权限校验就可随意访问、修改、删除别人的数据&#xff0c;比如查看他人的私信 内容、修改他人的订单。 2. 【强制】用户敏感数据禁止直接展示&#xff0c;必须对展示数据进…...

消息中间件RabbitMQ——学习笔记

❤ 作者主页&#xff1a;欢迎来到我的技术博客&#x1f60e; ❀ 个人介绍&#xff1a;大家好&#xff0c;本人热衷于Java后端开发&#xff0c;欢迎来交流学习哦&#xff01;(&#xffe3;▽&#xffe3;)~* &#x1f34a; 如果文章对您有帮助&#xff0c;记得关注、点赞、收藏、…...

爬虫005_python类型转换_其他类型转换为整型_转换为Float类型_转换为字符串_转换为布尔值---python工作笔记023

首先来看,字符串转换成int 很简单 float转换成int 会把小数点后面的内容丢掉 boolean转换为int true是1 false 是0 然后字符串转换为int,要注意 不能有特殊字符比如1.23 中有点 就报错 上面字符串12ab,有ab也报错 看上面...

SpringBoot复习:(5)使用PropertySource注解

一、自定义的一个配置文件 age33 nameliu二、实体类 package com.example.demo.domain;public class Student {private String name;private int age;public String getName() {return name;}public void setName(String name) {this.name name;}public int getAge() {retur…...

webrtc 支持H265(三) 总结

文章目录 web端的解码及渲染的实现应用场景单向视频流的场景datachannel通道的稳定性解码性能 双向视频流的场景有音频流的场景 web端的解码及渲染的实现 在前面的文章中介绍了ZLMediaKit的修改方法&#xff0c;在web端的播放器可以参照这个实现&#xff0c;基于wasm H265播放…...

生成xcframework

打包 XCFramework 的方法 XCFramework 是苹果推出的一种多平台二进制分发格式&#xff0c;可以包含多个架构和平台的代码。打包 XCFramework 通常用于分发库或框架。 使用 Xcode 命令行工具打包 通过 xcodebuild 命令可以打包 XCFramework。确保项目已经配置好需要支持的平台…...

Objective-C常用命名规范总结

【OC】常用命名规范总结 文章目录 【OC】常用命名规范总结1.类名&#xff08;Class Name)2.协议名&#xff08;Protocol Name)3.方法名&#xff08;Method Name)4.属性名&#xff08;Property Name&#xff09;5.局部变量/实例变量&#xff08;Local / Instance Variables&…...

linux 下常用变更-8

1、删除普通用户 查询用户初始UID和GIDls -l /home/ ###家目录中查看UID cat /etc/group ###此文件查看GID删除用户1.编辑文件 /etc/passwd 找到对应的行&#xff0c;YW343:x:0:0::/home/YW343:/bin/bash 2.将标红的位置修改为用户对应初始UID和GID&#xff1a; YW3…...

《基于Apache Flink的流处理》笔记

思维导图 1-3 章 4-7章 8-11 章 参考资料 源码&#xff1a; https://github.com/streaming-with-flink 博客 https://flink.apache.org/bloghttps://www.ververica.com/blog 聚会及会议 https://flink-forward.orghttps://www.meetup.com/topics/apache-flink https://n…...

算法岗面试经验分享-大模型篇

文章目录 A 基础语言模型A.1 TransformerA.2 Bert B 大语言模型结构B.1 GPTB.2 LLamaB.3 ChatGLMB.4 Qwen C 大语言模型微调C.1 Fine-tuningC.2 Adapter-tuningC.3 Prefix-tuningC.4 P-tuningC.5 LoRA A 基础语言模型 A.1 Transformer &#xff08;1&#xff09;资源 论文&a…...

【C++特殊工具与技术】优化内存分配(一):C++中的内存分配

目录 一、C 内存的基本概念​ 1.1 内存的物理与逻辑结构​ 1.2 C 程序的内存区域划分​ 二、栈内存分配​ 2.1 栈内存的特点​ 2.2 栈内存分配示例​ 三、堆内存分配​ 3.1 new和delete操作符​ 4.2 内存泄漏与悬空指针问题​ 4.3 new和delete的重载​ 四、智能指针…...

Go语言多线程问题

打印零与奇偶数&#xff08;leetcode 1116&#xff09; 方法1&#xff1a;使用互斥锁和条件变量 package mainimport ("fmt""sync" )type ZeroEvenOdd struct {n intzeroMutex sync.MutexevenMutex sync.MutexoddMutex sync.Mutexcurrent int…...

LabVIEW双光子成像系统技术

双光子成像技术的核心特性 双光子成像通过双低能量光子协同激发机制&#xff0c;展现出显著的技术优势&#xff1a; 深层组织穿透能力&#xff1a;适用于活体组织深度成像 高分辨率观测性能&#xff1a;满足微观结构的精细研究需求 低光毒性特点&#xff1a;减少对样本的损伤…...

【WebSocket】SpringBoot项目中使用WebSocket

1. 导入坐标 如果springboot父工程没有加入websocket的起步依赖&#xff0c;添加它的坐标的时候需要带上版本号。 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-websocket</artifactId> </dep…...

Matlab实现任意伪彩色图像可视化显示

Matlab实现任意伪彩色图像可视化显示 1、灰度原始图像2、RGB彩色原始图像 在科研研究中&#xff0c;如何展示好看的实验结果图像非常重要&#xff01;&#xff01;&#xff01; 1、灰度原始图像 灰度图像每个像素点只有一个数值&#xff0c;代表该点的​​亮度&#xff08;或…...