c++链表归并排序的迭代版本
之前用js写了个归并排序非递归版,而这一次,c++封装链表的时候也遇到了一个归并排序的接口。邓老师实现了递归版本的归并排序,但是递归的调用函数栈的累积是很占内存空间的。于是乎,那试试在链表结构上实现以下归并排序吧。但是一旦开始,就遇到难题了,在链表下,我们无法按索引访问,所以,在迭代过程中,左右序列就无法很好的用o(1)时间就解决。先看看我实现的代码吧,初步测试没问题,如果有什么问题,希望大神指出,不知为何,用c++写东西总觉得哪里有问题,即使程序可以运行。
template<typename T>
void List<T>::mergeSort(Posi(T) p, int n)
{
Posi(T) h = p->pred;
Posi(T) r_p;
for (int i = ; i < n; i*=)
{
p = h->next;
r_p = p;
int count = ;
while (count*<n)
{
int r_len = ;
while (r_len<i&&r_p!=trailer)
{
r_p = r_p->next;
r_len++;
count++; //右链表移动的次数
}
if (r_p == trailer)
{
break;
}
if (count*>n)
{
r_len = n%i;
}
p = merge(p, i, r_p, r_len);
r_p = p;
}
}
}
这是主干程序,大体思路是从一开始的一个一个归并并,到四个四个合并,到最后二路归并,一开始,因为这个接口是在给定位置的p点已经后面的n-1个元素进行排序,所以我无法直接使用header,于是,用p点的前一个左右目前头结点,为什么这么做呢,一开始的时候,我是打算用p左右每次迭代的开始,如果你对迭代版归并熟悉的话,那么应该很容易理解,每一次合并完所有链表上的分列表,都需要重新从第一个结点开始,然后继续合并下一个循环,而因为链表是动态的,很有可能,一次合并后,你的p就不是现在的第一个结点了,这是h变量的含义,他是一个头结点,当然,按照这个接口,尾结点也需要重新定义,然而定义尾结点需要n时间复杂度,所以这里,我的代码是假设n个的下一个刚好是尾结点。之后是需要合并的右链表的位置,一开始是和左链表重合,然后通过循环来达到它的位置,count变量是r_p走的步数,这个步数两倍是可能超过长度n的,一旦超过,说明了r_p时间上已经到达了末尾,这时候一个级数的合并已经完成,可以进入下一个迭代归并。
而r_len = n%i;的意义是如果越界,那么需要取得余数,是正好等于尾巴的长度。
如果不这么做,那么merge函数可能链表越界,接下来看看别的部件。
template<typename T>//包括p在内的n个元素
Posi(T) List<T>::merge(Posi(T) left_p, int left_len,Posi(T) right_p, int right_len)
{
while (left_len > &&right_len > )
{
if (left_p->data < right_p->data)
{
left_p = left_p->next;
left_len--;
}
else
{
Posi(T) temp_p = right_p->next;
insertAsPre(right_p, left_p);
right_p = temp_p;
right_len--;
}
}
while (right_len-->)
{
right_p = right_p->next;
}
return right_p;
}
template<typename T> //重写接口,以至于可以插入已有节点,同一个链表的节点
Posi(T) List<T>::insertAsPre(Posi(T) p, Posi(T) n)
{
p->next->pred = p->pred;
p->pred->next = p->next;
p->pred = n->pred;
p->next = n;
n->pred = p;
p->pred->next = p;
return p;
}
现在可以分析下这个算法的时间复杂度了。
最外层的for循环应该是logn,这个很好理解,而为了得到左指针的while循环必然是n/2,merge函数花费是n,while和merge同级,所以花费是3n/2.
如果算上这个接口尾指针。应该是o(3n/2*logn+n),整体是比数组的归并要慢一点,但是还是在同一个数量级的,这是值得开心的。
c++链表归并排序的迭代版本的更多相关文章
- STL 源码分析《1》---- list 归并排序的 迭代版本, 神奇的 STL list sort
最近在看 侯捷的 STL源码分析,发现了以下的这个list 排序算法,乍眼看去,实在难以看出它是归并排序. 平常大家写归并排序,通常写的是 递归版本..为了效率的考虑,STL库 给出了如下的 归并排序 ...
- SICP 换零钱的迭代版本
看到换零钱方式统计这里, 书中给出了递归的实现但没有给出迭代版本说要留给读者作为挑战, 既然说是作为挑战那肯定是能解决的,在想了一天无果之后最终在别人博客的帮助下终于实现了迭代的版本...也算是经历坎 ...
- [Leetcode]148. 排序链表(归并排序)
题目 在 O(n log n) 时间复杂度和常数级空间复杂度下,对链表进行排序. 示例 1: 输入: 4->2->1->3 输出: 1->2->3->4 示例 2: ...
- 21 Merge Two Sorted Lists(两链表归并排序Easy)
题目意思:对两个递增链表进行归并排序 思路:没什么好说的,二路归并 /** * Definition for singly-linked list. * struct ListNode { * int ...
- [LeetCode]148. Sort List链表归并排序
要求时间复杂度O(nlogn),空间复杂度O(1),采用归并排序 传统的归并排序空间复杂度是O(n),原因是要用一个数组表示合并后的数组,但是这里用链表表示有序链表合并后的链表,由于链表空间复杂度是O ...
- c#循环迭代匿名类链表(可迭代的匿名类)
Main(){ //为什么?object是基类啊!! //报错.不能从List<anonymous>换成List<object>. //var q=(List<objec ...
- git在多迭代版本的应用
名词解释: 1.迭代: 就是对于项目功能的一个分类.如项目需要新增一个地图功能,则地图功能是一个迭代. 2.gitlab机器人 操作: 1.如果将要进行一个新功能的开发,从稳定分支上拉取创建一个新的分 ...
- [LeetCode] 148. 排序链表 ☆☆☆(归并排序)
148.排序链表 描述 在 O(n log n) 时间复杂度和常数级空间复杂度下,对链表进行排序. 示例 1: 输入: 4->2->1->3输出: 1->2->3-> ...
- Python 3.9.0 首个迭代版本发布了
Python 3.9.0 alpha 1 发布了,这是 3.8 之后的首个 3.9 系列版本. ! 官方没有介绍新特性,也没有添加新模块,但是以下模块有所改进: ast asyncio curses ...
随机推荐
- MySQL数据库相关命令
1.命令:show create table 表名 功能:获取建表语句 2.命令:desc 表名 功能:展示表字段及其类型
- ready是先执行的,load后执行,DOM文档的加载步骤
在jq中在文档载入完毕后有这几种方式去执行指定函数: $(document).ready(function() { // ...代码... }); //document ready 简写 $(func ...
- 20145337《Java程序设计》第八周学习总结
20145337<Java程序设计>第八周学习总结 教材学习内容总结 15.1日志 15.1.1日志API简介 使用日志的起点是logger类,logger实例的创建有许多要处理的要素,必 ...
- 不正确使用WeakHashMap引起的卡死
公司的jenkins今天出了一点问题,起来以后,总是处于等待状态,所有的任务无法正常加载.登陆界面也出不了.而且cpu占用率100% 把线程导出来,看到: “Loading job NMS_Patch ...
- 【iCore3 双核心板_FPGA】实验二十:基于FIFO的ARM+FPGA数据存取实验
实验指导书及代码包下载: http://pan.baidu.com/s/1cmisnO iCore3 购买链接: https://item.taobao.com/item.htm?id=5242294 ...
- openstack-flat 网络问题
使用packstack安装,版本为M,默认为vxlan网络, 操作系统为centos7. 由于学习的需要修改为flat网络. 首先修改配置文件: /etc/neutron/plugins/ml2/ml ...
- Java 集合快速失败异常
快速失败 在JDK中,查看集合有很多关于快速失败的描述: 注意,此实现不是同步的.如果多个线程同时访问一个哈希映射,而其中至少一个线程从结构上修改了该映射,则它必须 保持外部同步.(结构上的修改是指添 ...
- LINUX VI 常用命令
vi 打开或新建 vi filename 打开或新建文件 并将光标置于第一行首 光标 ) 光标移至句尾 ( 光标移至句首 屏幕翻滚类命令 Ctrl+u 向文件首翻半屏 Ctrl+d 向文件尾翻半屏 ...
- Mac OS 电脑播放 iPhone音乐
http://apple.stackexchange.com/questions/6173/can-i-play-audio-from-my-iphone-on-my-mac Simple, and ...
- Git 学习01
一.下载并安装git bash 双击打开出现命令窗口 创建一个版本库非常简单,首先,选择一个合适的地方,创建一个空目录: cd F: mkdir learngit pwd F/learngit 显示当 ...