归并非递归、快排递归及非递归的C++实现及时间效率对比。。
今天看剑指offer突然发现下学期都要去面试了,还没自己实现过快排非递归和归并非递归,这怎么能行呢,于是就写了一下。
(虽然有点卡壳,又回去翻了下算导,还是顺利写出来了)
先放图: 一亿数据量:

#pragma warning(disable:4996)
#include <iostream>
#include <string>
#include <cctype>
#include<vector>
#include<list>
#include<cstring>
#include<typeinfo>
#include<set>
#include<map>
#include<deque>
#include<regex>
#include<sstream>
#include<cstdlib>
#include<queue>
#include<stdlib.h>
#include<stdio.h>
#include<stack>
#include<algorithm>
#include<thread>
#include<mutex>
#include<assert.h>
using namespace std;
void MergeSort(vector<int>& nums)
{
if (nums.empty())
{
return;
}
int len = nums.size();
vector<int> temp(len, 0); //临时数组,下面会用
for (int merge_step = 1; merge_step < len; merge_step <<= 1)//当merge_step为k,表示merge两个长度k的有序序列成一个长度2k的序列
{
for (int begin = 0; begin < len - merge_step; begin += (merge_step << 1))
{
int le = begin, ri = begin + merge_step; //左右区间起点
int le_end = ri, ri_end = min(ri + merge_step, len); //左右区间终点(半闭半开区间)
int i = 0;
while (le < le_end and ri < ri_end)
{
if (nums[le] < nums[ri])
{
temp[i] = nums[le];
++le;
}
else
{
temp[i] = nums[ri];
++ri;
}
++i;
}
while (le < le_end)
{
temp[i] = nums[le];
++i;
++le;
}
while (ri < ri_end)
{
temp[i] = nums[ri];
++i;
++ri;
}
for (int j = 0; j < i; ++j) //写回原数组
{
nums[begin + j] = temp[j];
}
}
}
}
void InitiateRandomVector(vector<int>& nums)
{
srand(0);
for (auto& num : nums)
{
num = rand();
}
}
int partition(vector<int>&nums,int le,int ri) //划分区间函数
{
srand(0);
int rand_num=le+rand()%(ri-le); //随机生成一个索引
swap(nums[ri],nums[rand_num]); //换到结尾
int stable=nums[ri];
int i=le-1,j=le;
while(j<ri)
{
if(nums[j]<stable)
{
int temp=nums[j];
nums[j]=nums[++i];
nums[i]=temp; //i存的都是小于基准数的值
}
++j;
}
swap(nums[i+1],nums[ri]);
return i+1; //返回基准数的索引
}
void quick_sort(vector<int>& nums,int le,int ri) //[le,ri]闭区间
{
if(le>=ri)
{
return;
}
int mi=partition(nums,le,ri);
quick_sort(nums,le,mi-1);
quick_sort(nums,mi+1,ri);
}
void QuickSort_Recursion(vector<int>& nums) //递归版
{
quick_sort(nums,0,nums.size()-1);
}
void QuickSort_Iteration(vector<int>&nums) //迭代版
{
if(nums.empty())
{
return;
}
stack<int> sta;
sta.push(0);
sta.push(nums.size()-1);
int le,ri,mi;
while(!sta.empty())
{
ri=sta.top(); //后push的右边界故先pop
sta.pop();
le=sta.top();
sta.pop();
mi=partition(nums,le,ri);
if(mi-1>le)
{
sta.push(le);
sta.push(mi-1);
}
if(mi+1<ri)
{
sta.push(mi+1);
sta.push(ri);
}
}
}
int main(int argc, char** argv)
{
cout<<"Input data size:";
int siz;
cin>>siz;
vector<int>nums(siz);
cout<<"Data size:"<<siz<<endl;
InitiateRandomVector(nums);
int t1=time(0);
MergeSort(nums);
cout<<"My merge sort time:"<<time(0)-t1<<"s"<<endl;
InitiateRandomVector(nums);
t1=time(0);
QuickSort_Recursion(nums);
cout<<"My quick sort(recursion) time:"<<time(0)-t1<<"s"<<endl;
InitiateRandomVector(nums);
t1=time(0);
sort(nums.begin(),nums.end());
cout<<"STL sort time:"<<time(0)-t1<<"s"<<endl;
InitiateRandomVector(nums);
t1=time(0);
QuickSort_Iteration(nums);
cout<<"My quick sort(iteration) time:"<<time(0)-t1<<"s"<<endl;
getchar();
return 0;
}
归并非递归、快排递归及非递归的C++实现及时间效率对比。。的更多相关文章
- 快排(golang实现) 递归方法
递归方法,逻辑简洁清晰.这个算法还是很重要的,需要重点记忆理解,面试经常考手写.据说是与傅里叶变换等并称“20世纪十大算法”.https://blog.csdn.net/v_JULY_v/articl ...
- 先贴上代码:Random快排,快排的非递归实现
设要排序的数组是A[0]……A[N-1],首先任意选取一个数据(通常选用数组的第一个数)作为主元,然后将所有比它小的数都放到它前面,所有比它大的数都放到它后面,这个过程称为一趟快速排序.值得注意的是, ...
- C语言 | 栈的应用 | 非递归栈实现快排
/* 非递归栈实现快排 */ #include <stdio.h> #include <math.h>> #include <malloc.h> #inclu ...
- java面试-公平锁/非公平锁/可重入锁/递归锁/自旋锁谈谈你的理解
一.公平锁/非公平锁/可重入锁/递归锁/自旋锁谈谈你的理解 公平锁:多个线程按照申请的顺序来获取锁. 非公平锁:多个线程获取锁的先后顺序与申请锁的顺序无关.[ReentrantLock 默认非公平.s ...
- 数据结构(3) 第三天 栈的应用:就近匹配/中缀表达式转后缀表达式 、树/二叉树的概念、二叉树的递归与非递归遍历(DLR LDR LRD)、递归求叶子节点数目/二叉树高度/二叉树拷贝和释放
01 上节课回顾 受限的线性表 栈和队列的链式存储其实就是链表 但是不能任意操作 所以叫受限的线性表 02 栈的应用_就近匹配 案例1就近匹配: #include <stdio.h> in ...
- 排序 之 快排、归并、插入 - <时间复杂度>----掌握思想和过程
俗话说:天下武功无坚不破,唯快不破.对于算法当然也是要使用时间最短.占用空间最小的算法来实现了. 注意:我代码里面打的备注仅供参考,建议不要背模板(因为没有固定的模板),可以写一个数列按着代码跑两圈或 ...
- Java 排序(快排,归并)
Java 排序有Java.util.Arrays的sort方法,具体查看JDK API(一般都是用快排实现的,有的是用归并) package yxy; import java.util.Arrays; ...
- java——快排、冒泡、希尔、归并
直接贴代码 快排: public class Test { private static void sort(int[] nums){ if(nums == null || nums.length = ...
- Java常见的几种排序算法-插入、选择、冒泡、快排、堆排等
本文就是介绍一些常见的排序算法.排序是一个非常常见的应用场景,很多时候,我们需要根据自己需要排序的数据类型,来自定义排序算法,但是,在这里,我们只介绍这些基础排序算法,包括:插入排序.选择排序.冒泡排 ...
随机推荐
- Mysql单实例数据库安装
第1章 MySQL数据库安装 在当今的大中型互联网企业里,MySQL数据库服务几乎都是运行在Linux系列操作系统上,当然,你在可以运行在Windows/Unix等商业操作系统上,大中型互联网企业使用 ...
- centos7下自动备份mysql数据库
环境:centos7 .mysql5.7 第一步:创建自动备份脚本文件 mkdir backup cd backup touch backup.sh 第二步:在备份脚本文件中写入备份脚本 vim ba ...
- pytest学习3-断言
断言: 一个标准的用例都包含了断言,编写pytest自动化脚本的时候,也需要设置断言 pytest常用断言比较大小与是否相等.是否包含.验证boolean 例子一: 验证是否相等: import py ...
- 转载:android audio flinger
https://blog.csdn.net/innost/article/details/6142812 https://blog.csdn.net/zyuanyun/article/details/ ...
- 嵌入式Linux学习---进程(1)
什么是一个进程?当用户敲入命令执行一个程序的时候,对系统而言,它将启动一个进程.但和程序不同的是,在这个进程中,系统可能需要再启动一个或多个进程来完成独立的多个任务.多进程编程的主要内容包括进程控制和 ...
- Runtime.getRuntime.exec()执行linux脚本导致程序卡死有关问题
Runtime.getRuntime.exec()执行linux脚本导致程序卡死问题问题: 在Java程序中,通过Runtime.getRuntime().exec()执行一个Linux脚本导致程序被 ...
- Windows10 远程桌面连接失败,报CredSSP加密oracle修正错误解决办法
最近Windows10 升级后,发现不能远程连接. 不能访问的都报下面这个错了: 原因:按照提示的微软地址,看了下大致就是服务器端没有更新,而我的win10已经更新了一个安全补丁,如果双方都没有打补丁 ...
- <img src = "..."/>的一个图片上面怎么在放上字
转自:https://zhidao.baidu.com/question/1495805873400412779.html 例子1: html中可以用css相对定位让文字在图片的上面. 1.新建htm ...
- 剑指offer 面试题. 剪绳子
题目描述 给你一根长度为n的绳子,请把绳子剪成整数长的m段(m.n都是整数,n>1并且m>1),每段绳子的长度记为k[0],k[1],...,k[m].请问k[0]xk[1]x...xk[ ...
- NOIP做题练习(day5)
A - 中位数图 题面 题解 先找出题意中的\(b\)所在的位置. 再以这个位置为中心,向右\(for\)一遍有多少个大于/小于该数的数 大于就\(++cs\) 小于就\(--cs\). 因为这个数是 ...