<题目链接>

题目大意:

有向图,求从起点1到每个点的最短路然后再回到起点1的最短路之和。

解题分析:

在求每个点到1点的最短路径时,如果仅仅只是遍历每个点,对它们每一个都进行一次最短路算法,那么即使是用了堆优化的dijkstra,时间复杂度也高达$O(n^2log(n))$,而本题有1000000个点,毫无疑问,这种想法必然是不可行的,所以我们可以采用逆向思维,将图中的每一条有向边全部反向,然后以1为起点,仅做一次dijkstra,就能得到1到所有点的最短距离,即反向前的,所有点到1点的最短距离。所以,本题的正解应为:先以1为起点,做一次dijkstra,算出,1到所有点的最短距离,然后将边反向,再以1为起点,做一次dijkstra,此时就能得到,其他所有点到1的最短距离,将所有的最短距离相加,即为答案。时间复杂度为$O(nlogn)$。

 #include <cstdio>
#include <cstring>
#include <queue>
#include <algorithm>
#include <iostream>
using namespace std; #define INF 0x3f3f3f3f
const int maxn =+; int n,m;
struct Edge{
int to;
int next;
int w;
}; Edge edge[maxn],redge[maxn]; struct NODE{
int index;
int dis;
bool operator < (NODE const &tmp)const{
return dis>tmp.dis;
}
}d[maxn]; int dist[maxn];
int cnt,rcnt,head1[maxn],head2[maxn],vis[maxn]; void init(){
memset(head1,-,sizeof(head1));
memset(head2,-,sizeof(head2));
cnt=,rcnt=;
} void add1(int u,int v,int w){
edge[cnt].to=v;edge[cnt].w=w;
edge[cnt].next=head1[u];
head1[u]=cnt++;
} void add2(int u,int v,int w){
redge[rcnt].to=v;redge[rcnt].w=w;
redge[rcnt].next=head2[u];
head2[u]=rcnt++;
} void dijkstra1(int st){
for(int i=;i<=n;i++){
vis[i]=;d[i].dis=INF,d[i].index=i;
} priority_queue<NODE>q;
d[st].dis=;q.push(d[st]);
while(!q.empty()){
int u=q.top().index;
q.pop();
if(vis[u])continue;
vis[u]=;
for(int i=head1[u];i!=-;i=edge[i].next){
int v=edge[i].to;
if(d[v].dis>d[u].dis+edge[i].w){
d[v].dis=d[u].dis+edge[i].w;
q.push(d[v]);
}
}
}
} void dijkstra2(int st){ //因为正、反向边的edge[],和head[]散组不同,所以要将另外再写一个dijkstra函数
for(int i=;i<=n;i++){
vis[i]=;d[i].dis=INF,d[i].index=i;
} priority_queue<NODE>q;
d[st].dis=;q.push(d[st]);
while(!q.empty()){
int u=q.top().index;
q.pop();
if(vis[u])continue;
vis[u]=;
for(int i=head2[u];i!=-;i=redge[i].next){
int v=redge[i].to;
if(d[v].dis>d[u].dis+redge[i].w){
d[v].dis=d[u].dis+redge[i].w;
q.push(d[v]);
}
}
}
} int main(){
int t;scanf("%d",&t);
while(t--){
scanf("%d %d",&n,&m);
init();
for(int i=;i<=m;i++){
int a,b,c;
scanf("%d %d %d",&a,&b,&c);
add1(a,b,c); //存储该有向图正确的边
add2(b,a,c); //将该有向图的所有边反向存储
} long long sum=; dijkstra1(); //边未反向之前,求出1到所有点的最短路
for(int i=;i<=n;i++){
sum+=d[i].dis;
} dijkstra2(); //将边反向后,求出所有点到1点的最短路
for(int i=;i<=n;i++){
sum+=d[i].dis;
}
printf("%lld\n",sum);
}
return ;
}

2018-08-27

POJ-1511 Invitation Cards (单源最短路+逆向)的更多相关文章

  1. Invitation Cards POJ - 1511 (双向单源最短路)

    In the age of television, not many people attend theater performances. Antique Comedians of Malidine ...

  2. POJ 1511 Invitation Cards / UVA 721 Invitation Cards / SPOJ Invitation / UVAlive Invitation Cards / SCU 1132 Invitation Cards / ZOJ 2008 Invitation Cards / HDU 1535 (图论,最短路径)

    POJ 1511 Invitation Cards / UVA 721 Invitation Cards / SPOJ Invitation / UVAlive Invitation Cards / ...

  3. poj 1511 Invitation Cards (最短路)

    Invitation Cards Time Limit: 8000MS   Memory Limit: 262144K Total Submissions: 33435   Accepted: 111 ...

  4. POJ 1511 Invitation Cards(单源最短路,优先队列优化的Dijkstra)

    Invitation Cards Time Limit: 8000MS   Memory Limit: 262144K Total Submissions: 16178   Accepted: 526 ...

  5. poj 1511 Invitation Cards(最短路中等题)

    In the age of television, not many people attend theater performances. Antique Comedians of Malidine ...

  6. POJ 1511 Invitation Cards (最短路spfa)

    Invitation Cards 题目链接: http://acm.hust.edu.cn/vjudge/contest/122685#problem/J Description In the age ...

  7. [POJ] 1511 Invitation Cards

    Invitation Cards Time Limit: 8000MS   Memory Limit: 262144K Total Submissions: 18198   Accepted: 596 ...

  8. DIjkstra(反向边) POJ 3268 Silver Cow Party || POJ 1511 Invitation Cards

    题目传送门 1 2 题意:有向图,所有点先走到x点,在从x点返回,问其中最大的某点最短路程 分析:对图正反都跑一次最短路,开两个数组记录x到其余点的距离,这样就能求出来的最短路以及回去的最短路. PO ...

  9. POJ 1511 Invitation Cards (spfa的邻接表)

    Invitation Cards Time Limit : 16000/8000ms (Java/Other)   Memory Limit : 524288/262144K (Java/Other) ...

  10. Poj 1511 Invitation Cards(spfa)

    Invitation Cards Time Limit: 8000MS Memory Limit: 262144K Total Submissions: 24460 Accepted: 8091 De ...

随机推荐

  1. HTML5 CSS JavaScript在网页中扮演的角色

    HTML (content layer) CSS (presentation layer) JavaScript (Interactive layer) 参考链接: [1] https://www.y ...

  2. ubuntu 14.04 安装 eclipse

    在安装 eclipse 之前必须先安装 jdk 1. 卸载默认的 jdk,以防安装出错 sudo apt-get purge openjdk* 2. 安装 jdk1.8.0_111 下载jdk1.8. ...

  3. 同步阿里云镜像到本地,在本地搭建YUM仓库

    1.下载阿里云镜像repo文件 项目使用CentOS6系统,因此我下载的文件是: # CentOS-Base.repo # # The mirror system uses the connectin ...

  4. Linux系统编程【转】

    转自:https://blog.csdn.net/majiakun1/article/details/8558308 一.Linux系统编程概论 1.1 系统编程基石 syscall: libc:标准 ...

  5. jq常用功能操作

    //表示所有选中的商品 var $goods=$(".goods:checked"); var arr=[]; for(i=0;i<$goods.length;i++){ a ...

  6. 【转】C++ map的基本操作和使用

    1.map简介 map是一类关联式容器.它的特点是增加和删除节点对迭代器的影响较小,除了那个操作节点,对其它的节点都没有什么影响.对于迭代器来说,可以修改实值,而不能修改key. 2.map的功能 自 ...

  7. 重装系统windows

    1 boot没有usb启动选项,驱动没加载好,先进入xp系统正确安装usb驱动,重启 2 file not fount bootmgr 用pe自带修复程序修复,修复驱动盘符为最小那个,非c盘 3 wi ...

  8. GitHub学习一-本地电脑与GitHub绑定

    1.创建ssh key $ ssh-keygen -t rsa -C "your_email@youremail.com"安装完git,右键git bash here,创建ssh ...

  9. 随机森林学习-sklearn

    随机森林的Python实现 (RandomForestClassifier) # -*- coding: utf- -*- """ RandomForestClassif ...

  10. js闭包之应用场景

    闭包的解释 当函数可以记住并访问所在的词法作用域,即使函数是在当前词法作用域之外执行,这时就产生了闭包 在javascript中,只有函数内部的子函数才能读取局部变量,所以说,闭包可以简单理解成“定义 ...