(冒泡,选择,插入,希尔,快速,归并,堆排)
周末加班学习C++,打算用C++写七大经典排序代码。
发现3个月前自己写的七大经典排序代码(C Language)突然运行出错。

Makefile内容

testmysort: testmysort.c mysortlib.c mysortlib.h
# gcc testmysort.c mysortlib.c -o testmysort
gcc $< mysortlib.c -o $@
./$@

直接make几次总是core dumped错误。

这时我用git已有快一年经验,当时就后悔没用git来跟踪代码变化。如果用git,只要git diff 就可以快速定位导致影响的代码。
git真的是一个好工具,一次又一次发现。
我想以后我的代码都要这样管理。

没办法,自己debug,看代码,多谢前面自己做到工作,本来就输出了很多有用的消息,竟然没动用gdb。
1、定位问题,#define sortfunc _msort可知与归并排序函数有关,换成_qsort测试没问题,因此肯定_msort()有问题。
2、排除问题:问题在一个地方,合并时的条件判断出错了!debug过程在源码中有记录,改5次后成功。

void _msort(int arr[], int len)
{/* 2014-03-26 一次写好就对!!???==>>那是错觉@2014-08-10@*/
if (len<10){
_isort(arr, len);
return;
}
/* sorting each half part */
int m = len/2;
_msort(arr, m);
_msort(arr+m, len-m); /* merge, use memery outside */
int *arrtmp = (int*)malloc(len*sizeof(int));
int i=0,j=m,k=0; /* i是前游标,j是后游标,k申请内存的游标*/
for (k=0;k<len;k++){
/* bug v1 发现排序结果是规则的波峰型
* 偶尔运行时core dumped
if (arr[i]<arr[j]){
arrtmp[k] = arr[i++];
}else{
arrtmp[k] = arr[j++];
}
*/
/* bug v2 不全面(i<m && arr[i]>=arr[j] 条件下:如果j>=m,也该执行分支一)
* 偶尔运行时core dumped
if (i<m && arr[i]<arr[j]){
arrtmp[k] = arr[i++];
}else{
arrtmp[k] = arr[j++];
}
*/
/* bug v3 已接近正确,仅错在j>=m,一开始没发现
if (j>=m || i<m && arr[i]<arr[j]){
arrtmp[k] = arr[i++];
}else{
arrtmp[k] = arr[j++];
}
*/
/* bug v4 括号没改变逻辑
if (j>=m || (i<m && arr[i]<arr[j])){
arrtmp[k] = arr[i++];
}else{
arrtmp[k] = arr[j++];
}
*/
/* 总结:
* 一、if中的两分支具有*互换性*,所以地位相等,推出走每个分支的期望必须都是1/2。
* 二、概率公式:P(A或B) = P(A) + P(B) - P(A且B)
* 三、考虑到隐藏的附加条件:j>=len与i>=m不可同时成立,推出j>=len时必i<m,同理i>=m时必j<len
*/
if (j>=len || (i<m && arr[i]<arr[j])){
arrtmp[k] = arr[i++]; /* 走此路概率:1/2+1/2*1/2-1/2*1/2 = 1/2 */
}else{ /* 等价:if (i>=m || (j<len && arr[i]>= arr[j])) */
arrtmp[k] = arr[j++];
}
}
for (k=0;k<len;k++){
arr[k] = arrtmp[k];
}
free(arrtmp);
}

3、testmysort.c是个辅助测试的模块,调用mysortlib.c中的函数,通过宏#define sortfunc ×××每次测试一个。输出信息对估量排序函数有比较大的帮助。
输出类似如下:

< [ 35687,3377,49990,58351,12237 ]
> [ 3377,12237,35687,49990,58351 ]
---------------------------------
< [ 35687,3377,49990,58351,12237,64841,62230,41585,52580,18926 ]
> [ 3377,12237,18926,35687,41585,49990,52580,58351,62230,64841 ]
---------------------------------
< [ 35687,3377,49990,58351,12237,64841,62230,41585,52580,18926,46382,40611,49858,20665,2984,60676,24442,59752,8288,21844 ]
> [ 2984,3377,8288,12237,18926,20665,21844,24442,35687,40611,41585,46382,49858,49990,52580,58351,59752,60676,62230,64841 ]
---------------------------------
< [ 35687,3377,49990,58351,12237,64841,62230,41585,52580,18926,46382,40611,49858,20665,2984,60676,24442,59752,8288,21844,51890,13460,36039,20282,49485,9391,25777,41009,52448,1089,7594,22600,4466,57585,15416,49470,24123,12112,25521,43935 ]
> [ 1089,2984,3377,4466,7594,8288,9391,12112,12237,13460,15416,18926,20282,20665,21844,22600,24123,24442,25521,25777,35687,36039,40611,41009,41585,43935,46382,49470,49485,49858,49990,51890,52448,52580,57585,58351,59752,60676,62230,64841 ]
---------------------------------
sorted 1000000 used: 0.327s
sorted 2000000 used: 0.676s
sorted 4000000 used: 1.398s
sorted 8000000 used: 2.883s
sorted 1000000 used: 0.176s when same
sorted 2000000 used: 0.371s when same
sorted 4000000 used: 0.782s when same
sorted 8000000 used: 1.639s when same
sorted 1000000 used: 0.176s (inc) | sort 1000000 used: 0.177s (desc)
sorted 2000000 used: 0.371s (inc) | sort 2000000 used: 0.371s (desc)
sorted 4000000 used: 0.780s (inc) | sort 4000000 used: 0.779s (desc)
sorted 8000000 used: 1.639s (inc) | sort 8000000 used: 1.635s (desc)

4、最后git commit好,心里比较踏实了。

 

【实习记】2014-08-10(上)代码跟踪git的想法+归并排序的debug过程的更多相关文章

  1. openstack学习笔记一 虚拟机启动过程代码跟踪

    openstack学习笔记一 虚拟机启动过程代码跟踪 本文主要通过对虚拟机创建过程的代码跟踪.观察虚拟机启动任务状态的变化,来透彻理解openstack各组件之间的作用过程. 当从horizon界面发 ...

  2. 在Ubuntu 12.10 上安装部署Openstack

    OpenStack系统有几个关键的项目,它们能够独立地安装但是能够在你的云计算中共同工作.这些项目包括:OpenStack Compute,OpenStack Object Storage,OpenS ...

  3. trace与代码跟踪服务

    首先开篇引用<MVC2 2 in action>里面一段关于这个跟踪服务的话 When you called Trace.Write() in Web Forms, you were in ...

  4. 六星经典CSAPP-笔记(12)并发编程(上)

    六星经典CSAPP-笔记(12)并发编程(上) 1.并发(Concurrency) 我们经常在不知不觉间就说到或使用并发,但从未深入思考并发.我们经常能"遇见"并发,因为并发不仅仅 ...

  5. 2014年10月底/终于/HTML5定稿……/技术从来不会成为发展的绝对瓶颈/反而商业成了无法逾越的鸿沟【转载+整理】

    原文地址 本文内容 一.HTML5 诞生 二.HTML5 第一阶段: Web 增强与打破垄断 三.HTML5 第二阶段: 移动互联网 四.HTML5 这回真的来了 五.颠覆原生 App 六.还有什么会 ...

  6. NASA的10条代码编写原则

    NASA的10条代码编写原则 作者: Gerard J. Holzmann 来源: InfoQ 原文链接 英文原文:NASA's 10 Coding Rules for Writing Safety ...

  7. 实习记——《Rethink》

    版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/gmszone/article/details/30045055 最终能够在和自己的电脑上写下这些字了 ...

  8. 解Bug之路-记一次线上请求偶尔变慢的排查

    解Bug之路-记一次线上请求偶尔变慢的排查 前言 最近解决了个比较棘手的问题,由于排查过程挺有意思,于是就以此为素材写出了本篇文章. Bug现场 这是一个偶发的性能问题.在每天几百万比交易请求中,平均 ...

  9. AM335x tscadc platform driver 相关代码跟踪

    TI AM335x ti am335x_tsc.c 代码跟踪 在kernel 首层目录: 先运行make ARCH=arm tags 这个作用是建立tags文件,只含有arm架构的,利用ctag即可进 ...

随机推荐

  1. HDU-1035 Robot Motion

    http://acm.hdu.edu.cn/showproblem.php?pid=1035 Robot Motion Time Limit: 2000/1000 MS (Java/Others)   ...

  2. 一键清除cvs/svn 目录

    步骤一.编写注册表脚本      新建一个文本文件,把下面的代码COPY进去,保存为delSVNorCVS.reg(可直接从本文附件中下载) Windows Registry Editor Versi ...

  3. TOMCAT启动完成但是ECLIPSE仍然显示starting....

    最近重新部署了一个TOMCAT服务,但是启动碰到个问题,虽然TOMCAT控制台已显示启动成功,但是ECLIPSE右下角仍然一直显示STARTING,最后TOMCAT超时,启动失败. 之前以为是拷贝工程 ...

  4. python实现应用程序在右键菜单中添加打开方式

    最近项目组开发的一个小工具想要在右键菜单中添加打开方式,以有道云笔记为例进行了需求拆解和代码编写 1.需求拆解: 如何实现手动添加右键菜单的打开方式: Step1:打开注册表编辑器,Win+R-> ...

  5. C# WinForm 透明控件 PictureBox透明 分类: WinForm 2014-07-30 13:27 591人阅读 评论(0) 收藏

    1.要实现C# WinForm中的控件与背景的透明,可以通过设置控件的BackColor属性为Transparent,同时设置其父控件.因为在C#中,控件的透明指对父窗体透明.如果不设置Parent属 ...

  6. 构建ASP.NET MVC4+EF5+EasyUI+Unity2.x注入的后台管理系统(6)-Unity 2.x依赖注入by运行时注入[附源码]

    原文:构建ASP.NET MVC4+EF5+EasyUI+Unity2.x注入的后台管理系统(6)-Unity 2.x依赖注入by运行时注入[附源码] Unity 2.x依赖注入(控制反转)IOC,对 ...

  7. UML进行Linux内核调试

    http://www.lenky.info/ http://blog.csdn.net/ztz0223/article/details/7874759 http://user-mode-linux.s ...

  8. [转] android 中 任务、进程和线程的区别

    PS: handler的目的是在组件进程中开辟一个线程作为消息的poller,收到消息后可以更新Activity中的控件(特殊的view) 任务.进程和线程     关于Android中的组件和应用, ...

  9. Objective-C:KVC

    1 概述 1.1 访问方法 Key-value coding(KVC)是一种间接访问对象属性的机制,类似键值对,通过名字(或键)可以直接获得对象的属性值.事实上,key-value coding定义了 ...

  10. 9.29noip模拟试题

    环上的游戏(cycle) 有一个取数的游戏.初始时,给出一个环,环上的每条边上都有一个非负整数.这些整数中至少有一个0.然后,将一枚硬币放在环上的一个节点上.两个玩家就是以这个放硬币的节点为起点开始这 ...