今天看剑指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++实现及时间效率对比。。的更多相关文章

  1. 快排(golang实现) 递归方法

    递归方法,逻辑简洁清晰.这个算法还是很重要的,需要重点记忆理解,面试经常考手写.据说是与傅里叶变换等并称“20世纪十大算法”.https://blog.csdn.net/v_JULY_v/articl ...

  2. 先贴上代码:Random快排,快排的非递归实现

    设要排序的数组是A[0]……A[N-1],首先任意选取一个数据(通常选用数组的第一个数)作为主元,然后将所有比它小的数都放到它前面,所有比它大的数都放到它后面,这个过程称为一趟快速排序.值得注意的是, ...

  3. C语言 | 栈的应用 | 非递归栈实现快排

    /* 非递归栈实现快排 */ #include <stdio.h> #include <math.h>> #include <malloc.h> #inclu ...

  4. java面试-公平锁/非公平锁/可重入锁/递归锁/自旋锁谈谈你的理解

    一.公平锁/非公平锁/可重入锁/递归锁/自旋锁谈谈你的理解 公平锁:多个线程按照申请的顺序来获取锁. 非公平锁:多个线程获取锁的先后顺序与申请锁的顺序无关.[ReentrantLock 默认非公平.s ...

  5. 数据结构(3) 第三天 栈的应用:就近匹配/中缀表达式转后缀表达式 、树/二叉树的概念、二叉树的递归与非递归遍历(DLR LDR LRD)、递归求叶子节点数目/二叉树高度/二叉树拷贝和释放

    01 上节课回顾 受限的线性表 栈和队列的链式存储其实就是链表 但是不能任意操作 所以叫受限的线性表 02 栈的应用_就近匹配 案例1就近匹配: #include <stdio.h> in ...

  6. 排序 之 快排、归并、插入 - <时间复杂度>----掌握思想和过程

    俗话说:天下武功无坚不破,唯快不破.对于算法当然也是要使用时间最短.占用空间最小的算法来实现了. 注意:我代码里面打的备注仅供参考,建议不要背模板(因为没有固定的模板),可以写一个数列按着代码跑两圈或 ...

  7. Java 排序(快排,归并)

    Java 排序有Java.util.Arrays的sort方法,具体查看JDK API(一般都是用快排实现的,有的是用归并) package yxy; import java.util.Arrays; ...

  8. java——快排、冒泡、希尔、归并

    直接贴代码 快排: public class Test { private static void sort(int[] nums){ if(nums == null || nums.length = ...

  9. Java常见的几种排序算法-插入、选择、冒泡、快排、堆排等

    本文就是介绍一些常见的排序算法.排序是一个非常常见的应用场景,很多时候,我们需要根据自己需要排序的数据类型,来自定义排序算法,但是,在这里,我们只介绍这些基础排序算法,包括:插入排序.选择排序.冒泡排 ...

随机推荐

  1. 页面内容不满屏幕高度时,footer底部显示

    底部高度固定的情况下 <style> body,html{ height: 100%; margin: 0; } .content{ min-height: 100%; padding-b ...

  2. python多项式拟合:np.polyfit 和 np.polyld

    python数据拟合主要可采用numpy库,库的安装可直接用pip install numpy等. 1. 原始数据:假如要拟合的数据yyy来自sin函数,np.sin import numpy as ...

  3. js--javascript学习

    js -- javascript ECMAscript5 ECMAscript6 -- vue.js react .. 由三个部分组成 1 ECMAscript5的核心 js语言 2 BOM 浏览器对 ...

  4. Centos7更改网卡名称Eth0

    标签: Centos7更改网卡名称 2016-12-06 21:55 8737人阅读 评论(1) 收藏 举报  分类: linux(6)  作者同类文章X 版权声明:本文为博主原创文章,未经博主允许不 ...

  5. 记录 shell学习过程(4)for 循环

    1. for in ` #seq 生成从1到10 如果生成从10到1则写作 seq 10 -1 1 do echo $i done for in 也可以循环出字符串 for i in where is ...

  6. K3修改字段名

    在K3的BOS中,自定义字段之后我们往往会修改字段名,便于记忆和理解,但是修改字段名之后,只是数据库中的字段名被修改了,BOS中的字段标识并没有被修改,可以通过以下语句将标识和字段名改成一致. sel ...

  7. 边捆绑: Content Importance Based Edge Bundling for Graph Visualization

    Problem 当图所要表达的信息较多时, 图中可能会充满交叉的线[1-2], 甚至整个显示空间都被点.线所覆盖, 这时想通过观察来获取图中的重要信息将会变得非常困难, 这种现象称为图的视觉混乱. K ...

  8. 在已部署好的docker环境下配置nginx项目路径

    第一步:申请一个docker连接账号,可以借用putty工具,如果使用sublime,可以下载sftp插件,上传.下载来同步你线上线下的文件: 第二步:修改nginx区域配置文件,在conf文件夹里放 ...

  9. 【转载】各种SQL在PIG中实现

    转自:http://guoyunsky.iteye.com/blog/1317084 我这里以Mysql 5.1.x为例,Pig的版本是0.8 同时我将数据放在了两个文件,存放在/tmp/data_f ...

  10. C正数负数的原码补码反码以及内存地址分析

    #include<stdio.h> void swap(int a, int b); void main1(){ int i = 10; //正数的原码 00000000 00000000 ...