C++模板元编程----选择排序
目录
前言
模板在C++一直是比较神秘的存在。STL
和Boost
中都有大量运用模板,但是对于普通的程序员来说,模板仅限于使用。在一般的编程中,很少会有需要自己定义模板的情况。但是作为一个有理想的程序员,模板是一个绕不过去的坎。由于C++标准的不断改进,模板的能力越来越强,使用范围也越来越广。
在C++11中,模板增加了constexpr
,可变模板参数,回返类型后置的函数声明扩展了模板的能力;增加了外部模板加快了模板的编译速度;模板参数的缺省值,角括号和模板别名使模板的定义和使用变得更加的简洁。
C++14中,放宽了constexpr
的限制,增加了变量模板。
C++17中,简化模板的构造函数,使模板更加易用;Folding使得模板在定义中更加方便。
C++20是一个大版本更新,对于模板来说,也有很大的进步。对于个人来说,最喜欢的应该就是concept
了,它让模板可以判断模板参数是不是符合要求,同时也对模板的特化提供了更进一部的支持(以后再也不用看着模板成吨的报错流泪了。);同时它还要求大部分的STL库都支持constexpr
,使得很多类可以在编译期直接使用(以后模板元编程就不是单纯的函数式语言了吧,感觉以后C++的编程会变得非常奇怪)。
而随着模板一步步的完善,大佬们发现模板的功能居然已经实现了图灵完备,于是各种骚操作层出不穷,比如俄罗斯方块Super Template Tetris。
作为一个小老弟,当然是还没有能力写出一个可以媲美俄罗斯方块的程序,不过写一些简单的排序还是可以的。
这里我分享的是一个选择排序算法。为什么选择选择排序呢?因为它排序的时候,他对于元素的位置改变是比较少的。个人感觉函数元编程最复杂的就是对元素进行修改位置了吧。
代码详解
数据的结构
template<int ...data>
struct mvector;
template<int first, int ...data>
struct mvector<first, data...> {
static constexpr int size = sizeof...(data) + 1;
static constexpr int value = first;
typedef mvector<data...> next_type;
constexpr static std::array<int, sizeof...(data) + 1> array = {first, data...};
};
template<int first>
struct mvector<first> {
static constexpr int size = 1;
static constexpr int value = first;
typedef mvector<> next_type;
constexpr static int array[] = {first};
};
template<>
struct mvector<> {
static constexpr int size = 0;
static constexpr int value = -1;
typedef mvector<> next_type;
constexpr static int array[] = {};
};
这里我们定义了一个mvcetor
模板,他的作用就是用来保存数据的。模板的原型是
template<int ...data>
struct mvector;
他可以输入任意数量的整数(模板参数可以看作是输入)。
根据后面的特化,模板一共有四个属性或类型(这些可以看作是模板的输出),分别是size
,value
(第一个元素的值,方便后面的迭代),next_type
(除去头的尾部,方便迭代),array
(mvector
的数组表现形式)。
数据的操作
分割向量
// 分割向量
template<int index, typename T, typename S>
struct SplitVector;
template<int index, int ...LeftData, int ...RightData>
struct SplitVector<index, mvector<LeftData...>, mvector<RightData...>> {
typedef SplitVector<index - 1, mvector<LeftData..., mvector<RightData...>::value>, typename mvector<RightData...>::next_type> next_split;
typedef typename next_split::LeftVector LeftVector;
typedef typename next_split::RightVector RightVector;
};
template<int ...LeftData, int ...RightData>
struct SplitVector<0, mvector<LeftData...>, mvector<RightData...>> {
typedef mvector<LeftData...> LeftVector;
typedef typename mvector<RightData...>::next_type RightVector;
};
这个模板的主要目的是将向量从某一部分分离出来(取最大值)。
模板的输入有三个:index
(要分离的元素的位置在RightData
的位置),LeftData
(分离的左边),RightData
(分离的右边)。
输出有LeftVector
(出来的左边),RightVector
(出来的右边)。
合并向量
// 合并向量
template<typename T, typename S>
struct MergeVector;
template<int ...dataa, int ...datab>
struct MergeVector<mvector<dataa...>, mvector<datab...>> {
typedef mvector<dataa..., datab...> result_type;
};
将两个向量合并,主要是用在分割后的向量。
寻找最大值
template<int now_index, typename U, typename V>
struct FindMax;
template<int now_index, int ...Looped, int ...unLooped>
struct FindMax<now_index, mvector<Looped...>, mvector<unLooped...>> {
typedef FindMax<now_index + 1, mvector<Looped..., mvector<unLooped...>::value>, typename mvector<unLooped...>::next_type> next_max;
constexpr static int max = mvector<unLooped...>::value > next_max::max ? mvector<unLooped...>::value : next_max::max;
constexpr static int max_index = mvector<unLooped...>::value > next_max::max ? now_index : next_max::max_index;
};
template<int now_index, int ...Looped>
struct FindMax<now_index, mvector<Looped...>, mvector<>> {
typedef FindMax<now_index, mvector<Looped...>, mvector<>> next_max;
constexpr static int max = -1;
constexpr static int max_index = now_index;
};
寻找向量中的最大值。输入有now_index
,Looped
(已经比较的部分),unLooped
(未比较的部分)。其中now_index
是多余的,可以使用sizeof...(Looped)
来代替。
输出是max
(最大值),max_index
(最大值的位置,方便后面的分割)
排序
对数据操作完成了,这个程序也就完成了一大半了,排序也是非常的简单,从未排序的列表中,选择最大的值,放到已经排序好的列表的前面就好了。
// 排序
template<typename T, typename S>
struct SelectSortWork;
template<int ...unSorted, int ...Sorted>
struct SelectSortWork<mvector<unSorted...>, mvector<Sorted...>> {
typedef FindMax<0, mvector<>, mvector<unSorted...>> max_find_type;
constexpr static int max = max_find_type::max;
constexpr static int max_index = max_find_type::max_index;
typedef SplitVector<max_index, mvector<>, mvector<unSorted...>> split_type;
typedef SelectSortWork<typename MergeVector<typename split_type::LeftVector, typename split_type::RightVector>::result_type, mvector<max, Sorted...>> next_select_sort_work_type;
typedef typename next_select_sort_work_type::sorted_type sorted_type;
};
template<int ...Sorted>
struct SelectSortWork<mvector<>, mvector<Sorted...>> {
typedef mvector<Sorted...> sorted_type;
};
总结
代码我放在了github的gist上,select_sort.cpp。
总的来说,代码还是非常的简单的,只要合理的进行分解,大部分的算法应该都是可以实现的。
在编程的过程中,我也有一些自己的领悟,对于模板元编程的几点小Tips,在这里给大家介绍一下吧。
- 如果熟悉函数式编程的话,再来学习模板元编程,对于其中的理解会更加的深刻,所以最好在开始准备学习之前,先学习一下函数式编程会比较好(虽然这个过程会非常的痛苦)。
- 类模板可以看作是一个函数,有输入输出。输入是模板的参数,输出是模板里面的类型或者变量,这些输出也可以作为函数计算的中间变量,方便编码。
- 模板元编程,一定要有耐心,特别是debug,会特别的难受
博客原文:https://www.cnblogs.com/ink19/p/cpp_template_select_sort.html
C++模板元编程----选择排序的更多相关文章
- C++模板元编程 - 函数重载决议选择工具(不知道起什么好名)完成
这个还是基于之前实现的那个MultiState,为了实现三种类型“大类”的函数重载决议:所有整数.所有浮点数.字符串,分别将这三种“大类”的数据分配到对应的Converter上. 为此实现了一些方便的 ...
- C++模板元编程(C++ template metaprogramming)
实验平台:Win7,VS2013 Community,GCC 4.8.3(在线版) 所谓元编程就是编写直接生成或操纵程序的程序,C++ 模板给 C++ 语言提供了元编程的能力,模板使 C++ 编程变得 ...
- 读书笔记_Effective_C++_条款四十八:了解模板元编程
作为模板部分的结束节,本条款谈到了模板元编程,元编程本质上就是将运行期的代价转移到编译期,它利用template编译生成C++源码,举下面阶乘例子: template <int N> st ...
- effective c++ Item 48 了解模板元编程
1. TMP是什么? 模板元编程(template metaprogramming TMP)是实现基于模板的C++程序的过程,它能够在编译期执行.你可以想一想:一个模板元程序是用C++实现的并且可以在 ...
- 读书笔记 effective c++ Item 48 了解模板元编程
1. TMP是什么? 模板元编程(template metaprogramming TMP)是实现基于模板的C++程序的过程,它能够在编译期执行.你可以想一想:一个模板元程序是用C++实现的并且可以在 ...
- 初识C++模板元编程(Template Mega Programming)
前言:毕设时在开源库上做的程序,但是源码看得很晕(当时导师告诉我这是模板元编程,可以不用太在乎),最近自己造轮子时想学习STL的源码,但也是一样的感觉,大致了解他这么做要干什么,但是不知道里面的机制. ...
- 现代c++与模板元编程
最近在重温<c++程序设计新思维>这本经典著作,感慨颇多.由于成书较早,书中很多元编程的例子使用c++98实现的.而如今c++20即将带着concept,Ranges等新特性一同到来,不得 ...
- C++ 模板元编程 学习笔记
https://blog.csdn.net/K346K346/article/details/82748163 https://www.jianshu.com/p/b56d59f77d53 https ...
- C++模板元编程----堆排序
目录 目录 前言 实现的一些小细节 Debug 惰性求值 总结 Ref 前言 经过前两次经验的积累,终于来到了麻烦的堆排序.在一开始接触模板元编程的时候,我就期望有一天能够写出元编程堆排序的代码.原因 ...
随机推荐
- bootstrap-datetimepicker 编辑回显
官网上居然没给出解决方案....汗 stackoverflow给出了灵感: $("#dateOfManufacture").find("input").val( ...
- Win7 安装 Docker 踩的那些坑
公司电脑是 WIN7 x64 旗舰版 SP1,安装 Docker 时踩了好多雷,分享出来给大家排排雷. 首先,Docker Desktop Installer 的 Windows 版只支持 Win10 ...
- Java-Annotation的一种用法(消除代码中冗余的if/else或switch语句)
Java-Annotation的一种用法(消除代码中冗余的if/else或switch语句) 1.冗余的if/else或switch 有没有朋友写过以下的代码结构,大量的if/esle判断,来选择 ...
- H265Nalu头部解析
一 NALU头部解析 F: 必须为0,为1表示语法错误.整包将被丢弃 NalType:nalu包的类型,其中VCL NAL和non-VCL NAL各有32类.0-31是vcl nal单元:32-63, ...
- git 强制更新本地和强制提交覆盖
强制更新覆盖本地: git pull时出现冲突 放弃本地修改,使远程库内容强制覆盖本地代码git fetch --all //只是下载代码到本地,不进行合并操作git reset --hard ori ...
- zookeeper和kafka的leader和follower
来源于:https://www.cnblogs.com/aspirant/p/9179045.html 一.zookeeper 与kafka保持数据一致性的不同点: (1)zookeeper使用了ZA ...
- 「CSP-S 2019」Emiya 家今天的饭
description loj 3211 solution 看到题目中要求每种主要食材至多在一半的菜中被使用,容易想到补集转换. 即\(ans=\)总方案数-存在某一种食材在一半以上的菜中被使用的方案 ...
- Elasticsearch搜索资料汇总
Elasticsearch 简介 Elasticsearch(ES)是一个基于Lucene 构建的开源分布式搜索分析引擎,可以近实时的索引.检索数据.具备高可靠.易使用.社区活跃等特点,在全文检索.日 ...
- Java基础教程——Socket编程
Socket通常也称作"套接字",用于描述IP地址和端口,可以用来实现不同虚拟机或不同计算机之间的通信. --百度百科 套接字允许应用程序插入到网络中,并与插入到网络中的其他应用程 ...
- 关于uniapp无法navigateTo跳转的解决办法
今天在分包时突然无法跳转了,记个笔记 场景: 位于tabbar页面(主包)的子组件跳转到分包页面时,无法跳转 尝试办法: 使用uniapp原生跳转 uni.navigateTo({ url:'xxxx ...