题目链接: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. PHP生成CSV文件

    CSV文件的定义这里就不多做介绍了,难能可贵的是用Excel可以直接打开CSV文件.用PHP输出CSV文件本身很简单,但是大家如果有业务需求,下面的代码可以作为参考. $tableheader = a ...

  2. js init : function ()

    这个init外面应该还有一层,比如 var a = { init: function () {...}, exit: function () {...} } 这样的话,可以用a.init()来调用这个 ...

  3. poj2778

    题意:给出字符串长度n(<=2000000000),给出不可以包含的序列,最多10个,每个长度最大是10.问长度为n的合法序列有多少个?序列中只可能包含ACTG四个字符. 分析:AC自动机(DF ...

  4. poj1185

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

  5. 菜单栏展开和收起效果(纯js)

    2014年6月25日 15:36:29 需要关注的是: 1.用cookie保存用户当前点击的菜单项,不打扰后端代码 2.通过数学计算得到要显示和隐藏的div 3.点击事件是动态绑定到a标签上的,因此当 ...

  6. Java for LeetCode 144 Binary Tree Preorder Traversal

    Given a binary tree, return the preorder traversal of its nodes' values. For example: Given binary t ...

  7. hdu 1022 Train Problem I 解题报告

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1022 又是一道栈的练习,这次也是没有用到STL中的栈来实现.用来保存操作过程的数组(process[] ...

  8. WebStorm快捷键操作

    1. ctrl + shift + n: 打开工程中的文件,目的是打开当前工程下任意目录的文件. 2. ctrl + j: 输出模板 3. ctrl + b: 跳到变量申明处 4. ctrl + al ...

  9. ubuntu 安装tigervnc

    root@iZ25q0jd99eZ:~# apt-get install tigervnc Reading package lists... Done Building dependency tree ...

  10. 第六步:Lucene查询索引

    package cn.harmel.lucene; import java.io.IOException; import java.nio.file.Paths; import org.apache. ...