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


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. linux下vim更改注释颜色

    我linux下默认底色是黑色,字体是绿色,但是注释默认是蓝色,,很不爽...所以要更改注释颜色......以下文章来自转载. from: http://blog.csdn.net/gz109/arti ...

  2. webapp中的meta

    <!--开发后删除--> <meta http-equiv="Pragma" name="no-store" /><!--必须联网 ...

  3. [转]C++模板学习

    1. 模板的概念. 我们已经学过重载(Overloading),对重载函数而言,C++的检查机制能通过函数参数的不同及所属类的不同.正确的调用重载函数.例如,为求两个数的最大值,我们定义MAX()函数 ...

  4. HDOJ 1576 A/B

    数论水题... A/B Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total ...

  5. 如何用SQL命令行工具删除dedecms指定id文章

    用dedecms采集时标题字段设置错了,出现了注释符号<!---->,导致后台的文章列表出现错误,也无法直接从列表中删除,可以远程登录数据库去操作,这个相对比较麻烦,想着直接从后台的SQL ...

  6. dede自定义表单增加添加时间怎么弄

    我们在用dedecms添加自定义表单时有时想要设置一个用户提交的时间,方便查询,比如我们的客服人员查询昨天晚上下班后有哪些订单是刚生成的,比较好查看,如下图所示.那么,dedecms自定义表单增加添加 ...

  7. Swift2.1 语法指南——错误处理

    原档:https://developer.apple.com/library/prerelease/ios/documentation/Swift/Conceptual/Swift_Programmi ...

  8. struts2-(2)HelloWorld

    1.环境配置 1).进入http://struts.apache.org/download.cgi#struts23241 下载 struts官方源码 2).解压,进入apps/struts2-bla ...

  9. 关于CSS中对IE条件注释的问题

    一.通用区分方式:IE6.IE7能识别*,标准浏览器(如FF)不能识别*:IE6能识别*,但不能识别 !important:IE7能识别*,也能识别 !important:IE8能识别\0,不能识别* ...

  10. Camel——涨知识了,骆驼命名法

    骆驼式命名法(Camel-Case)又称驼峰命名法,是电脑程式编写时的一套命名规则(惯例).正如它的名称CamelCase所表示的那样,是指混合使用大小写字母来构成变量和函数的名字.程序员们为了自己的 ...