求单源最短路到其余各点,然后返回源点的总最短路长,以构造邻接表的方法不同分为两种解法。


POJ1511(ZOJ2008)-Invitation Cards

  改变构造邻接表的方法后,分为两种解法

解法一:

 //POJ1511-ZOJ2008
//Time:7766Ms Memory:99112K
//求从1处到各点后再返回1处的最短总路长
//需要构造邻接表和逆邻接表
//构造方法1:vector构造邻接表
//SPFA+邻接表
#include<iostream>
#include<cstring>
#include<cstdio>
#include<vector>
#include<queue>
using namespace std; #define MAX 1000005
#define INF 0x3f3f3f3f struct Edge {
int u, w;
Edge(int uu, int ww) :u(uu), w(ww) {}
}; vector<Edge> e1[MAX], e2[MAX]; //邻接表-逆邻接表 int n, m;
int d[MAX];
long long sum;
bool v[MAX]; void SPFA(int x, vector<Edge> e[MAX])
{
memset(d, INF, sizeof(d));
memset(v, false, sizeof(v));
queue<int> q;
q.push(x);
d[x] = ;
while (!q.empty()){
int cur = q.front();
q.pop();
v[cur] = false;
for (int i = ; i < e[cur].size(); i++)
{
int u = e[cur][i].u;
if (d[u] > d[cur] + e[cur][i].w)
{
d[u] = d[cur] + e[cur][i].w;
if (!v[u]) { q.push(u); v[u] = true; }
}
}
}
for (int i = ; i <= n; i++)
sum += d[i];
} int main()
{
int T;
scanf("%d", &T);
while (T--) {
sum = ;
memset(e1, , sizeof(e1));
memset(e2, , sizeof(e2));
scanf("%d%d", &n, &m);
while (m--) {
int u, v, w;
scanf("%d%d%d", &u, &v, &w);
e1[u].push_back(Edge(v, w)); //正向
e2[v].push_back(Edge(u, w)); //逆向
} SPFA(, e1);
SPFA(, e2);
printf("%lld\n", sum);
} return ;
}

解法二:

 //POJ1511-ZOJ2008
//Time:2000Ms Memory:36424K
//求从1处到各点后再返回1处的最短总路长
//需要构造邻接表和逆邻接表
//构造方法2:偏序关系构造邻接表
//SPFA+邻接表
#include<iostream>
#include<cstring>
#include<cstdio>
#include<vector>
#include<queue>
using namespace std; #define MAX 1000005
#define INF 0x3f3f3f3f struct Edge {
int u, w, next;
Edge() {}
Edge(int uu, int ww, int nn) :u(uu), w(ww), next(nn) {}
}e1[MAX], e2[MAX]; //邻接表-逆邻接表 int h1[MAX], h2[MAX]; //正表表头-逆表表头
int n, m;
int d[MAX];
long long sum;
bool v[MAX]; void SPFA(int x, Edge e[MAX], int h[MAX])
{
memset(d, INF, sizeof(d));
memset(v, false, sizeof(v));
queue<int> q;
q.push(x);
d[x] = ;
while (!q.empty()){
int cur = q.front();
q.pop();
v[cur] = false;
for (int i = h[cur]; i != -; i = e[i].next)
{
int u = e[i].u;
int w = e[i].w;
if (d[u] > d[cur] + w)
{
d[u] = d[cur] + w;
if (!v[u]) { q.push(u); v[u] = true; }
}
}
}
for (int i = ; i <= n; i++)
sum += d[i];
} int main()
{
int T;
scanf("%d", &T);
while (T--) {
sum = ;
memset(e1, , sizeof(e1));
memset(e2, , sizeof(e2));
memset(h1, -, sizeof(h1));
memset(h2, -, sizeof(h2));
scanf("%d%d", &n, &m);
for (int i = ; i < m; i++) {
int u, v, w;
scanf("%d%d%d", &u, &v, &w);
e1[i] = Edge(v, w, h1[u]);
e2[i] = Edge(u, w, h2[v]);
h1[u] = h2[v] = i;
} SPFA(, e1, h1);
SPFA(, e2, h2);
printf("%lld\n", sum);
} return ;
}

ACM/ICPC 之 最短路-SPFA+正逆邻接表(POJ1511(ZOJ2008))的更多相关文章

  1. SPFA中 正逆邻接表的建立

    正邻接表的建立: 先定义一个结构体: struct node { int r,v,w,next; }Map[]; 每输入一组数据 就通过Add函数加入到邻接表中,上图已经说得很明白了,结合着下面的代码 ...

  2. HDU 1874 畅通工程续(最短路/spfa Dijkstra 邻接矩阵+邻接表)

    题目链接: 传送门 畅通工程续 Time Limit: 1000MS     Memory Limit: 65536K Description 某省自从实行了很多年的畅通工程计划后,终于修建了很多路. ...

  3. ACM/ICPC 之 最短路-Floyd+SPFA(BFS)+DP(ZOJ1232)

    这是一道非常好的题目,融合了很多知识点. ZOJ1232-Adventrue of Super Mario 这一题折磨我挺长时间的,不过最后做出来非常开心啊,哇咔咔咔 题意就不累述了,注释有写,难点在 ...

  4. 2016 ACM/ICPC Asia Regional Qingdao Online 1001/HDU5878 打表二分

    I Count Two Three Time Limit: 3000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others ...

  5. 最短路径SPFA算法(邻接表存法)

    queue <int> Q; void SPFA (int s) { int i, v; for(int i=0; i<=n; i++) dist[i]=INF; //初始化每点i到 ...

  6. POJ 3013 SPFA算法,邻接表的使用

    Big Christmas Tree Time Limit: 3000MS   Memory Limit: 131072K Total Submissions: 19029   Accepted: 4 ...

  7. POJ 3259 Wormholes(最短路&spfa正权回路)题解

    题意:给你m条路花费时间(双向正权路径),w个虫洞返回时间(单向负权路径),问你他能不能走一圈回到原点之后,时间倒流. 思路:题意有点难看懂,我们建完边之后找一下是否存在负权回路,存在则能,反之不能. ...

  8. POJ 1860 Currency Exchange(最短路&spfa正权回路)题解

    题意:n种钱,m种汇率转换,若ab汇率p,手续费q,则b=(a-q)*p,你有第s种钱v数量,问你能不能通过转化让你的s种钱变多? 思路:因为过程中可能有负权值,用spfa.求是否有正权回路,dis[ ...

  9. ACM/ICPC 之 数论-素数筛选法 与 "打表"思路(POJ 1595)

    何为"打表"呢,说得简单点就是: 有时候与其重复运行同样的算法得出答案,还不如直接用算法把这组数据所有可能的答案都枚举出来存到一个足够大的容器中去-例如数组(打表),然后再输入数据 ...

随机推荐

  1. 从HTML原型到jsp页面完美转型攻略(教你即使不会写代码也能弄出漂亮的网页)

    大家都知道软件项目(web)开发之前都要先做原型设计,而我们使用的比较多的一款原型设计软件就是Axure rp了.在Axure rp上画原型不需要任何编码能力,而且生成的原型可以在浏览器上运行.除了没 ...

  2. Java通过JNI调用C

    Java调用C有多种方式,本文介绍笔者最近的学习过程,避免今后再犯类似的错误. 首先,Java肯定是调用C的动态链接库,即通过编译器编译后的dll/so文件. 下面介绍gcc编译dll的方法. 一般情 ...

  3. [转载]解析用户生命周期价值:LTV

    http://www.sykong.com/2014/07/23144 http://youxiputao.com/articles/1288 http://www.woshipm.com/opera ...

  4. [译]Exploring Angular 1.3: Binding to Directive Controllers

    原文: http://blog.thoughtram.io/angularjs/2015/01/02/exploring-angular-1.3-bindToController.html Angul ...

  5. 从一个弱引用导致的奔溃 谈 weak assign strong的应用场景【iOS开发教程】

    从一个弱引用导致的奔溃 谈 weak assign strong的应用场景 .h中的定义方法一: @property (nonatomic, assign) NSArray *dataSource; ...

  6. jquery隐藏按钮

    $(function () { jhbs = getQueryStringByName('jhbs'); shhbs = getQueryStringByName('shhbs'); if (shhb ...

  7. ML_R Kmeans

    Kmeans作为机器学习中入门级算法,涉及到计算距离算法的选择,聚类中心个数的选择.下面就简单介绍一下在R语言中是怎么解决这两个问题的. 参考Unsupervised Learning with R ...

  8. XDU 1160 - 科协的数字游戏I

    Problem 1160 - 科协的数字游戏I Time Limit: 1000MS   Memory Limit: 65536KB   Difficulty: Total Submit: 184   ...

  9. Redis命令

    redis的常用命令主要分为两个方面.一个是键值相关命令.一个是服务器相关命令(redis-cli进入终端) 1.键值相关命令 keys * 取出当前所有的key exists name 查看n是否有 ...

  10. 【C语言入门教程】2.2 常量 与 变量

    2.2 常量 与 变量 顾名思义,常量是运算中不能改变数值的数据类型,变量是可改变数值的数据类型.根据需要,可将一些在程序中不必改变数值的类型定义为常量,这样也可避免因修改数值造成程序错误.任何改变常 ...