C++ type list 模板

C++ 实现一个type list 模板,在编译期计算。这个type list主要有构造,列表头类型,列表尾类型,concat操作,去除列表元素重复,获取指定元素,删除指定元素的操作。实现代码贴在下面:

#pragma once
#include <iostream>
#include <typeinfo>namespace type_list {// Step 1: 基础类型列表定义// 定义空列表template <typename ...Types>struct list {};// 非空列表的递归定义template <typename Type, typename ...Types>struct list<Type, Types...> {using head = Type;using tail = list<Types...>;};// 定义空列表类型using empty_list = list<>;// Step 2: 获取列表头部类型template <typename TypeList>using head_t = typename TypeList::head;// 获取列表尾部类型template <typename TypeList>using tail_t = typename TypeList::tail;// 构造新列表template <typename Head, typename Tail>struct construct;template <typename Head, typename Tail>using construct_t = typename construct<Head, Tail>::type;template <typename Head, typename ...Types>struct construct<Head, list<Types...>> {using type = list<Head, Types...>;};///template<typename TypeList>struct size;// 模板辅助类template<typename TypeList>constexpr size_t size_v = size<TypeList>::value;//size的模板特化,继承了std::integral_constanttemplate<typename...Types>struct size<list<Types...>> : std::integral_constant<std::size_t, sizeof...(Types)> {};template<class TypeList>constexpr bool empty_v = (size_v<TypeList> == 0);template<class TypeList>struct empty : std::bool_constant<empty_v<TypeList>> {};template<class TypeList>constexpr bool empty_s = empty<TypeList>::value;/// 查找模板的制定索引的value//声明一个模板;template<std::size_t Index, class TypeList>struct get;template<std::size_t Index, class TypeList>using get_t = typename get<Index, TypeList>::type;template<std::size_t Index, class TypeList>struct get {using type = get_t<Index - 1, tail_t<TypeList>>;};template<class TypeList>struct get<0, TypeList> {using type = head_t<TypeList>;};/// concattemplate<typename TypeList1, typename TypeList2>class concat;template<typename T, typename U>using concat_t = typename concat<T, U>::type;template<typename ...T, typename ...U>class concat<list<T...>, list<U...>> {public:using type = list<T..., U...>;};// delete specific typetemplate<typename TypeList, typename Type>class remove_all;template<typename TypeList, typename Type>using remove_all_t = typename remove_all<TypeList, Type>::type;template<typename TypeList, typename Type>class remove_all {public:using head = head_t<TypeList>;using tail = tail_t<TypeList>;using clean_tail = remove_all_t<tail, Type>;using type = std::conditional_t<std::is_same_v<head, Type>, clean_tail, construct_t<head, clean_tail>>;};template<typename Type>class remove_all<empty_list, Type> {public:using type = empty_list;};// get the last typetemplate<typename TypeList>class last;template<typename TypeList>using last_t = typename last<TypeList>::type;template<typename TypeList>class last {public:using type = last_t<tail_t<TypeList>>;};// 递归终止template<typename Type>class last<list<Type>> {public:using type = Type;};// distinct list typetemplate<typename TypeList>class distinct;template<typename TypeList>using distinct_t = typename distinct<TypeList>::type;template <typename TypeList>class distinct{public:using type = construct_t<head_t<TypeList>,distinct_t<remove_all_t<tail_t<TypeList>, head_t<TypeList>>>>;};// 递归终止template <>struct distinct<empty_list> {using type = empty_list;};// print listtemplate<typename T>void print_type() {int status;std::cout << typeid(T).name() << std::endl;}// 打印类型列表template <typename TypeList>struct print_list;template <typename TypeList>void print_list_func() {print_list<TypeList>::print();}// 主模板,递归地打印类型列表template <typename TypeList>struct print_list {static void print() {print_type<head_t<TypeList>>();print_list_func<tail_t<TypeList>>();}};// 终止递归的特化版本template <>struct print_list<empty_list> {static void print() {// 空列表,不打印任何内容}};} // namespace type_list

测试代码

#pragma once#include "type_list.h"
#include <iostream>using namespace std;void test_type_list() {//1.define listusing MyList = type_list::list<int, double, char>;using AnotherList = type_list::list<float, int>;using EmptyList = type_list::list<>;//2. get the head and tail type of listusing MyListHead = type_list::head_t<MyList>;using MyListTail = type_list::tail_t<MyList>;static_assert(std::is_same_v<MyList::tail, type_list::list<double,char>>, "MyList head shoud be int");static_assert(std::is_same_v<MyListTail, type_list::list<double, char>>, "MyList head shoud be int");static_assert(std::is_same_v<MyListHead, int>, "MyList head shoud be int");static_assert(std::is_same_v<MyList::head, int>, "MyList head shoud be int");//3. get list sizeconstexpr std::size_t listSize = type_list::size_v<MyList>;std::cout << listSize << std::endl;// 4. emptystatic_assert(type_list::empty_s<EmptyList>, "is empty");static_assert(type_list::empty_v<EmptyList>, "is empty");// 5. get index posusing FirstType = type_list::get_t<1, MyList>;static_assert(std::is_same_v<FirstType, double>, "not equal");// 6.concat listusing ConcatList = type_list::concat_t<MyList, AnotherList>;using ConcatListHead = type_list::head_t<ConcatList>;static_assert(std::is_same_v<ConcatListHead, int>, "head should be int");static_assert(std::is_same_v<ConcatList::head, int>, "head should be int");constexpr std::size_t concat_list_size = type_list::size_v<ConcatList>;cout << "concat_list_size : " << concat_list_size << endl;// 6.delete list typeusing RemoveIntList = type_list::remove_all<MyList, int>;static_assert(std::is_same_v<type_list::head_t<RemoveIntList>, int>, "the list head shoud not be int");// 6.delete list typeusing LastType = type_list::last_t<MyList>;static_assert(std::is_same_v<LastType, char>, "the last type should be char");// distinct list typeusing RepeatList = type_list::list<int, double, char, int, double, char, int,int,int>;using DistType = type_list::distinct_t<RepeatList>;std::cout << "--------------------------" << endl;type_list::print_list_func<RepeatList>();std::cout << "--------------------------" << endl;type_list::print_list_func<DistType>();std::cout << "--------------------------" << endl;}

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

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

相关文章

Linux系统编程项目——FTP网盘

目录 一、项目实现的功能 二、项目实现思路 2.1 socket服务器和客户端开发步骤&#xff1a; 2.2 指令的实现思路&#xff1a; 三、项目用到的函数 3.1 判断文件是否存在函数access()原型和头文件&#xff1a; 3.2 字符串输入函数fgets()原型和头文件&#xff1a; 3.3 字…

无线麦克风哪个品牌音质最好,一文告诉你无线领夹麦克风怎么挑选

随着直播带货和个人视频日志&#xff08;Vlog&#xff09;文化的兴起&#xff0c;以及自媒体内容创作的蓬勃发展&#xff0c;我们见证了麦克风行业的迅猛发展。在这一浪潮中&#xff0c;无线领夹麦克风以其无与伦比的便携性和操作效率&#xff0c;迅速赢得了广大视频制作者的喜…

前端HTML/CSS知识点系列

1. 什么是块级格式化上下文&#xff1f;【BFC(Block formatting context)】 BFC&#xff08;Block FormattingContext&#xff0c;块级格式化上下文&#xff09;是一个独立的渲染区域&#xff0c;其中的元素的布局不会受到外部元素的影响&#xff0c;反之亦然。BFC的创建有助于…

docker环境部署ruoyi系统前后端分离项目

创建局域网 docker network create net-ry 安装Redis 1 安装 创建两个目录 mkdir -p /data/redis/{conf,data} 上传redis.conf文件到/data/redis/conf文件夹中 cd /data/redis/conf 3.2 配置redis.conf文件 配置redis.conf文件&#xff1a; redis.conf文件配置注意&…

【数据库】四、数据库编程(SQL编程)

四、数据库编程 另一个大纲&#xff1a; 5.1存储过程 5.1.1存储过程基本概念 5.1.2创建存储过程 5.1.3存储过程体 5.1.4调用存储过程 5.1.5删除 5.2存储函数 5.2.1创建存储函数 5.2.2调用存储函数 5.2.3删除存储函数 目录 文章目录 四、数据库编程1.SQL编程基础1.1常量1.2变…

轻松两步,借助向量数据库 VectorDB 与千帆 Appbuilder 构建个性化本地问答知识库

在我们日常的工作和生活中&#xff0c;经常会遇到需要快速获取和管理大量信息的情况。无论是解答客户的问题&#xff0c;还是整理公司内部的资料&#xff0c;一个高效的知识库系统都能帮我们省下大量时间和精力。 为了帮助大家快速构建 RAG 应用&#xff0c;我们之前发布了一个…

虚拟机中VSCode+gcc环境配置

一、安装VSCode 1、在官网下载软件包&#xff1a; 地址&#xff1a;Documentation for Visual Studio Code 2、下载后在放置deb包的文件夹直接打开终端&#xff0c;然后输入sudo dpkg -i code_1.90.2-1718751586_amd64.deb 3、安装成功提示&#xff0c;并显示该图标 二、配…

nginx+keepalived+tomcat集群实验

如遇星河 | nginx+keepalived高可用集群实验 木子87 | Keepalived+Nginx+Tomcat 实现高可用Web集群 环境 192.168.40.204 tomcat-1 192.168.40.138 tomcat-2 安装tomcat [root@bogon local]# vim /etc/profile 添加环境变量 JAVA_HOME=/usr/local/java PATH=$J…

mac 常用工具命令集合

一、vim 快捷键 1、移动光标 h j k l 左 下 上 右 箭头上 上移一行 箭头下 下移一行 0 跳至行首&#xff0c;不管有无缩进&#xff0c;就是跳到第0个字符 ^ 跳至行首的第一个字符 $ 跳至行尾 gg 跳至文首 G 调至文尾 5gg/5G 调至第5行w 跳到下一个字首&#xff0c;按标点或…

Java中如何处理ArrayIndexOutOfBoundsException异常?

Java中如何处理ArrayIndexOutOfBoundsException异常&#xff1f; 大家好&#xff0c;我是免费搭建查券返利机器人省钱赚佣金就用微赚淘客系统3.0的小编&#xff0c;也是冬天不穿秋裤&#xff0c;天冷也要风度的程序猿&#xff01;在Java编程中&#xff0c;ArrayIndexOutOfBoun…

微信小程序学习(十):生命周期

1、应用生命周期 生命周期说明onLaunch监听小程序初始化&#xff0c;全局只会执行 1 次onShow监听小程序启动或切前台onHide监听小程序切后台 &#x1f517;应用生命周期官方文档 App({/*** 当小程序初始化完成时&#xff0c;会触发 onLaunch&#xff08;全局只触发一次&…

固特超声波清洗机怎么样?三大超声波清洗机美的、固特、希亦谁更好?

眼镜是我们日常生活中不可或缺的用具&#xff0c;但随着使用时间的增长&#xff0c;眼镜上的灰尘和污垢也会逐渐积累&#xff0c;传统的清洗方法往往难以彻底清洁。为了解决这一难题&#xff0c;超声波清洗机出现了&#xff01;它利用超声波振动原理&#xff0c;可以轻松、快速…

加载资源文件失败

背景 自己以前装了一个海康的深度学习算法平台&#xff0c;试用期是一个月&#xff0c;过了一个月之后&#xff0c;因为没有有效注册码或者加密狗的支持了导致无法使用&#xff0c;于是打算卸载掉&#xff0c;在卸载一个软件的时候&#xff0c;无论是使用控制面板还是软件自带的…

http 和 https 建立连接的不同

HTTP和HTTPS在建立连接时都需要进行“三次握手”&#xff0c;因为三次握手是TCP协议的一部分&#xff0c;用于建立可靠的传输层连接。 HTTP和HTTPS在建立连接时的过程 HTTP 的三次握手 HTTP协议本身是无状态的应用层协议&#xff0c;它依赖于传输层的TCP协议来建立和管理连接…

AI写文章生成器,这些工具都可以一键智能生成文章

在AI技术快速发展的今天&#xff0c;AI写作生成器成为我们创作内容的重要工具&#xff0c;它可以提高我们的写作效率&#xff0c;节省时间和精力。下面小编就来和大家分享几款优秀的AI写作生成器&#xff0c;帮助你快速生成高质量的文章。 1.专业AI写作工具-文章在线生成器 专…

LeetCode热题100——字母异位词分组

给你一个字符串数组&#xff0c;请你将 字母异位词 组合在一起。可以按任意顺序返回结果列表。 字母异位词 是由重新排列源单词的所有字母得到的一个新单词。 示例 1: 输入: strs [“eat”, “tea”, “tan”, “ate”, “nat”, “bat”] 输出: [[“bat”],[“nat”,“tan”…

计算机网络 访问控制列表以及NAT

一、理论知识 1. 单臂路由 单臂路由是一种在路由器上配置多个子接口的方法&#xff0c;每个子接口代表不同的 VLAN&#xff0c;用于在一个物理接口上支持多 VLAN 通信。此方法使得不同 VLAN 之间可以通过路由器进行通信。 2. NAT (网络地址转换) NAT 是一种在私有网络和公共…

步步精科技诚邀您参加2024慕尼黑上海电子展

尊敬的客户&#xff1a; 我们诚挚地邀请您参加即将于2024年7月8日至7月10日在上海新国际博览中心举办的2024慕尼黑上海电子展&#xff08;electronica China&#xff09;。此次展会汇聚了国内外优秀企业&#xff0c;展示从元器件到系统集成方案的完整产品链&#xff0c;为各行…

FragMent嵌套问题

在我们日常项目中应该经常会用到多个fragment嵌套的代码架构&#xff0c;虽然架构简单&#xff0c;但是其中会经常遇到各种不知名和各种奇葩的问题 1.页面显示空白 现象&#xff1a;当我们应用切换到其他应用&#xff0c;然后再重新进入到我们的应用&#xff0c;这时&#xff…

linux 下配置docker mirrors

一、配置mirrors vi /etc/docker/daemon.json {"registry-mirrors": ["https://docker.blfrp.cn"],"log-opts": {"max-size": "10m","max-file": "3"} }#完成配置后重启docker systemctl restart dock…