最短路模板(Dijkstra & Dijkstra算法+堆优化 & bellman_ford & 单源最短路SPFA)
关于几个的区别和联系:http://www.cnblogs.com/zswbky/p/5432353.html
d.每组的第一行是三个整数T,S和D,表示有T条路,和草儿家相邻的城市的有S个(草儿家到这个城市的距离设为0),草儿想去的地方有D个;
求D个城市中距离草儿家最近的距离。
s.进行1次单源最短路,找出距离最小的即可。
c.Dijkstra单源最短路
/*
Dijkstra单源最短路
权值必须是非负
单源最短路径,Dijkstra算法,邻接矩阵形式,复杂度为O(n^2)
求出源beg到所有点的最短路径,传入图的顶点数,和邻接矩阵cost[][]
返回各点的最短路径lowcost[],路径pre[].pre[i]记录beg到i路径上的父结点,pre[beg]=-1
可更改路径权类型,但是权值必须为非负
*/
#include<iostream>
#include<stdio.h>
#include<string.h>
using namespace std; const int MAXN=;
#define typec int
const typec INF=0x3f3f3f3f;//防止后面溢出,这个不能太大
bool vis[MAXN];
int pre[MAXN];
void Dijkstra(typec cost[][MAXN],typec lowcost[],int n,int beg){
for(int i=;i<n;i++){
lowcost[i]=INF;vis[i]=false;pre[i]=-;
}
lowcost[beg]=;
for(int j=;j<n;j++){
int k=-;
int Min=INF;
for(int i=;i<n;i++)
if(!vis[i]&&lowcost[i]<Min){
Min=lowcost[i];
k=i;
}
if(k==-)break;
vis[k]=true;
for(int i=;i<n;i++)
if(!vis[i]&&lowcost[k]+cost[k][i]<lowcost[i]){
lowcost[i]=lowcost[k]+cost[k][i];
pre[i]=k;
}
}
} int cost[MAXN][MAXN];
int lowcost[MAXN]; int main(){ int T,S,D;
int a,b,time;
int city1[MAXN];
int city2[MAXN]; while(~scanf("%d%d%d",&T,&S,&D)){
for(int i=;i<MAXN;++i){
for(int j=;j<MAXN;++j){
cost[i][j]=INF;
}
}
memset(vis,false,sizeof(vis)); for(int i=;i<T;++i){
scanf("%d%d%d",&a,&b,&time);
if(time<cost[a][b]){
cost[a][b]=time;
cost[b][a]=time;
}
}
//0作为草儿家
for(int i=;i<S;++i){
scanf("%d",&city1[i]);
cost[][city1[i]]=;
cost[city1[i]][]=;
}
for(int i=;i<D;++i){
scanf("%d",&city2[i]);
} Dijkstra(cost,lowcost,MAXN,);
int minTime=lowcost[city2[]];
for(int i=;i<D;++i){
if(lowcost[city2[i]]<minTime)
minTime=lowcost[city2[i]];
} printf("%d\n",minTime);
}
return ;
}
c2.Dijkstra算法+堆优化
/*
Dijkstra算法+堆优化
使用优先队列优化,复杂度O(E log E)
使用优先队列优化Dijkstra算法
复杂度O(E log E)
注意对vector<Edge>E[MAXN]进行初始化后加边
*/
#include<iostream>
#include<stdio.h>
#include<vector>
#include<string.h>
#include<queue>
using namespace std; const int INF=0x3f3f3f3f;
const int MAXN=;
struct qnode{
int v;
int c;
qnode(int _v=,int _c=):v(_v),c(_c){}
bool operator <(const qnode &r)const{
return c>r.c;
}
};
struct Edge{
int v,cost;
Edge(int _v=,int _cost=):v(_v),cost(_cost){}
};
vector<Edge>E[MAXN];
bool vis[MAXN];
int dist[MAXN];
//点的编号从1开始
void Dijkstra(int n,int start){
memset(vis,false,sizeof(vis));
for(int i=;i<=n;i++)dist[i]=INF;
priority_queue<qnode>que;
while(!que.empty())que.pop();
dist[start]=;
que.push(qnode(start,));
qnode tmp;
while(!que.empty()){
tmp=que.top();
que.pop();
int u=tmp.v;
if(vis[u])continue;
vis[u]=true;
for(int i=;i<E[u].size();i++){
int v=E[tmp.v][i].v;
int cost=E[u][i].cost;
if(!vis[v]&&dist[v]>dist[u]+cost){
dist[v]=dist[u]+cost;
que.push(qnode(v,dist[v]));
}
}
}
}
void addedge(int u,int v,int w){
E[u].push_back(Edge(v,w));
} int main(){
int T,S,D;
int a,b,time;
int city1[MAXN];
int city2[MAXN]; while(~scanf("%d%d%d",&T,&S,&D)){
for(int i=;i<MAXN;++i){
E[i].clear();
} for(int i=;i<T;++i){
scanf("%d%d%d",&a,&b,&time);
addedge(a,b,time);//这里有重边了。。没办法,
addedge(b,a,time);
}
//0作为草儿家
for(int i=;i<S;++i){
scanf("%d",&city1[i]);
addedge(,city1[i],);
addedge(city1[i],,);
}
for(int i=;i<D;++i){
scanf("%d",&city2[i]);
} Dijkstra(MAXN-,);
int minTime=dist[city2[]];
for(int i=;i<D;++i){
if(dist[city2[i]]<minTime)
minTime=dist[city2[i]];
} printf("%d\n",minTime);
}
return ;
}
c3.单源最短路bellman_ford算法
/*
单源最短路bellman_ford算法
单源最短khtkbellman_ford算法,复杂度O(VE)
可以处理负边权图。
可以判断是否存在负环回路。返回true,当且仅当图中不包含从源点可达的负权回路
vector<Edge>E;先E.clear()初始化,然后加入所有边
点的编号从1开始(从0开始简单修改就可以了)
*/
#include<iostream>
#include<stdio.h>
#include<vector>
using namespace std; const int INF=0x3f3f3f3f;
const int MAXN=;
int dist[MAXN];
struct Edge{
int u,v;
int cost;
Edge(int _u=,int _v=,int _cost=):u(_u),v(_v),cost(_cost){}
};
vector<Edge>E;
//点的编号从1开始
bool bellman_ford(int start,int n){
for(int i=;i<=n;i++)dist[i]=INF;
dist[start]=;
//最多做n-1次
for(int i=;i<n;i++){
bool flag=false;
for(int j=;j<E.size();j++){
int u=E[j].u;
int v=E[j].v;
int cost=E[j].cost;
if(dist[v]>dist[u]+cost){
dist[v]=dist[u]+cost;
flag=true;
}
}
if(!flag)return true;//没有负环回路
}
for(int j=;j<E.size();j++)
if(dist[E[j].v]>dist[E[j].u]+E[j].cost)
return false;//有负环回路
return true;//没有负环回路
}
void addedge(int u,int v,int cost){
E.push_back(Edge(u,v,cost));
} int main(){
int T,S,D;
int a,b,time;
int city1[MAXN];
int city2[MAXN]; while(~scanf("%d%d%d",&T,&S,&D)){
E.clear(); for(int i=;i<T;++i){
scanf("%d%d%d",&a,&b,&time);
addedge(a,b,time);//有重边了。
addedge(b,a,time);
}
//0作为草儿家
for(int i=;i<S;++i){
scanf("%d",&city1[i]);
addedge(,city1[i],);
addedge(city1[i],,);
}
for(int i=;i<D;++i){
scanf("%d",&city2[i]);
} bellman_ford(,MAXN-);//MAXN-1
int minTime=dist[city2[]];
for(int i=;i<D;++i){
if(dist[city2[i]]<minTime)
minTime=dist[city2[i]];
} printf("%d\n",minTime);
}
return ;
}
c4.单源最短路SPFA
/*
单源最短路SPFA
时间复杂度O(kE)
这个是队列实现,有时候改成栈实现会更加快,很容易修改
这个复杂度是不定的
*/
#include<iostream>
#include<stdio.h>
#include<vector>
#include<string.h>
#include<queue>
using namespace std; const int MAXN=;
const int INF=0x3f3f3f3f;
struct Edge{
int v;
int cost;
Edge(int _v=,int _cost=):v(_v),cost(_cost){}
};
vector<Edge>E[MAXN];
void addedge(int u,int v,int w){
E[u].push_back(Edge(v,w));
}
bool vis[MAXN];//在队列标志
int cnt[MAXN];//每个点的入队列次数
int dist[MAXN];
bool SPFA(int start,int n){
memset(vis,false,sizeof(vis));
for(int i=;i<=n;i++)dist[i]=INF;
vis[start]=true;
dist[start]=;
queue<int>que;
while(!que.empty())que.pop();
que.push(start);
memset(cnt,,sizeof(cnt));
cnt[start]=;
while(!que.empty()){
int u=que.front();
que.pop();
vis[u]=false;
for(int i=;i<E[u].size();i++){
int v=E[u][i].v;
if(dist[v]>dist[u]+E[u][i].cost){
dist[v]=dist[u]+E[u][i].cost;
if(!vis[v]){
vis[v]=true;
que.push(v);
if(++cnt[v]>n)return false;
//cnt[i] 为入队列次数,用来判定是否存在负环回路
}
}
}
}
return true;
} int main(){
int T,S,D;
int a,b,time;
int city1[MAXN];
int city2[MAXN]; while(~scanf("%d%d%d",&T,&S,&D)){
for(int i=;i<MAXN;++i){
E[i].clear();
} for(int i=;i<T;++i){
scanf("%d%d%d",&a,&b,&time);
addedge(a,b,time);//有重边了。
addedge(b,a,time);
}
//0作为草儿家
for(int i=;i<S;++i){
scanf("%d",&city1[i]);
addedge(,city1[i],);
addedge(city1[i],,);
}
for(int i=;i<D;++i){
scanf("%d",&city2[i]);
} SPFA(,MAXN-);//MAXN-1
int minTime=dist[city2[]];
for(int i=;i<D;++i){
if(dist[city2[i]]<minTime)
minTime=dist[city2[i]];
} printf("%d\n",minTime);
}
return ;
}
增加一个Floyd的邻接表(这个题不错,有时间看看)
http://blog.163.com/zjut_nizhenyang/blog/static/169570029201111841938607/
Floyd
http://www.cnblogs.com/zswbky/p/5432387.html
最短路模板(Dijkstra & Dijkstra算法+堆优化 & bellman_ford & 单源最短路SPFA)的更多相关文章
- 【算法】单源最短路——Dijkstra
对于固定起点的最短路算法,我们称之为单源最短路算法.单源最短路算法很多,最常见的就是dijkstra算法. dijkstra主要用的是一种贪心的思想,就是说如果i...s...t...j是最短路,那么 ...
- [ACM_图论] Domino Effect (POJ1135 Dijkstra算法 SSSP 单源最短路算法 中等 模板)
Description Did you know that you can use domino bones for other things besides playing Dominoes? Ta ...
- 单源最短路模板(dijkstra)
单源最短路(dijkstra算法及堆优化) 弱化版题目链接 n^2 dijkstra模板 #include<iostream> #include<cstdio> #includ ...
- 单源最短路——dijkstra算法
Dijkstra算法 1.定义概览 Dijkstra(迪杰斯特拉)算法是典型的单源最短路径算法,用于计算一个节点到其他所有节点的最短路径.主要特点是以起始点为中心向外层层扩展,直到扩展到终点为止. 问 ...
- 【算法系列学习】Dijkstra单源最短路 [kuangbin带你飞]专题四 最短路练习 A - Til the Cows Come Home
https://vjudge.net/contest/66569#problem/A http://blog.csdn.net/wangjian8006/article/details/7871889 ...
- 利用分支限界法求解单源最短路(Dijkstra)问题
分支限界法定义:采用Best fist search算法,并使用剪枝函数的算法称为分支界限法. 分支限界法解释:按Best first的原则,有选择的在其child中进行扩展,从而舍弃不含有最优解的分 ...
- 牛客编程巅峰赛S1第6场 - 黄金&钻石&王者 C.星球游戏 (单源最短路,Dijkstra)
题意:有\(n\)个点,\(m\)条双向边,两个方向的权值都是相等的,可以从\(A\)中的某个点出发走到\(B\)中的某个点,求所有路径中的最短距离,如果A和B中没有点联通,则输出\(-1\). 题解 ...
- 洛谷 P5837 [USACO19DEC]Milk Pumping G (单源最短路,dijkstra)
题意:有一\(n\)个点,\(m\)条边的双向图,每条边都有花费和流量,求从\(1\)~\(n\)的路径中,求\(max\frac{min(f)}{\sum c}\). 题解:对于c,一定是单源最短路 ...
- 模板C++ 03图论算法 1最短路之单源最短路(SPFA)
3.1最短路之单源最短路(SPFA) 松弛:常听人说松弛,一直不懂,后来明白其实就是更新某点到源点最短距离. 邻接表:表示与一个点联通的所有路. 如果从一个点沿着某条路径出发,又回到了自己,而且所经过 ...
随机推荐
- 关于ellipsis多行换行的方案
WebKit浏览器或移动端的页面在WebKit浏览器或移动端(绝大部分是WebKit内核的浏览器)的页面实现比较简单,可以直接使用WebKit的CSS扩展属性(WebKit是私有属性)-webkit- ...
- ubuntu14.04 下emacs 24 配置
目的: 配置emacs 24 适合编程开发 主要参考JerryZhang的配置(Emacs 简易教程) http://www.perfect-is-shit.com/emacs-simple-tuto ...
- 根据内存布局定位的一个fastdfs坑
在使用fastdfs时,编写数据上传代码时,遇到一个坑.最终根据指针对应的内存布局定位到一个其client API的一个坑,值得记录一下.具体是在 tracker_connect_server() 这 ...
- css的初步了解
学习了很多知识在这里,今天三月二十一日,老师讲了css的基础 对css有了初步的了解. 主要学习了以下几点: 一.css的选择器 1.派生选择器 2.类选择器 3.id选择器 4.属性选择器 二.cs ...
- springMvc3.0.5搭建全程 (转)
用了大半年的Spring MVC3.0,用着感觉不错.简单写一个搭建Spring MVC3.0的流程(以Spring3.0.5为列),数据库交互使用spring JDBC Template,附件有项目 ...
- OkHttp和Volley对比
OkHttp 物理质量 使用OkHttp需要 okio.jar (80k), okhttp.jar(330k)这2个jar包,总大小差不多400k,加上自己的封装,差不多得410k. 功能介绍 Squ ...
- 瞎BB
今天家里停电了,什么都没干,又开始胡思乱想了.或许有点时候真的应该沉迷一些东西. 小时候其实挺喜欢数学的,考试都是90分,100分,我喜欢思考钻研不懂的题目,花很多时间,所以有的时候会跳过课堂的东西, ...
- LeetCode----Linked List
Swap Nodes in Pairs 思路:需要构造一个头指针,指向链表.一次比较两个指针,将其翻转,临界条件是pre != null(链表长度为偶数) && pre.next != ...
- python中import和from...import区别
在python用import或者from...import来导入相应的模块.模块其实就是一些函数和类的集合文件,它能实现一些相应的功能,当我们需要使用这些功能的时候,直接把相应的模块导入到我们的程序中 ...
- out.print()和response.getWriter().write()区别
1.print()和write()区别: write():表示的是仅支持输入字符类型数据,字符,字符数组和字符串等, print():表示的是将各种数据类型(包括object)的数据通过默认编码换成b ...