图论中最短路径各算法的精典代码片段-来自ACM

1.最短路径(单源bellman_ford邻接阵形式)

//单源最短路径,bellman_ford算法,邻接阵形式,复杂度O(n^3)

//求出源s到所有点的最短路经,传入图的大小n和邻接阵mat

//返回到各点最短距离min[]和路径pre[],pre[i]记录s到i路径上i的父结点,pre[s]=-1

//可更改路权类型,路权可为负,若图包含负环则求解失败,返回0

//优化:先删去负边使用dijkstra求出上界,加速迭代过程

#define MAXN 200
#define inf1000000000
typedef int elem_t;
 
int bellman_ford(int n,elem_t mat[][MAXN],int s,elem_t* min,int* pre){
         int v[MAXN],i,j,k,tag;
         for (i=0;i<n;i++)
                   min[i]=inf,v[i]=0,pre[i]=-1;
         for (min[s]=0,j=0;j<n;j++){
                   for (k=-1,i=0;i<n;i++)
                            if(!v[i]&&(k==-1||min[i]<min[k]))
                                     k=i;
                   for (v[k]=1,i=0;i<n;i++)
                            if(!v[i]&&mat[k][i]>=0&&min[k]+mat[k][i]<min[i])
                                     min[i]=min[k]+mat[pre[i]=k][i];
         }
         for (tag=1,j=0;tag&&j<=n;j++)
                   for (tag=i=0;i<n;i++)
                            for (k=0;k<n;k++)
                                     if(min[k]+mat[k][i]<min[i])
                                               min[i]=min[k]+mat[pre[i]=k][i],tag=1;
         return j<=n;
}


2.最短路径(单源dijkstra_bfs邻接表形式)

//单源最短路径,用于路权相等的情况,dijkstra优化为bfs,邻接表形式,复杂度O(m)

//求出源s到所有点的最短路经,传入图的大小n和邻接表list,边权值len

//返回到各点最短距离min[]和路径pre[],pre[i]记录s到i路径上i的父结点,pre[s]=-1

//可更改路权类型,但必须非负且相等!

#define MAXN 200
#define inf 1000000000
typedef int elem_t;
struct edge_t{
	int from,to;
	edge_t* next;
};

void dijkstra(int n,edge_t* list[],elem_t len,int s,elem_t* min,int* pre){
	edge_t* t;
	int i,que[MAXN],f=0,r=0,p=1,l=1;
	for (i=0;i<n;i++)
		min[i]=inf;
	min[que[0]=s]=0,pre[s]=-1;
	for (;r<=f;l++,r=f+1,f=p-1)
		for (i=r;i<=f;i++)
			for (t=list[que[i]];t;t=t->next)
				if (min[t->to]==inf)
					min[que[p++]=t->to]=len*l,pre[t->to]=que[i];
}


3.最短路径(单源dijkstra_bfs正向表形式)

//单源最短路径,用于路权相等的情况,dijkstra优化为bfs,正向表形式,复杂度O(m)

//求出源s到所有点的最短路经,传入图的大小n和正向表list,buf,边权值len

//返回到各点最短距离min[]和路径pre[],pre[i]记录s到i路径上i的父结点,pre[s]=-1

//可更改路权类型,但必须非负且相等!

#define MAXN 200
#define inf1000000000
typedef int elem_t;
 
void dijkstra(intn,int* list,int* buf,elem_t len,int s,elem_t* min,int* pre){
         int i,que[MAXN],f=0,r=0,p=1,l=1,t;
         for (i=0;i<n;i++)
                   min[i]=inf;
         min[que[0]=s]=0,pre[s]=-1;
         for (;r<=f;l++,r=f+1,f=p-1)
                   for (i=r;i<=f;i++)
                            for(t=list[que[i]];t<list[que[i]+1];t++)
                                     if(min[buf[t]]==inf)
                                               min[que[p++]=buf[t]]=len*l,pre[buf[t]]=que[i];
}


4.最短路径(单源dijkstra+binary_heap邻接表形式)

//单源最短路径,dijkstra算法+二分堆,邻接表形式,复杂度O(mlogm)

//求出源s到所有点的最短路经,传入图的大小n和邻接表list

//返回到各点最短距离min[]和路径pre[],pre[i]记录s到i路径上i的父结点,pre[s]=-1

//可更改路权类型,但必须非负!

#define MAXN 200
#define inf1000000000
typedef int elem_t;
struct edge_t{
         int from,to;
         elem_t len;
         edge_t* next;
};
 
#define _cp(a,b)((a).d<(b).d)
structheap_t{elem_t d;int v;};
struct heap{
         heap_t h[MAXN*MAXN];
         int n,p,c;
         void init(){n=0;}
         void ins(heap_t e){
                   for(p=++n;p>1&&_cp(e,h[p>>1]);h[p]=h[p>>1],p>>=1);
                   h[p]=e;
         }
         int del(heap_t& e){
                   if (!n) return 0;
                   for(e=h[p=1],c=2;c<n&&_cp(h[c+=(c<n-1&&_cp(h[c+1],h[c]))],h[n]);h[p]=h[c],p=c,c<<=1);
                   h[p]=h[n--];return 1;
         }
};
 
void dijkstra(intn,edge_t* list[],int s,elem_t* min,int* pre){
         heap h;
         edge_t* t;heap_t e;
         int v[MAXN],i;
         for (i=0;i<n;i++)
                   min[i]=inf,v[i]=0,pre[i]=-1;
         h.init();min[e.v=s]=e.d=0,h.ins(e);
         while (h.del(e))
                   if (!v[e.v])
                            for(v[e.v]=1,t=list[e.v];t;t=t->next)
                                     if(!v[t->to]&&min[t->from]+t->len<min[t->to])
                                               pre[t->to]=t->from,min[e.v=t->to]=e.d=min[t->from]+t->len,h.ins(e);
}


5.最短路径(单源dijkstra+binary_heap正向表形式)

//单源最短路径,dijkstra算法+二分堆,正向表形式,复杂度O(mlogm)

//求出源s到所有点的最短路经,传入图的大小n和正向表list,buf

//返回到各点最短距离min[]和路径pre[],pre[i]记录s到i路径上i的父结点,pre[s]=-1

//可更改路权类型,但必须非负!

#define MAXN 200
#define inf1000000000
typedef int elem_t;
struct edge_t{
         int to;
         elem_t len;
};
 
#define _cp(a,b)((a).d<(b).d)
structheap_t{elem_t d;int v;};
struct heap{
         heap_t h[MAXN*MAXN];
         int n,p,c;
         void init(){n=0;}
         void ins(heap_t e){
                   for(p=++n;p>1&&_cp(e,h[p>>1]);h[p]=h[p>>1],p>>=1);
                   h[p]=e;
         }
         int del(heap_t& e){
                   if (!n) return 0;
                   for(e=h[p=1],c=2;c<n&&_cp(h[c+=(c<n-1&&_cp(h[c+1],h[c]))],h[n]);h[p]=h[c],p=c,c<<=1);
                   h[p]=h[n--];return 1;
         }
};
 
void dijkstra(intn,int* list,edge_t* buf,int s,elem_t* min,int* pre){
         heap h;heap_t e;
         int v[MAXN],i,t,f;
         for (i=0;i<n;i++)
                   min[i]=inf,v[i]=0,pre[i]=-1;
         h.init();min[e.v=s]=e.d=0,h.ins(e);
         while (h.del(e))
                   if (!v[e.v])
                            for(v[f=e.v]=1,t=list[f];t<list[f+1];t++)
                                     if(!v[buf[t].to]&&min[f]+buf[t].len<min[buf[t].to])
                                               pre[buf[t].to]=f,min[e.v=buf[t].to]=e.d=min[f]+buf[t].len,h.ins(e);
}


6.最短路径(单源dijkstra+mapped_heap邻接表形式)

//单源最短路径,dijkstra算法+映射二分堆,邻接表形式,复杂度O(mlogn)

//求出源s到所有点的最短路经,传入图的大小n和邻接表list

//返回到各点最短距离min[]和路径pre[],pre[i]记录s到i路径上i的父结点,pre[s]=-1

//可更改路权类型,但必须非负!

#define MAXN 200
#define inf1000000000
typedef int elem_t;
struct edge_t{
         int from,to;
         elem_t len;
         edge_t* next;
};
 
#define _cp(a,b)((a)<(b))
struct heap{
         elem_t h[MAXN+1];
         int ind[MAXN+1],map[MAXN+1],n,p,c;
         void init(){n=0;}
         void ins(int i,elem_t e){
                   for(p=++n;p>1&&_cp(e,h[p>>1]);h[map[ind[p]=ind[p>>1]]=p]=h[p>>1],p>>=1);
                   h[map[ind[p]=i]=p]=e;
         }
         int del(int i,elem_t& e){
                   i=map[i];if (i<1||i>n)return 0;
                   for(e=h[p=i];p>1;h[map[ind[p]=ind[p>>1]]=p]=h[p>>1],p>>=1);
                   for(c=2;c<n&&_cp(h[c+=(c<n-1&&_cp(h[c+1],h[c]))],h[n]);h[map[ind[p]=ind[c]]=p]=h[c],p=c,c<<=1);
                   h[map[ind[p]=ind[n]]=p]=h[n];n--;return1;
         }
         int delmin(int& i,elem_t& e){
                   if (n<1) return0;i=ind[1];
                   for(e=h[p=1],c=2;c<n&&_cp(h[c+=(c<n-1&&_cp(h[c+1],h[c]))],h[n]);h[map[ind[p]=ind[c]]=p]=h[c],p=c,c<<=1);
                   h[map[ind[p]=ind[n]]=p]=h[n];n--;return1;
         }
};
 
void dijkstra(intn,edge_t* list[],int s,elem_t* min,int* pre){
         heap h;
         edge_t* t;elem_t e;
         int v[MAXN],i;
         for (h.init(),i=0;i<n;i++)
                   min[i]=((i==s)?0:inf),v[i]=0,pre[i]=-1,h.ins(i,min[i]);
         while (h.delmin(i,e))
                   for (v[i]=1,t=list[i];t;t=t->next)
                            if(!v[t->to]&&min[i]+t->len<min[t->to])
                                     pre[t->to]=i,h.del(t->to,e),min[t->to]=e=min[i]+t->len,h.ins(t->to,e);
}


7.最短路径(单源dijkstra+mapped_heap正向表形式)

//单源最短路径,dijkstra算法+映射二分堆,正向表形式,复杂度O(mlogn)

//求出源s到所有点的最短路经,传入图的大小n和正向表list,buf

//返回到各点最短距离min[]和路径pre[],pre[i]记录s到i路径上i的父结点,pre[s]=-1

//可更改路权类型,但必须非负!

#define MAXN 200
#define inf1000000000
typedef int elem_t;
struct edge_t{
         int to;
         elem_t len;
};
 
#define _cp(a,b)((a)<(b))
struct heap{
         elem_t h[MAXN+1];
         int ind[MAXN+1],map[MAXN+1],n,p,c;
         void init(){n=0;}
         void ins(int i,elem_t e){
                   for(p=++n;p>1&&_cp(e,h[p>>1]);h[map[ind[p]=ind[p>>1]]=p]=h[p>>1],p>>=1);
                   h[map[ind[p]=i]=p]=e;
         }
         int del(int i,elem_t& e){
                   i=map[i];if (i<1||i>n)return 0;
                   for(e=h[p=i];p>1;h[map[ind[p]=ind[p>>1]]=p]=h[p>>1],p>>=1);
                   for(c=2;c<n&&_cp(h[c+=(c<n-1&&_cp(h[c+1],h[c]))],h[n]);h[map[ind[p]=ind[c]]=p]=h[c],p=c,c<<=1);
                   h[map[ind[p]=ind[n]]=p]=h[n];n--;return1;
         }
         int delmin(int& i,elem_t& e){
                   if (n<1) return0;i=ind[1];
                   for (e=h[p=1],c=2;c<n&&_cp(h[c+=(c<n-1&&_cp(h[c+1],h[c]))],h[n]);h[map[ind[p]=ind[c]]=p]=h[c],p=c,c<<=1);
                   h[map[ind[p]=ind[n]]=p]=h[n];n--;return1;
         }
};
 
void dijkstra(intn,int* list,edge_t* buf,int s,elem_t* min,int* pre){
         heap h;elem_t e;
         int v[MAXN],i,t;
         for (h.init(),i=0;i<n;i++)
                   min[i]=((i==s)?0:inf),v[i]=0,pre[i]=-1,h.ins(i,min[i]);
         while (h.delmin(i,e))
                   for(v[i]=1,t=list[i];t<list[i+1];t++)
                            if(!v[buf[t].to]&&min[i]+buf[t].len<min[buf[t].to])
                                     pre[buf[t].to]=i,h.del(buf[t].to,e),min[buf[t].to]=e=min[i]+buf[t].len,h.ins(buf[t].to,e);
}


8.最短路径(单源dijkstra邻接阵形式)

//单源最短路径,dijkstra算法,邻接阵形式,复杂度O(n^2)

//求出源s到所有点的最短路经,传入图的顶点数n,(有向)邻接矩阵mat

//返回到各点最短距离min[]和路径pre[],pre[i]记录s到i路径上i的父结点,pre[s]=-1

//可更改路权类型,但必须非负!

#define MAXN 200
#define inf1000000000
typedef int elem_t;
 
void dijkstra(intn,elem_t mat[][MAXN],int s,elem_t* min,int* pre){
         int v[MAXN],i,j,k;
         for (i=0;i<n;i++)
                   min[i]=inf,v[i]=0,pre[i]=-1;
         for (min[s]=0,j=0;j<n;j++){
                   for (k=-1,i=0;i<n;i++)
                            if(!v[i]&&(k==-1||min[i]<min[k]))
                                     k=i;
                   for (v[k]=1,i=0;i<n;i++)
                            if(!v[i]&&min[k]+mat[k][i]<min[i])
                                     min[i]=min[k]+mat[pre[i]=k][i];
         }
}


9.最短路径(多源floyd_warshall邻接阵形式)

//多源最短路径,floyd_warshall算法,复杂度O(n^3)

//求出所有点对之间的最短路经,传入图的大小和邻接阵

//返回各点间最短距离min[]和路径pre[],pre[i][j]记录i到j最短路径上j的父结点

//可更改路权类型,路权必须非负!

#define MAXN 200
#define inf1000000000
typedef int elem_t;
 
void floyd_warshall(int n,elem_t mat[][MAXN],elem_t min[][MAXN],int pre[][MAXN]){
         int i,j,k;
         for (i=0;i<n;i++)
                   for (j=0;j<n;j++)
                            min[i][j]=mat[i][j],pre[i][j]=(i==j)?-1:i;
         for (k=0;k<n;k++)
                   for (i=0;i<n;i++)
                            for (j=0;j<n;j++)
                                     if(min[i][k]+min[k][j]<min[i][j])
                                               min[i][j]=min[i][k]+min[k][j],pre[i][j]=pre[k][j];
}


  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值