题目链接:http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=32266

题目大意:①先求任意两点间的最短路径累加和,其中不连通的边权为L   ②删除任意一条边,求全局最短路径和的最大值。

解题思路

首先说下多源最短路中,floyd和和优先队列优化的dijkstra的取舍。floyd比较好拍,dijkstra具有常数有势,以及灵活性(如求第二问的时候)。

本题最烦人的是枚举删除一条边,按照正常思维,要重新做n次dijkstra,复杂度已经到了可怕的(n*m^2*logn),那么是否有必要每次修改一条边的时候,全源重新做一次最短路呢?

答案是否定的。只要构建一颗最短路径树即可。

不要被名字唬住,其实就是一个二维数组,belong[边id][s点],即初次做全源Dijkstra的时候,为每条边进行标记,标记内容为本次dij的s点。

注意这里的边指的是输入边id,不是图中的边(这题是无向图,输入边被add了两次)。

枚举删除边时,如果belong[边id][s点]=true,则说明这条边与这次的单源dij有关,必须重新dij。如果为false,则无关,值还是初次做dij的值。

标记belong的方法:在每次优先队列出队的时候,对出队点所在的入队前的边进行标记,具体方法是开一个p数组,每次Relax的时候,记录一下Relax的边即可。之后入队在取出,p数组就能立刻取出入队前的边了。

本题存在重边,不支持的重边的数据结构注意了,尤其是邻接表。推荐链式前向星。

另外两个问的结果都超过了int32。

#include "cstdio"
#include "queue"
#include "cstring"
using namespace std;
#define maxn 155
#define maxp 2005
#define inf 1<<28
#define LL long long
struct Edge
{
int next,to,d,id;
}e[maxp*];
struct status
{
int d,p;
status(int d,int p):d(d),p(p) {}
bool operator < (const status &a) const {return d > a.d;}
};
int n,m,l,tol,head[maxn],d[maxn],p[maxn],dis[maxn][maxn];
LL ans1,ans2,w[maxn];
bool vis[maxn],belong[maxp][maxn],del[maxp];
void addedge(int u,int v,int c,int id)
{
e[tol].id=id;
e[tol].d=c;
e[tol].to=v;
e[tol].next=head[u];
head[u]=tol++;
}
void dijkstra1(int s)
{
memset(vis,false,sizeof(vis));
memset(p,,sizeof(p));
for(int i=;i<=n;i++) d[i]=(i==s?:inf);
priority_queue<status> Q;
Q.push(status(,s));
while(!Q.empty())
{
status tt=Q.top();Q.pop();
int x=tt.p;
if(vis[x]) continue;
vis[x]=true;
belong[p[x]][s]=true;
for(int i=head[x];i!=-;i=e[i].next)
{
int v=e[i].to;
if(d[x]+e[i].d<d[v])
{
p[v]=e[i].id;
d[v]=d[x]+e[i].d;
Q.push(status(d[v],v));
}
}
}
for(int i=;i<=n;i++)
{
if(d[i]==inf)
{
ans1+=l;
w[s]+=l;
}
else
{
ans1+=d[i];
w[s]+=d[i];
}
}
}
LL dijkstra2(int s)
{
memset(vis,false,sizeof(vis));
for(int i=; i<=n; i++) d[i]=(i==s?:inf);
priority_queue<status> Q;
Q.push(status(,s));
while(!Q.empty())
{
status tt=Q.top();
Q.pop();
int x=tt.p;
if(vis[x]) continue;
vis[x]=true;
for(int i=head[x]; i!=-; i=e[i].next)
{
if(del[e[i].id]) continue; //标记为删除的边跳过
int v=e[i].to;
if(d[x]+e[i].d<d[v])
{
d[v]=d[x]+e[i].d;
Q.push(status(d[v],v));
}
}
}
long long tt=;
for(int i=; i<=n; i++)
{
if(d[i]==inf) tt+=l;
else tt+=d[i];
}
return tt;
}
int main()
{
int u,v,c;
while(scanf("%d%d%d",&n,&m,&l)!=EOF)
{
memset(head,-,sizeof(head));
memset(belong,false,sizeof(belong));
memset(del,false,sizeof(del));
memset(w,,sizeof(w));
tol=ans1=ans2=;
for(int i=;i<=m;i++)
{
scanf("%d%d%d",&u,&v,&c);
addedge(u,v,c,i);
addedge(v,u,c,i);
}
for(int i=;i<=n;i++)
dijkstra1(i);
for(int i=;i<=m;i++)
{
LL tt=;
del[i]=true;
for(int j=;j<=n;j++)
if(belong[i][j]) tt+=dijkstra2(j);
else tt+=w[j];
del[i]=false;
ans2=max(ans2,tt);
}
printf("%lld %lld\n",ans1,ans2);
}
}
2814660 neopenx UVALive 4080 Accepted 0 KB 409 ms C++ 4.5.3 2990 B 2014-10-04 18:34:44  

LA 4080 (多源最短路径+边修改+最短路径树)的更多相关文章

  1. 源码中修改Android的开机画面和动画【转】

    本文转载自:http://blog.csdn.net/dddxxxx/article/details/54343976 参照文章:http://blog.csdn.net/a345017062/art ...

  2. 死磕以太坊源码分析之MPT树-下

    死磕以太坊源码分析之MPT树-下 文章以及资料请查看:https://github.com/blockchainGuide/ 上篇主要介绍了以太坊中的MPT树的原理,这篇主要会对MPT树涉及的源码进行 ...

  3. 【bzoj1901】dynamic ranking(带修改主席树/树套树)

    题面地址(权限题) 不用权限题的地址 首先说说怎么搞带修改主席树? 回忆一般的kth问题,我们的主席树求的是前缀和,这样我们在目标区间的左右端点的主席树差分下就能求出kth. 那么我们如何支持修改操作 ...

  4. 【bzoj1901】dynamic ranking(带修改主席树)

    传送门(权限) 传送门(非权限) 花了一晚上总算把代码调好了……才知道待修改主席树怎么操作…… 然而还是一知半解orz…… 先说说我的理解吧 我们一般建主席树的时候都是直接在序列上建的 但是如果有修改 ...

  5. [BZOJ3295] [Cqoi2011]动态逆序对(带修改主席树)

    题目描述 对于序列A,它的逆序对数定义为满足i<j,且Ai>Aj的数对(i,j)的个数.给1到n的一个排列,按照某种顺序依次删除m个元素,你的任务是在每次删除一个元素之前统计整个序列的逆序 ...

  6. Codeforces Round #538 (Div. 2) F 欧拉函数 + 区间修改线段树

    https://codeforces.com/contest/1114/problem/F 欧拉函数 + 区间更新线段树 题意 对一个序列(n<=4e5,a[i]<=300)两种操作: 1 ...

  7. POJ 2528 - Mayor's posters - [离散化+区间修改线段树]

    题目链接:http://poj.org/problem?id=2528 Time Limit: 1000MS Memory Limit: 65536K Description The citizens ...

  8. 少年,想学带修改主席树吗 | BZOJ1901 带修改区间第k小

    少年,想学带修改主席树吗 | BZOJ1901 带修改区间第k小 有一道题(BZOJ 1901)是这样的:n个数,m个询问,询问有两种:修改某个数/询问区间第k小. 不带修改的区间第k小用主席树很好写 ...

  9. BZOJ 1901: Zju2112 Dynamic Rankings | 带修改主席树

    题目: emmmm是个权限题 题解: 带修改主席树的板子题,核心思想是用树状数组维护动态前缀和的性质来支持修改 修改的时候修改类似树状数组一样进行logn个Insert 查询的时候同理,树状数组的方法 ...

随机推荐

  1. Python os.system 和 os.popen的区别

    (1) os.system # 仅仅在一个子终端运行系统命令,而不能获取命令执行后的返回信息 system(command) -> exit_statusExecute the command ...

  2. 在 FREEBUF 投放广告

    在 FREEBUF 投放广告 FreebuF黑客与极客—高质量的全球互联网安全媒体,同时也是爱好者们交流.分享安全技术的最佳平台.本站读者群以IT.政企信息安全人员.互联网安全爱好者和学生为主,对互联 ...

  3. php中global与$GLOBALS的用法及区别

    php中global 与 $GLOBALS[""] 差别 原本觉得global和$GLOBALS除了写法不一样觉得,其他都一样,可是在实际利用中发现2者的差别还是很大的! 先看下面 ...

  4. 隐藏Nginx/Apache版本号的安全性与方法

    一.先介绍nginx隐藏版本号的方法. 搭建好nginx或者apache,为了安全起见我们都会隐藏他们的版本号,这边讲的是nginx的版本号,如果你也想隐藏apache的版本号,那请点前面的链接.请看 ...

  5. 【SpringMVC】SpringMVC系列13之关于 mvc:annotation-driven

    13.关于 mvc:annotation-driven 13.1.概述      会自动注册RequestMappingHandlerMapping.RequestMappingHandlerAdap ...

  6. C++多态公有继承

    面向对象的三个基本特征 面向对象的三个基本特征是:封装.继承.多态.其中,封装可以隐藏实现细节,使得代码模块化:继承可以扩展已存在的代码模块(类):它们的目的都是为了——代码重用.而多态则是为了实现另 ...

  7. object-c 继承多态 动态数据类型

    在c#中我们知道有继承的.同样在object-c中也有继承. 例如我们写一个人类(父),一个学生类.我们可以这么写: demo: @interface Person:NSobject{ NSStrin ...

  8. poj1185

    状态压缩dp #include <cstdio> #include <cstring> #include <cstdlib> #include <iostre ...

  9. iOS 基于UIWebView的应用特点

    现在有许多ios应用都是基于UIWebView的应用,比如phonegap做出的那些程序.最近我也接触到了一个类似的程序,今天在这里分享一下这类程序的特点. 用UIWebView来实现程序,自然少不了 ...

  10. HDU4870 Rating(概率)

    第一场多校,感觉自己都跳去看坑自己的题目里去了,很多自己可能会比较擅长一点的题目没看,然后写一下其中一道概率题的题解吧,感觉和自己前几天做的概率dp的思路是一样的.下面先来看题意:一个人有两个TC的账 ...