数据结构排序2-希尔,快速,归并排序

 

希尔排序

希尔排序是对插入排序的改进。增加了分组

分组插入排序 降低元素个数,然后对每组分别进行插入排序
利用分组增量遍历,因为最后必须全部排序一次,然后分别对每组插入排序。把每组的第一个数作为有序序列,
要比较的第一个基准数就是第二个

#include<iostream>
#include<stdio.h>
#include<stdlib.h>
#include<random>
#include<time.h>
#include<sys/timeb.h>
using namespace std;
#define MAX 50000long getSystemTime()
{struct timeb tb;ftime(&tb);return tb.time * 1000 + tb.millitm;  //毫秒
}void printArr(int arr[], int length)
{for (int i = 0; i < length; i++){printf("%d ", arr[i]);}printf("\n");
}
//插入排序
void insertSort(int arr[], int length)
{int j;for (int i = 1; i < length; i++)  //从第二个数开始是无序队列。第一个数为有序队列{if (arr[i]<arr[i - 1]){int temp = arr[i];  //将满足条件的元素缓冲到temp中for (j = i - 1; j >= 0 && temp < arr[j]; j--)  //将有序数列分别遍历{arr[j + 1] = arr[j];}arr[j + 1] = temp;  //填补坑}}
}
//错误
void shellSortme(int arr[], int length)
{int increasement = length;//确定分组的增量increasement = increasement /3+1;int i, j, k;for (i = 0; i < increasement; i++)  //外层分组循环,最后等于1时全部再一起排列一次{for (j = i + increasement; j < length; j += increasement)  //从i+creasement是无序的第一个数{if (arr[i + increasement] < arr[i]){int temp=arr[i];for (k = i + increasement; k < j && arr[k] < temp; k -= increasement){arr[k] = arr[k + 1];}arr[k + 1] = temp;}}}
}void shellSort(int arr[], int length)
{int increasement = length;int i, j, k;do {//确定分组的增量increasement = increasement / 3 + 1;//外层分组循环,最后等于1时全部再一起排列一次for (i = 0; i < increasement; i++)  {//从i+creasement是无序的第一个数for (j = i + increasement; j < length; j += increasement)  {if (arr[j] < arr[j - increasement]){int temp = arr[j];//temp如果大于arr[k]的话,就不需要移动元素。//只有temp小于arr[k]即有序数组的其中一个数,才需要继续往前移动for (k = j - increasement; k >= 0 && temp<arr[k]; k -= increasement){arr[k + increasement] = arr[k];  //将满足条件有序的元素往后移动}arr[k + increasement] = temp;}}}} while (increasement > 1);
}int main()
{int arr[MAX];srand((unsigned int)time(NULL));  //种子for (int i = 0; i < MAX; i++){arr[i] = rand() % MAX;}//printf("未排序数组为:");//printArr(arr, MAX);long t_start = getSystemTime();shellSort(arr, MAX);long t_end = getSystemTime();//printf("希尔排序后数组为:");//printArr(arr, MAX);printf("希尔排序%d个元素,所需时间:%1d\n", MAX, t_end - t_start);system("pause");return 0;
}

运行结果:

快速排序

分治法加挖坑填数
找一个基准,小于放左边,大于放右边。一直遍历。填坑。
一般将第一数作为基准数。第一个数下标为i.挖坑。最后一个下标为j,从右边找一个数填坑。找到小于基准数的然后填坑。则此时j的位置就是坑了,此时从左往右找比基准数大的然后填坑。。。。此时i的地方为坑,继续一直循环下去,直到i==j时,把基准数填i位置的坑。然后此时i的位置是正确的,然后将i的左边和右边分别进行相同的操作。所以用到了递归i<j就是递归结束条件>

#include<iostream>
#include<stdio.h>
#include<stdlib.h>
#include<random>
#include<time.h>
#include<sys/timeb.h>
using namespace std;
#define MAX 50000long getSystemTime()
{struct timeb tb;ftime(&tb);return tb.time * 1000 + tb.millitm;  //毫秒
}void printArr(int arr[], int length)
{for (int i = 0; i < length; i++){printf("%d ", arr[i]);}printf("\n");
}void quickSortme(int arr[], int start, int end)
{int mid = (start + end) / 2;int temp = arr[start];int i = start;int j = end;if (i >= j){//从后往前比较,找一个比基准数小的数,放前面if (arr[j] < temp){arr[i] = arr[j];arr[j] = temp;j--;}//从前往后比较,找一个比基准数大的数,放后面if (arr[i] > temp){arr[i] = arr[j];arr[i] = temp;i++;}}
}
//快速排序,从小到大
void quickSort(int arr[], int start, int end)
{	int i = start;int j = end;int temp = arr[start];  //作为基准数if (i < j){while (i<j){//从右向左比较,找一个比基准数小的元素,放左边while (i < j && arr[j] >= temp)  //从右往左遍历,寻找满足小于基准数的元素{j--;}if (i<j) {arr[i] = arr[j];  //填左边坑i++;}//从左往右比较,找一个比基准数大的数,放右边while (i < j && arr[i] <= temp)  //从左往右遍历,寻找满足大于基准数的元素{i++;}if (i<j){arr[j] = arr[i];  //填右边坑j--;}}//把基准数放到i位置arr[i] = temp;//对左半部分进行快速排序quickSort(arr,start, i - 1);//对右半部分进行快速排序quickSort(arr, i+1, end);}
}int main()
{int arr[MAX];srand((unsigned int)time(NULL));  //种子for (int i = 0; i < MAX; i++){arr[i] = rand() % MAX;}//printf("未排序数组为:");//printArr(arr, MAX);long t_start = getSystemTime();quickSort(arr, 0,MAX-1);long t_end = getSystemTime();//printf("冒希尔排序后数组为:");//printArr(arr, MAX);printf("快速排序%d个元素,所需时间:%1d\n", MAX, t_end - t_start);system("pause");return 0;
}

结果:

归并排序

分组,然后排序,然后存到数组,然后覆盖原来

#include<iostream>
#include<stdio.h>
#include<stdlib.h>
#include<random>
#include<time.h>
#include<sys/timeb.h>
using namespace std;
#define MAX 50000long getSystemTime()
{struct timeb tb;ftime(&tb);return tb.time * 1000 + tb.millitm;  //毫秒
}void printArr(int arr[], int length)
{for (int i = 0; i < length; i++){printf("%d ", arr[i]);}printf("\n");
}int* CreateArr()
{srand((unsigned int)time(NULL));  //种子int* arr =(int *)malloc(sizeof(int)*MAX);for (int i = 0; i < MAX; i++){arr[i] = rand() % MAX;}return arr;
}
//合并算法  从小到大 
void merge(int arr[], int start, int end, int mid, int *temp)
{int i_start = start;int i_end = mid;int j_start = mid + 1;int j_end = end;//表示辅助空间有多少个元素int length = 0;//合并两个有序序列while (i_start <= i_end && j_start <= j_end){if (arr[i_start] < arr[j_start]){temp[length] = arr[i_start];length++;i_start++;}else{temp[length] = arr[j_start];length++;j_start++;}}//i这个序列while (i_start <= i_end){temp[length] = arr[i_start];length++;i_start++;}//j这个序列while (j_start <= j_end){temp[length] = arr[j_start];length++;j_start++;}//将辅助空间数据覆盖原空间for (int i = 0; i < length; i++){arr[start+i]=temp[i];}
}//归并排序,将两个有序序列合并成一个有序序列   不会写
void mergeSort(int arr[], int start,int end,int *temp)
{if (start >= end){return;}int mid = (start + end) / 2;//左半边mergeSort(arr, start, mid, temp);//右半边mergeSort(arr, mid+1, end, temp);//合并merge(arr, start, end, mid, temp);}int main()
{int *arr = CreateArr();long t_start = getSystemTime();//申请辅助空间int* temp = (int *)malloc(sizeof(int)*MAX);mergeSort(arr, 0, MAX - 1,temp);long t_end = getSystemTime();printf("归并排序%d个元素,所需时间:%1d\n", MAX, t_end - t_start);/*free(temp);free(arr);*/system("pause");return 0;
}

运行结果:

归并排序示意图:

思路:

因为题目要求复杂度为O(nlogn),故可以考虑归并排序的思想。

归并排序的一般步骤为:

1)将待排序数组(链表)取中点并一分为二;

2)递归地对左半部分进行归并排序;

3)递归地对右半部分进行归并排序;

4)将两个半部分进行合并(merge),得到结果。

参考自https://blog.csdn.net/boguesfei/article/details/80413365

 

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

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

相关文章

数据结构排序3-堆排序

堆排序 思想&#xff1a;假设数组放入完全二叉树中&#xff0c; 1、初始化堆&#xff1a;调节父结点与子结点的大小。让所有的子结点都小于父结点。 2、将完全二叉树中的叶子结点和根结点进行互换后&#xff0c;继续调整堆。直至结束 #include<iostream> #include<…

删除代码中的空行

打开替换&#xff08;ctrlh?&#xff09;&#xff0c;使用正则表达式&#xff1a; ^\s*\n ^ 匹配输入字符串的开始位置\s 表示匹配任何空白字符&#xff0c;包括制表符&#xff0c;空格&#xff0c;换页符等等* 表示匹配多次\n 表示匹配换行符 替换为&#xff1a; <…

数据结构-哈希与映射

二叉树映射map #include<iostream> #include<map> //映射&#xff08;也叫字典&#xff09;&#xff0c;二叉树映射&#xff0c;不是哈希映射 using namespace std; int main() {cout << "二叉树&#xff08;红黑树&#xff09;" << endl;…

python3.6.5+cuda9+cudnn7.1+win10+tensorflow-gpu1.9.0下载配置

一、首先明确cuda和cudnn对应版本 tensorflow版本与cuda cuDNN版本对应使用 tensorflow-gpu v1.9.0 | cuda9.0 | cuDNN7.1.4可行 | 备注&#xff1a;7.0.4/ 7.0.5/ 7.1.2不明确 tensorflow-gpu v1.8.0 | cuda9.0 | cuDNN 不明确 | 备注&#xff1a;7.0.4/ 7.0.5/ 7.1.2/…

python配置opencv镜像安装

直接利用pip install opencv-python 安装即可&#xff0c;但是会出现time out情况&#xff0c;所以采用镜像安装 pip install opencv-python -i https://pypi.tuna.tsinghua.edu.cn/simple 打开pycharm验证&#xff1a;输入import cv2 as cv 不报错&#xff0c;即安装成功

win10+vs2015+opencv3配置

一、下载opencv3和vs2015&#xff0c;并解压与安装 二、配置opencv环境变量 右键“我的电脑---属性---高级系统设置---环境变量&#xff0c;在下方“环境变量”里找到“Path”&#xff0c;进入编辑&#xff1b; 添加”…opencv\build\x64\vc14\bin”&#xff0c;如…

数据结构-图及其遍历

图-邻接矩阵 #include<iostream> #define MAX_VERTS 20 using namespace std; //邻接矩阵 浪费空间class Vertex { //顶点 public:Vertex(char lab) { Label lab; } private:char Label; };class Graph { //图 public:Graph();~Graph();void addVertex(char lab);voi…

ios http长连接_Nginx篇05——http长连接和keeplive

nginx中http模块使用http长连接的相关配置(主要是keepalive指令)和http长连接的原理解释。1、http长连接1.1 预备知识连接管理是一个 HTTP 的关键话题&#xff1a;打开和保持连接在很大程度上影响着网站和 Web 应用程序的性能。在 HTTP/1.x 里有多种模型&#xff1a;短连接, 长…

php 目录555 权限_CMS网站安全权限划分设置教程

DiYunCMS(帝云CMS)-免费开源可商用的PHP万能建站程序CMS网站安全权限划分设置教程网站安全是网站搭建运营过程中非常重要的一部分&#xff0c;DiYunCMS非常注重安全&#xff0c;开发了强大的安全功能。本文给大家介绍下DiYunCMS网站的安全设置方法。首先需要安装【系统安全】插…

python 列表生成式

python笔记21-列表生成式 前言 python里面[]表示一个列表&#xff0c;快速生成一个列表可以用range()函数来生成。 对列表里面的数据进行运算和操作&#xff0c;生成新的列表最高效快速的办法&#xff0c;那就是列表生成式了。 range() 1.一个连续的数字类型列表&#xff0…

php mysql grant_mysql grant命令详解_MySQL

bitsCN.comgrant 权限 on 数据库对象 to 用户一、grant 普通数据用户&#xff0c;查询、插入、更新、删除 数据库中所有表数据的权利。grant select on testdb.* to common_user%grant insert on testdb.* to common_user%grant update on testdb.* to common_user%grant delet…

python matplotlib简单使用

一、简单介绍 Matplotlib是Python的一个绘图库&#xff0c;是Python中最常用的可视化工具之一。 二、安装方法 安装方法&#xff1a;pip install matplotlib 注意&#xff1a;安装matplotlib前需要先安装numpy才可以 三、基本绘图命令 1、plt.fig([num]) 在绘图过程中&a…

python窗体设计插件_Python 界面生成器 wxFormBuilder 的入门使用(wxPython的界面设计工具的初学笔记)...

环境&#xff0c;Win10&#xff0c;python3.7.3&#xff0c;wxPython 4.0.4&#xff0c;wxFormBuilder 3.91、准备一个窗体。点击wxformbuilder上方的标签“forms”&#xff0c;并点击标签下方的第一个类似窗体的图标“Frame”然后&#xff0c;下面就会出现一个窗体。但是现在还…

用cmd编译c++程序

1、设置好环境变量&#xff08;已安装vs&#xff09; ①在计算机的系统环境变量--》用户变量--》path中添加 D:\VS15\VC\bin ②新建变量INCLUDE&#xff1a;D:\VS15\VC\include ③新建变量LIB&#xff1a;D:\VS15\VC\lib 2、利用cl编译c文件 打开cmd 输入c…

python插入排序_直接插入排序(python实现)

这篇博文用来介绍直接插入排序直接插入排序基本思想&#xff1a;每次将一个待排序的记录插入到已经排好序的数据区中&#xff0c;直到全部插入完为止直接插入排序算法思路&#xff1a;在直接插入排序中&#xff0c;数据元素分为了有序区和无序区两个部分&#xff0c;在这里我们…

STL1-函数模板

1、函数模板和普通函数区别 //普通函数可以进行自动类型转换&#xff0c; //函数模板必须精确类型匹配; //函数模板可以被重载;c优先考虑普通函数;#include<iostream> using namespace std; //函数模板-->产生模板函数-->调用函数 template<class T> T MyAd…

jupyter安装与迁移文件

1、安装 pip install jupyter notebook -i https://pypi.tuna.tsinghua.edu.cn/simple 2、测试安装成功 安装完后输入 jupyter notebook 出现一个jupyter网址&#xff0c;即证明安装成功 3、数据迁移 将之前的jupyter notebook产生的文件复制在python所安装的盘目录下。然…

修正的判定条件覆盖例题_语句覆盖、判断覆盖、条件覆盖、条件判定组合覆盖、多条件覆盖、修正条件覆盖...

int function(bool a,bool b,boolc){intx;x0;if(a&&(b||c)){x1;returnx;}}1、语句覆盖(SC)选择足够多的测试数据&#xff0c;使得被测程序中的每条语句至少执行一次。测试用例&#xff1a;aT,bT,cT2、判断覆盖(DC)设计足够的测试用例&#xff0c;使得程序中的每个判定至…

STL2-类模板

1、类模板实现 函数模板在调用时可以自动类型推导 类模板必须显式指定类型 #include<iostream> using namespace std;template<class T> class Person { public:T mId;T mAge; public:Person(T id,T age){this->mAge age;this->mId id;}void Show(){cout…

STL3-MyArray动态数组类模板实现

注意 1、右值的拷贝使用 2、拷贝构造函数的使用 #include<iostream> using namespace std;template<class T> class MyArray{ public:MyArray(int capacity){this->mCapacity capacity;this->mSize 0;//申请内存this->pAddr new T[this->mCapac…