Algorithms in C第一章笔记

归并——查找算法

 

程序1.1

#include <stdio.h>

#define N 20

 

int main(void)

{

         int i,p,q,t,id[N];

         for(i=0;i<N;i++)id[i]=i;

         while(scanf("%d%d",&p,&q)==2)

         {

                   if(id[p]==id[q])continue;

                   for(t=id[p],i=0;i<N;i++)

                   if(id[i]==t)id[i]=id[q];

                   for(i=0;i<N;i++)

                   printf("%d ",id[i]);

                  

         }       

         return 0;

}

程序的执行结果如下:

 

 

思路:首先我们把每个点所在集合初始化为其自身。比如数组下标2,他对于一个名为2的集合。【在同一个集合里面表示连通】

然后我们接受2个数字

(scanf("%d%d",&p,&q)

 

如果所示,1和2 ,此时下标1对应的集合是集合1,下标2对应的集合为集合2,很明显他们不在同一个集合中,所以为了将他们连通,就要将他们的集合归并。我们遍历数组,把所有名为p的元素改为q。

 

 对于第一个输入来说,下标1对应的集合1被改成了集合2 。

 

分析:这个算法判断连通性很便捷,但是如果让他归并呢?慢的要死,对吧。所以这个算法被称为快速查找算法,不适合合并,下面是这个算法的图形化表示,此图中,对于查找而言,同一个集合里的元素只需要一步就能查找到“根”,所以速度非常快。

 

 

我们考虑下面这个算法,来优化归并的速度

 

   思路:在一个没有环的结构中,每个对象指向同一个集合的另一个对象。要确定两个对象是否在同一个集合里,只要跟随每个对象的“指针”,直到到达指向自身的一个对象。当且仅当这个过程使两个对象到同一个对象时,这二个对象在同一个集合中。否则,便不在。此时构造合并,只需将一个对象链接到另一个对象就行。

 

程序1.2

 

#include <stdio.h>

#define N 20

 

int main(void)

{

         int i,p,q,j,id[N];

         for(i=0;i<N;i++)id[i]=i;

         while(scanf("%d%d",&p,&q)==2)

         {

                   for(i=p;i!=id[i];i=id[i]);  //一直找到根节点

                   for(j=q;j!=id[j];j=id[j]);

                   if(i==j)continue;

                   id[i]=j;

        

                   for(i=0;i<N;i++)

                   printf("%d ",id[i]);

                   printf("\n");

         }       

         return 0;

}

 

 

 

 

 

 

 

一图胜千言。显而易见,此时查找的速度变慢了,但是归并的速度变的很快。

但是仍然有弊端

 

 

 

 

 

 

 

 

 

 

 

 

 

 

加权快速合并算法:

程序1.3

 

#include<stdio.h>

#define N 20

 

int main(void)

{

         int i,j,p,q,id[N],sz[N];

         for(i=0;i<N;i++)  //初始化

         {

                   id[i]=i;

                   sz[i]=1;

         }

        

         while(scanf("%d%d",&p,&q)==2)

         {

                   for(i=p;i!=id[i];i=id[i]);

                   for(j=q;j!=id[j];j=id[j]);

                   if(i==j)continue;

                   if(sz[i]<sz[j])

                   {

                            id[i]=j;sz[j]+=sz[i];   

                            printf("sz[j]= %d\n",sz[j]);

                   }

                   else{

                            id[j]=i;sz[i]+=sz[j];

                            printf("sz[i]= %d\n",sz[i]);

                   }

                   for(i=0;i<N;i++)

                   printf("%d ",id[i]);

                   printf("\n");

                  

         }       

        

}

 

 

 

分析:利用了Sz数组记录了每个树“根”开始的树的大小。一直是把小的合并到打的上面。

 

 

路径压缩,作者不经想到,怎么样才能查找和归并一样快呢?

于是给了个神级代码。初读算法的我,惊叹不已。

#include<stdio.h>

#define N 20

 

int main(void)

{

         int i,j,p,q,id[N],sz[N];

         for(i=0;i<N;i++)  //初始化

         {

                   id[i]=i;

                   sz[i]=1;

         }

        

         while(scanf("%d%d",&p,&q)==2)

         {

                   for(i=p;i!=id[i];i=id[i]) //直接跳到下一个节点,实现路径压缩

                            id[i] = id[id[i]];

                   for(j=q;j!=id[j];j=id[j])

                            id[j] = id[id[j]];

                   if(i==j)continue;

                   if(sz[i]<sz[j])

                   {

                            id[i]=j;sz[j]+=sz[i];   

                            printf("sz[j]= %d\n",sz[j]);

                   }

                   else{

                            id[j]=i;sz[i]+=sz[j];

                            printf("sz[i]= %d\n",sz[i]);

                   }

                   for(i=0;i<N;i++)

                   printf("%d ",id[i]);

                   printf("\n");

                  

         }       

        

}

转载于:https://www.cnblogs.com/You0/p/4797550.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
《算法:C语言实现(第1-4部分)基础知识、数据结构、排序及搜索(原书第3版)》细腻讲解计算机算法的C语言实现。全书分为四部分,共16章。包括基本算法分析原理,基本数据结构、抽象数据结构、递归和树等数据结构知识,选择排序、插入排序、冒泡排序、希尔排序、快速排序方法、归并和归并排序方法、优先队列与堆排序方法、基数排序方法以及特殊用途的排序方法,并比较了各种排序方法的性能特征,在进一步讲解符号表、树等抽象数据类型的基础上,重点讨论散列方法、基数搜索以及外部搜索方法。书中提供了用C语言描述的完整算法源程序,并且配有丰富的插图和练习,还包含大量简洁的实现将理论和实践成功地相结合,这些实现均可用在真实应用上。 《算法:C语言实现(第5部分)图算法(原书第3版)》是深入论述算法的三卷本教程《算法:C语言实现》(第3版)中的第二卷——图算法。作者在这次修订中重写了许多内容,增加了数千个新练习、数百个新图表、数十个新程序,并对图表和程序做了详尽的注释说明。新版中不仅涵盖了新的主题,而且还提供了对许多经典算法的更充分的解释,包括图的性质、图搜索、有向图、最小生成树、最短路径和网。《算法:C语言实现(第5部分)图算法(原书第3版)》涵盖了足够的基本内容及较详细的图算法高级主题,既可单独用作数据结构与算法课程的教材,也可与第一卷(第1~4部分)结合使用。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值