LA 4080 (多源最短路径+边修改+最短路径树)
题目链接: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 (多源最短路径+边修改+最短路径树)的更多相关文章
- 源码中修改Android的开机画面和动画【转】
本文转载自:http://blog.csdn.net/dddxxxx/article/details/54343976 参照文章:http://blog.csdn.net/a345017062/art ...
- 死磕以太坊源码分析之MPT树-下
死磕以太坊源码分析之MPT树-下 文章以及资料请查看:https://github.com/blockchainGuide/ 上篇主要介绍了以太坊中的MPT树的原理,这篇主要会对MPT树涉及的源码进行 ...
- 【bzoj1901】dynamic ranking(带修改主席树/树套树)
题面地址(权限题) 不用权限题的地址 首先说说怎么搞带修改主席树? 回忆一般的kth问题,我们的主席树求的是前缀和,这样我们在目标区间的左右端点的主席树差分下就能求出kth. 那么我们如何支持修改操作 ...
- 【bzoj1901】dynamic ranking(带修改主席树)
传送门(权限) 传送门(非权限) 花了一晚上总算把代码调好了……才知道待修改主席树怎么操作…… 然而还是一知半解orz…… 先说说我的理解吧 我们一般建主席树的时候都是直接在序列上建的 但是如果有修改 ...
- [BZOJ3295] [Cqoi2011]动态逆序对(带修改主席树)
题目描述 对于序列A,它的逆序对数定义为满足i<j,且Ai>Aj的数对(i,j)的个数.给1到n的一个排列,按照某种顺序依次删除m个元素,你的任务是在每次删除一个元素之前统计整个序列的逆序 ...
- Codeforces Round #538 (Div. 2) F 欧拉函数 + 区间修改线段树
https://codeforces.com/contest/1114/problem/F 欧拉函数 + 区间更新线段树 题意 对一个序列(n<=4e5,a[i]<=300)两种操作: 1 ...
- POJ 2528 - Mayor's posters - [离散化+区间修改线段树]
题目链接:http://poj.org/problem?id=2528 Time Limit: 1000MS Memory Limit: 65536K Description The citizens ...
- 少年,想学带修改主席树吗 | BZOJ1901 带修改区间第k小
少年,想学带修改主席树吗 | BZOJ1901 带修改区间第k小 有一道题(BZOJ 1901)是这样的:n个数,m个询问,询问有两种:修改某个数/询问区间第k小. 不带修改的区间第k小用主席树很好写 ...
- BZOJ 1901: Zju2112 Dynamic Rankings | 带修改主席树
题目: emmmm是个权限题 题解: 带修改主席树的板子题,核心思想是用树状数组维护动态前缀和的性质来支持修改 修改的时候修改类似树状数组一样进行logn个Insert 查询的时候同理,树状数组的方法 ...
随机推荐
- urllib2加入header及解决中文乱码问题
def main(): url = "http://www.douban.com" #伪装桌面浏览器 headers = {'User-Agent':'Mozilla/5.0 (W ...
- 用Matplotlib绘制二维图像
唠叨几句: 近期在做数据分析,需要对数据做可视化处理,也就是画图,一般是用Matlib来做,但Matlib安装文件太大,不太想直接用它,据说其代码运行效率也很低,在网上看到可以先用Java做数据处理, ...
- (部署新java程序,程序报错,需copy的一个包)——java使用siger 获取服务器硬件信息
mcat-siger.sh 查看是否安装siger rsync -aPuv /usr/lib64/libsigar-amd64-linux.so $i:/usr/lib64/ java使用siger ...
- Netbeans 配置英文和默认utf-8编码
JAVA SDK 下载地址 http://www.oracle.com/technetwork/java/javase/downloads 解决办法: 进入netbeans 6.5\etc ,其中ne ...
- iOS 中的第三方库管理工具
xcode没有android studio中的gradle进行第三方库管理,但是有第三方的库管理工具CocoaPods,https://github.com/CocoaPods/CocoaPods/w ...
- iOS 推荐学习__bridge等ARC知识的好资料
请下载 iOS5 by Tutorials!写得很好的!
- POSIX线程
大多数线程函数以pthread_开头,.h为pthread.h, 用-lpthread来链接线程库. 编写多线程时,定义宏_REENTRANT告诉编译器需要可重入,此宏必须位于任何#include ...
- hadoop机架感知
背景 分布式的集群通常包含非常多的机器,由于受到机架槽位和交换机网口的限制,通常大型的分布式集群都会跨好几个机架,由多个机架上的机器共同组成一个分布式集群.机架内的机器之间的网络速度通常都会高于跨机架 ...
- mysql 超级管理员
mysql> grant all privileges on *.* to 'master'@'%' identified by '3306' with grant option; Query ...
- 如何在postgresql中模拟oracle的dual表,来测试数据库最基本的连接功能?
还好,网上弄到的,,没有dual的数据库,可以试图用select函数不带from数据表的方式来实现返回值. 一段测试代码: try: conn = psycopg2.connect(database= ...