c++ java 选择排序,C++模板元编程实现选择排序

news/2024/7/4 13:42:38

前言

模板在C++一直是比较神秘的存在。 STL 和 Boost 中都有大量运用模板,但是对于普通的程序员来说,模板仅限于使用。在一般的编程中,很少会有需要自己定义模板的情况。但是作为一个有理想的程序员,模板是一个绕不过去的坎。由于C++标准的不断改进,模板的能力越来越强,使用范围也越来越广。

在C++11中,模板增加了 constexpr ,可变模板参数,回返类型后置的函数声明扩展了模板的能力;增加了外部模板加快了模板的编译速度;模板参数的缺省值,角括号和模板别名使模板的定义和使用变得更加的简洁。

C++14中,放宽了 constexpr 的限制,增加了变量模板。

C++17中,简化模板的构造函数,使模板更加易用;Folding使得模板在定义中更加方便。

C++20是一个大版本更新,对于模板来说,也有很大的进步。对于个人来说,最喜欢的应该就是 concept 了,它让模板可以判断模板参数是不是符合要求,同时也对模板的特化提供了更进一部的支持(以后再也不用看着模板成吨的报错流泪了。);同时它还要求大部分的STL库都支持 constexpr ,使得很多类可以在编译期直接使用(以后模板元编程就不是单纯的函数式语言了吧,感觉以后C++的编程会变得非常奇怪)。

而随着模板一步步的完善,大佬们发现模板的功能居然已经实现了图灵完备,于是各种骚操作层出不穷,比如俄罗斯方块Super Template Tetris。

作为一个小老弟,当然是还没有能力写出一个可以媲美俄罗斯方块的程序,不过写一些简单的排序还是可以的。

这里我分享的是一个选择排序算法。为什么选择选择排序呢?因为它排序的时候,他对于元素的位置改变是比较少的。个人感觉函数元编程最复杂的就是对元素进行修改位置了吧。

代码详解

数据的结构

template

struct mvector;

template

struct mvector {

static constexpr int size = sizeof...(data) + 1;

static constexpr int value = first;

typedef mvector next_type;

constexpr static std::array array = {first, data...};

};

template

struct mvector {

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

struct mvector;

他可以输入任意数量的整数(模板参数可以看作是输入)。

根据后面的特化,模板一共有四个属性或类型(这些可以看作是模板的输出),分别是 size , value (第一个元素的值,方便后面的迭代), next_type (除去头的尾部,方便迭代), array ( mvector 的数组表现形式)。

数据的操作

分割向量

// 分割向量

template

struct SplitVector;

template

struct SplitVector, mvector> {

typedef SplitVector::value>, typename mvector::next_type> next_split;

typedef typename next_split::LeftVector LeftVector;

typedef typename next_split::RightVector RightVector;

};

template

struct SplitVector<0, mvector, mvector> {

typedef mvector LeftVector;

typedef typename mvector::next_type RightVector;

};

这个模板的主要目的是将向量从某一部分分离出来(取最大值)。

模板的输入有三个: index (要分离的元素的位置在 RightData 的位置), LeftData (分离的左边), RightData (分离的右边)。

输出有 LeftVector (出来的左边), RightVector (出来的右边)。

合并向量

// 合并向量

template

struct MergeVector;

template

struct MergeVector, mvector> {

typedef mvector result_type;

};

将两个向量合并,主要是用在分割后的向量。

寻找最大值

template

struct FindMax;

template

struct FindMax, mvector> {

typedef FindMax::value>, typename mvector::next_type> next_max;

constexpr static int max = mvector::value > next_max::max ? mvector::value : next_max::max;

constexpr static int max_index = mvector::value > next_max::max ? now_index : next_max::max_index;

};

template

struct FindMax, mvector<>> {

typedef FindMax, 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

struct SelectSortWork;

template

struct SelectSortWork, mvector> {

typedef FindMax<0, mvector<>, mvector> max_find_type;

constexpr static int max = max_find_type::max;

constexpr static int max_index = max_find_type::max_index;

typedef SplitVector, mvector> split_type;

typedef SelectSortWork::result_type, mvector> next_select_sort_work_type;

typedef typename next_select_sort_work_type::sorted_type sorted_type;

};

template

struct SelectSortWork, mvector> {

typedef mvector sorted_type;

};

总结

代码我放在了github的gist上, select_sort.cpp。

总的来说,代码还是非常的简单的,只要合理的进行分解,大部分的算法应该都是可以实现的。

在编程的过程中,我也有一些自己的领悟,对于模板元编程的几点小Tips,在这里给大家介绍一下吧。

如果熟悉函数式编程的话,再来学习模板元编程,对于其中的理解会更加的深刻,所以最好在开始准备学习之前,先学习一下函数式编程会比较好(虽然这个过程会非常的痛苦)。

类模板可以看作是一个函数,有输入输出。输入是模板的参数,输出是模板里面的类型或者变量,这些输出也可以作为函数计算的中间变量,方便编码。

模板元编程,一定要有耐心,特别是debug,会特别的难受

到此这篇关于C++模板元编程实现选择排序的文章就介绍到这了,更多相关C++ 选择排序内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!


http://www.niftyadmin.cn/n/4072841.html

相关文章

产品经理十六章:如何提升项目管理效率

在项目团队成立之后&#xff0c;项目经理就是船长&#xff0c;他不仅仅要根据航行过程中的情况控制航行的方向和路线&#xff0c;而且要负责管理其他所有与成功到达终点相关的工作。 为了提升项目管理效率&#xff0c;项目经理还应该重点做好以下工作&#xff1a; 一、严格控制…

ridgelet matlab,第一代curvelet变换matlab程序

【实例简介】用于图像处理的Curelet变换常用工具箱&#xff0c;有VC,Matlab等实现的代码。【实例截图】【核心代码】curvelet_first_generation├── curvelet_first_generation│ ├── addwgn.m│ ├── barbara_256x256.jpg│ ├── cameraman.tif│ ├── car…

互联网之父:文登•瑟夫

互联网每一步关键的进化背后&#xff0c;都有一位了不起的的奠基人。现年67岁的文登•瑟夫(Vinton G. Cerf)就是这样一位当之无愧的“互联网之父”。他和罗伯特•卡恩一起构建了TCP&#xff0c;使其成为标准走向世界。作为因特网方面为数不多的权威之一&#xff0c;Cerf作为TCP…

Linux用户管理(八)Shell编程基础

Shell编程一&#xff0e; Shell的基本概念n Linux shell是内置编程语言。提供管道、命令替换、自动补齐机制n BashShell(bash):Bourne Shell的增强版&#xff0c;Linux系统的默认Shell二&#xff0e; Bash Shell编程基础1&#xff0e;Shell的变量和参数&#xff…

matlab图像平均背景建模,【Opencv手记】平均背景法建模提取前景

#include "highgui.h"#include "cv.h"#include "cxcore.h"/*为不同临时图像和统计属性图像创建指针*///三通道float图像IplImage *IavgF,*IdiffF,*IprevF,*IhiF,*IlowF;IplImage *Iscratch,*Iscratch2;//单通道float图像IplImage *Igray1,*Igray…

LVN与其在Linux上的实现

参考资料&#xff1a; LVM详解-骏马金龙-博客园 How to reduce the size of an LVM partition formatted with xfs filesystem on CentOS7? 骏马兄的博文会相对深入一点&#xff0c;并且他是基于ext系列文件系统来演示扩容与缩容&#xff0c;而我使用的是xfs文件系统。 基本概…

dubbo的一次请求源码分析

调用某个服务首先会进入到动态代理。 InvokerInvocationHandler#invoke(Object proxy, Method method, Object[] args)public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {String methodName method.getName();Class<?>[] parameter…

matlab mwarray,C语言与matlab混合编程中mwArray的Get函数的简单用法解释

网上的通用示例&#xff1a;double data[4] {1.0, 2.0, 3.0, 4.0};double x;mwArray a(2, 2, mxDOUBLE_CLASS);a.SetData(data, 4);x a.Get(1,1); // x 1.0x a.Get(2, 1, 2); // x 3.0x a.Get(2, 2, 2); // x 4.0这个示例我就看得很蛋疼&#xff0c;八成是官方示例(笑)。…