《C++新经典对象模型》之第7章 模板实例化语义学

《C++新经典对象模型》之第7章 模板实例化语义学

    • 7.1 模板及其实例化详细分析
      • 7.1.1 函数模板
      • 7.1.2 类模板的实例化分析
      • 7.1.3 多个源文件中使用类模板
        • 07.01.cpp
    • 7.2 炫技写法
      • 7.2.1 不能被继承的类
      • 7.2.2 类外调用私有虚成员函数
        • 07.02.cpp

7.1 模板及其实例化详细分析

7.1.1 函数模板

template <class T>
T funcadd(const T &a, const T &b)
{T addhe = a + b;return addhe;
}cout<<funcadd(12, 14)<<endl;
//dumpbin
//nm

编译器编译时根据funcadd的调用来确定函数模板中T的类型。
int funcadd(int const &, int const &);

7.1.2 类模板的实例化分析

template <class T>
struct ATPL
{T m_i, m_j;ATPL(T tmpi = 0, T tmpj = 0) : m_i(tmpi), m_j(tmpj) {}
};
  1. 类模板中的枚举类型
template <class T>
struct ATPL
{enum ECURRSTATUS{E_CS_Busy,E_CS_Free,};T m_i, m_j;ATPL(T tmpi = 0, T tmpj = 0) : m_i(tmpi), m_j(tmpj) {}
};ATPL<int>::ECURRSTATUS myenum = ATPL<int>::E_CS_Busy;ATPL<int>::ECURRSTATUS myenum2 = ATPL<int>::E_CS_Free;ATPL<double>::ECURRSTATUS myenum3 = ATPL<double>::E_CS_Free;
  1. 类模板中的静态成员变量
template <class T>
struct ATPL
{T m_i, m_j;ATPL(T tmpi = 0, T tmpj = 0) : m_i(tmpi), m_j(tmpj) {}static int m_sti;static T m_sti2;
};template <class T>
int ATPL<T>::m_sti = 10;
template <class T>
T ATPL<T>::m_sti2 = 10;//必须要和10兼容的类型ATPL<int>::m_sti = 18;cout << ATPL<int>::m_sti << endl;ATPL<float>::m_sti = 21;cout << ATPL<float>::m_sti << endl;//ATPL<string>::m_sti2;//errorATPL<int>::m_sti2 = 132;cout << ATPL<int>::m_sti2 << endl;ATPL<float>::m_sti2 = 1050.5;cout << ATPL<float>::m_sti2 << endl;
  1. 类模板的实例化
template <class T>
struct ATPL
{T m_i, m_j;ATPL(T tmpi = 0, T tmpj = 0) : m_i(tmpi), m_j(tmpj) {}
};ATPL<int> *pobj = nullptr;//未实例化类const ATPL<int> &yobj = 0; // 因为构造函数允许缺省参数,这里有隐式类型转换//等价于//ATPL<int> tmpobj(0);//const ATPL<int> &yobj = tmpobj;
  1. 成员函数的实例化
template <class T>
struct ATPL
{T m_i, m_j;ATPL(T tmpi = 0, T tmpj = 0) : m_i(tmpi), m_j(tmpj) {}void func1() const { cout << "ATPL::func1()" << endl; }void func2() const { cout << "ATPL::func2()" << endl; }
};const ATPL<int> &yobj = 0;yobj.func1();//成员函数调用时,才会实例化出成员函数

7.1.3 多个源文件中使用类模板

  • mytemplate.h
#pragma once
#include <iostream>
using namespace std;template <class T>
T funadd(const T &a, const T &b)
{T addhe = a + b;return addhe;
}template <class T>
struct ATPL
{enum ECURRSTATUS{E_CS_Busy,E_CS_Free,};T m_i, m_j;ATPL(T tmpi = 0, T tmpj = 0) : m_i(tmpi), m_j(tmpj) {}static int m_sti;static T m_sti2;void func1() const { cout << "ATPL::func1()" << endl; }void func2() const { cout << "ATPL::func2()" << endl; }
};template <class T>
int ATPL<T>::m_sti = 10;template <class T>
T ATPL<T>::m_sti2 = 10;
  • MyProject.cpp
#include "mytemplate.h"void myfunc()
{ATPL<int> myobj;//即使没调用myfunc()函数,也会实例化ATPL<int>类myobj.m_sti2 = 18;cout << myobj.m_sti2 << endl;
}
  • myfunc.cpp
#include "mytemplate.h"int ftest()
{ATPL<int> myobj;//即使没调用myfunc()函数,也会实例化ATPL<int>类myobj.m_sti2 = 21;cout << myobj.m_sti2 << endl;return 15;
}

编译时可能多个.obj文件产生了多个相同的ATPL类,但链接时只保留一个ATPL类实体,其余忽略掉。

  1. 虚函数的实例化
template <class T>
struct ATPL
{enum ECURRSTATUS{E_CS_Busy,E_CS_Free,};T m_i, m_j;ATPL(T tmpi = 0, T tmpj = 0) : m_i(tmpi), m_j(tmpj) {}static int m_sti;static T m_sti2;void func1() const { cout << "ATPL::func1()" << endl; }void func2() const { cout << "ATPL::func2()" << endl; }virtual void virfunc1() { cout << "ATPL::virfunc1()" << endl; }virtual void virfunc2() { cout << "ATPL::virfunc2()" << endl; }
};template <class T>
int ATPL<T>::m_sti = 10;template <class T>
T ATPL<T>::m_sti2 = 10;

类模板存在虚函数时,会实例化所以虚函数,即使未调用。
虚函数,产生虚函数表,虚函数表需要放置各个虚函数的地址,所以实例化出所以虚函数。

  1. 显示实例化
//.cpp文件中
template class ATPL<double>;

类模板的显示实例化,所有内容(类、成员函数、虚函数、静态成员变量等)都实例化出来,无论是否调用。

//.cpp文件中
template void ATPL<double>::func2() const;

实例化单独的成员函数,类并未实例化。

07.01.cpp
#include <cstdio>
#include <iostream>
using namespace std;template <class T>
T funadd(const T &a, const T &b)
{T addhe = a + b;return addhe;
}template <class T>
struct ATPL
{enum ECURRSTATUS{E_CS_Busy,E_CS_Free,};T m_i, m_j;ATPL(T tmpi = 0, T tmpj = 0) : m_i(tmpi), m_j(tmpj) {}static int m_sti;static T m_sti2;void func1() const { cout << "ATPL::func1()" << endl; }void func2() const { cout << "ATPL::func2()" << endl; }virtual void virfunc1() { cout << "ATPL::virfunc1()" << endl; }virtual void virfunc2() { cout << "ATPL::virfunc2()" << endl; }
};template <class T>
int ATPL<T>::m_sti = 10;template <class T>
T ATPL<T>::m_sti2 = 10;void myfunc()
{ATPL<int> myobj;myobj.m_sti2 = 18;cout << myobj.m_sti2 << endl;
}int main()
{cout << funadd(12, 14) << endl;{ATPL<int>::ECURRSTATUS myenum = ATPL<int>::E_CS_Busy;ATPL<int>::ECURRSTATUS myenum2 = ATPL<int>::E_CS_Free;ATPL<double>::ECURRSTATUS myenum3 = ATPL<double>::E_CS_Free;}{ATPL<int>::m_sti = 18;cout << ATPL<int>::m_sti << endl;ATPL<float>::m_sti = 21;cout << ATPL<float>::m_sti << endl;ATPL<int>::m_sti2 = 132;cout << ATPL<int>::m_sti2 << endl;ATPL<float>::m_sti2 = 1050.5;cout << ATPL<float>::m_sti2 << endl;}ATPL<int> *pobj = nullptr;{const ATPL<int> &yobj = 0; // 因为构造函数允许缺省参数,这里有隐式类型转换yobj.func1();}cout << sizeof(ATPL<int>) << endl;cout << "Over!\n";return 0;
}

7.2 炫技写法

7.2.1 不能被继承的类

  1. 使用final关键字
  2. 友元函数+虚继承(友元破坏封装,虚继承消耗大)

class A
{
private:A() {}friend class B; // B可以调用A的私有构造函数
};
class B : virtual public A // 虚继承A
{
public:int m_b;
};
// 虚基类A的构造函数由孙子类C调用
// 但虚基类A的构造函数私有,只有友元类B能调用,孙子类C无法调用
// 无法创建C类,即B类无法被继承
class C : public B
{
public:int m_c;
};{B myobjb;myobjb.m_b = 15;// C myobjc;// myobjc.m_c = 20;}

7.2.2 类外调用私有虚成员函数

class A2
{
private:virtual void virfunc(){myfunc();}void myfunc(){cout << "A::myfunc()" << endl;}
};{A2 aobj;(reinterpret_cast<void (*)()>(**(long ***)(&aobj)))();//通过虚函数表指针调用虚函数//long*** pvptr = (long ***)&aobj;//long** vptr = *pvptr;//typedef void(*Func)();//Func f = (Func)*vptr;//f();}
07.02.cpp
#include <iostream>
using namespace std;class A
{
private:A() {}friend class B; // B可以调用A的私有构造函数
};
class B : virtual public A // 虚继承A
{
public:int m_b;
};
// 虚基类A的构造函数由孙子类C调用
// 但虚基类A的构造函数私有,只有友元类B能调用,孙子类C无法调用
// 无法创建C类,即B类无法被继承
class C : public B
{
public:int m_c;
};class A2
{
private:virtual void virfunc(){myfunc();}void myfunc(){cout << "A::myfunc()" << endl;}
};int main()
{{B myobjb;myobjb.m_b = 15;// C myobjc;// myobjc.m_c = 20;}{A2 aobj;(reinterpret_cast<void (*)()>(**(long ***)(&aobj)))();// 通过虚函数表指针调用虚函数// long*** pvptr = (long ***)&aobj;// long** vptr = *pvptr;// typedef void(*Func)();// Func f = (Func)*vptr;// f();}cout << "Over!\n";return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/728106.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

对于泛型的学习

泛型&#xff1a;是JDK5中引入的新特性&#xff0c;可以在编译阶段约束操作的数据类型&#xff0c;并进行检查。 泛型的格式&#xff1a;<数据类型> 注意&#xff1a;泛型只能支持引用的数据类型 import java.util.ArrayList; import java.util.Iterator;public class…

990-39产品经理:Top 5 Most Common Incident Response Scenarios 五大最常见的事件响应场景

Top 5 Most Common Incident Response Scenarios 五大最常见的事件响应场景 Dealing with a cyber incident can be a daunting experience. Whether you’re targeted by phishing, malicious network scanning, or ransomware, it’s easy to feel overwhelmed. Even if you…

蓝桥集训之日志统计

蓝桥集训之日志统计 核心思想&#xff1a; 双指针 用对组储存每次的时刻和编号双指针遍历对组 每次记录对应编号的赞1当该对组区间长度>d时 将上次记录的赞消掉若最终仍满足赞>k 则标记true #include<iostream>#include <algorithm>#include <cstring&…

Frontend - Boostrap 消息弹窗

目录 一、下载 &#xff08;一&#xff09;中文官网 &#xff08;二&#xff09;bootstrap v3 依赖 jQuery 插件 二、解压并安装 &#xff08;一&#xff09;解压 1. 压缩包解压 2. 简化文件 &#xff08;二&#xff09;安装 三、配置 &#xff08;一&#xff09;bas…

Docker 配置阿里云镜像加速器

一、首先需要创建一个阿里云账号 二、登录阿里云账号 三、进入控制台 四、搜索容器镜像服务&#xff0c;并选择 五、选择镜像工具中的镜像加速 六 、配置镜像源 注意&#xff1a;有/etc/docker文件夹的直接从第二个命令开始

网络安全成全国两会热点话题,网络安全你知多少?

2024全国两会正召开的如火如荼&#xff0c;目前已诞生多个热点话题&#xff0c;比如教育、稳就业、促增收等等&#xff0c;其中网络安全也成全国两会热点话题之一。会上&#xff0c;多位全国政协委员、全国人大代表、行业专家、学者、企业大咖纷纷携网络安全相关提案、议案上会…

Sleuth(Micrometer)+ZipKin分布式链路追踪

Sleuth(Micrometer)ZipKin分布式链路追踪 Micrometer springboot3之前还可以用sleuth&#xff0c;springboot3之后就被Micrometer所替代 官网https://github.com/spring-cloud/spring-cloud-sleuth 为什么会出现这个技术&#xff1f; 在微服务框架中&#xff0c;一个由客户…

Vue中如何处理用户权限?

在前端开发中&#xff0c;处理用户权限是非常重要的一个方面。Vue作为一种流行的前端框架&#xff0c;提供了很多便捷的方式来管理用户权限。本文将介绍一些Vue中处理用户权限的方法 1. 使用路由守卫 Vue Router提供了一个功能强大的功能&#xff0c;即导航守卫&#xff08;N…

决定马里兰州地区版图的关键历史事件

1. 马里兰殖民地的建立&#xff1a; - 1632年&#xff0c;英国国王查理一世将一大片土地赐予塞西尔卡尔弗特男爵&#xff0c;这片土地是为了纪念国王的妻子亨丽埃塔玛丽亚而命名为“马里兰”。卡尔弗特和他的儿子随后建立了马里兰殖民地&#xff0c;这标志着马里兰作为一个独立…

java 关于 JDK 中自带的类加载器

关于 JDK 中自带的类加载器&#xff1a; 1、什么是类加载器&#xff1f; 专门负责加载类的命令/工具。ClassLoader 2、JDK 中自带了 3 个类加载器 启动类加载器&#xff08;父加载器&#xff09;&#xff1a;rt.jar 扩展类加载器&#xff08;母加载器&#xff0c;同父加载器一起…

【Java数据结构】——枚举以及Lambda表达式

目录 1 枚举背景及定义 &#x1f6a9;使用 &#x1f388;、switch语句 &#x1f388;、常用方法 &#x1f388;.构造方法 2. Lambda表达式 &#x1f6a9;背景 &#x1f388;Lambda表达式的语法 &#x1f308;函数式接口 &#x1f308;Lambda表达式的基本使用 ❗无参…

网安面试(协议篇)

动态路由协议的功能? 动态路由协议的功能是根据网络中实际情况动态地选择路由路径&#xff0c;以保证数据包能够快速、准确地到达目的地。常见的动态路由协议有OSPF、EIGRP、BGP等。 一次完整的http请求过程? 1. 客户端进行DNS域名解析&#xff0c;得到对应的IP地址 2. 根…

RabbitMQ使用

目录 初识MQ 同步通讯和异步通讯​编辑 同步通讯 同步调用存在的问题 总结 同步调用优点&#xff1a; 同步调用的问题&#xff1a; 异步通讯 事件驱动优势 总结 什么是MQ RabbitMQ快速入门 RabbitMQ概述和安装 RabbitMQ结构和概念​编辑 总结 常见消息模型 不同…

基于springboot+vue的旅游管理系统

博主主页&#xff1a;猫头鹰源码 博主简介&#xff1a;Java领域优质创作者、CSDN博客专家、阿里云专家博主、公司架构师、全网粉丝5万、专注Java技术领域和毕业设计项目实战&#xff0c;欢迎高校老师\讲师\同行交流合作 ​主要内容&#xff1a;毕业设计(Javaweb项目|小程序|Pyt…

基于YOLOv5的驾驶员疲劳驾驶行为​​​​​​​检测系统

&#x1f4a1;&#x1f4a1;&#x1f4a1;本文主要内容:详细介绍了疲劳驾驶行为检测整个过程&#xff0c;从数据集到训练模型到结果可视化分析。 博主简介 AI小怪兽&#xff0c;YOLO骨灰级玩家&#xff0c;1&#xff09;YOLOv5、v7、v8优化创新&#xff0c;轻松涨点和模型轻量…

MongoDB聚合运算符:$dateTrunc

聚合运算符$dateTrunc用于将日期字段的时间部分截断到指定的单位。 $dateTrunc 可以用于以下场景&#xff1a; 按指定的时间单位对日期进行截断&#xff1a;这意味着可以将日期字段舍入到最近的单位&#xff0c;如天、小时、分钟等。例如&#xff0c;将所有记录舍入到最近的小…

xtrabackup-docker实战

xtrabackup 引言 案例代码可以从https://github.com/WeiXiao-Hyy/k8s_example获取&#xff0c;欢迎Star&#xff01; 环境准备 创建数据文件夹和备份文件夹 mkdir -p ~/GitCode/xtrabackup/mysql/data mkdir -p ~/GitCode/xtrabackup/mysql/backup制作MySQL:5.7运行环境 …

Spring事务注解@Transactional的流程和源码分析

Spring事务简介 Spring事务有两种方式&#xff1a; 编程式事务&#xff1a;编程式事务通常使用编程式事务管理API实现&#xff0c;比如Spring提供的PlatformTransactionManager接口&#xff0c;使用它操控事务。声明式事务&#xff1a;注解式事务使用AOP&#xff08;面向切面…

字符串匹配问题(strs)

作者 刘昆 单位 中国矿业大学徐海学院 字符串中只含有括号 (),[],<>,{},判断输入的字符串中括号是否匹配。如果括号有互相包含的形式&#xff0c;从内到外必须是<>,(),[],{}&#xff0c;例如。输入: [()] 输出 YES&#xff0c;而输入([])&#xff0c;([)]都应该输…

通过Dockerfile创建镜像

通过Dockerfile创建镜像 Docker 提供了一种更便捷的方式&#xff0c;叫作 Dockerfile docker build命令用于根据给定的Dockerfile构建Docker镜像。 docker build语法&#xff1a; # docker build [OPTIONS] <PATH | URL | -> 1. 常用选项说明--build-arg&#xff0c;设置…