题意:给出一个n个点m条边的无向边,q次询问每次询问把一条边权值增大后问新的MST是多少,输出Sum(MST)/q。

解法:一开始想的是破圈法,后来想了想应该不行,破圈法应该只能用于加边的情况而不是修改边,因为加边可以保证以前MST不用的边加边之后也一定不用,但是修改边不能保证以前不用的边修改边之后会不会再用。

正解是参考https://blog.csdn.net/Ramay7/article/details/52236040这位大佬的。

大佬真的分析得巨好。我的理解就是:假如我们要计算dp[u][v]代表去掉MST上u-v这条边之后能替代的最好边,设u这一边的连通点集是(u1,u2,u3...),v这一边的点集是(v1,v2,v3...),那么我们朴素算法是暴力枚举每一对ui和vi然后取最小值,显然这样超时。用树形DP的方法是,我们枚举一个根节点rt,然后dfs一边计算以rt为根节点的时候的MST的所有边的dp值,计算方式就是dp[u][v]=min(dis[y][rt])(即子树中到根rt的最小值)。枚举完根节点rt之后我们的dp数组就出来了。  为什么这样能达到朴素算法一样的效果呢?因为考虑我们每一次rt对dp[u][v]的贡献,显然每一个rt都是在点集(u1,u2,u3...)中的,然后这次dfs可以计算(u1,u2,u3...)中的某一个ui和所有的(v1,v2,v3...)的点对对答案的贡献,然后所以的rt加起来必定等于(u1,u2,u3...)。  这里说得有点乱了,就是这样:

(rt=u1)x(v1,v2,v3...)+(rt=u2)*(v1,v2,v3...)+(rt=u3)*(v1,v2,v3...)+....(rt=ui)*(v1,v2,v3...) == (u1,u2,u3...)*(v1,v2,v3...)   (上面说的一大堆想说的就是这个等式的意思qwq)

最后处理下询问看看修改边在不在原MST上,就可以获得AC了。

#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
const int N=3e3+;
int n,m,fa[N],dis[N][N],dp[N][N];
LL sum,ans;
struct edge{
int x,y,z;
bool operator < (const edge &rhs) const {
return z<rhs.z;
}
}e[N*N];
bool mst[N][N]; int cnt,head[N],nxt[N<<],to[N<<];
void add_edge(int x,int y) {
nxt[++cnt]=head[x]; to[cnt]=y; head[x]=cnt;
} int getfa(int x) { return x==fa[x] ? x : fa[x]=getfa(fa[x]); } void Kruskal() {
sort(e+,e+m+);
for (int i=;i<=n;i++) fa[i]=i;
int num=;
for (int i=;i<=m;i++) {
int fx=getfa(e[i].x),fy=getfa(e[i].y);
if (fx==fy) continue;
fa[fx]=fa[fy];
add_edge(e[i].x,e[i].y); add_edge(e[i].y,e[i].x);
mst[e[i].x][e[i].y]=mst[e[i].y][e[i].x]=;
sum+=e[i].z;
if (++num==n) break;
}
} int dfs(int rt,int x,int fa) {
int Min=0x3f3f3f3f;
for (int i=head[x];i;i=nxt[i]) {
int y=to[i];
if (y==fa) continue;
int tmp=dfs(rt,y,x);
Min=min(Min,tmp);
dp[x][y]=min(dp[x][y],tmp); //用子树的Min更新dp[][]
dp[y][x]=min(dp[y][x],tmp);
}
if (dis[rt][x] && !mst[rt][x]) Min=min(Min,dis[rt][x]); //更新Min
return Min;
} int main()
{
while (scanf("%d%d",&n,&m) && n) {
cnt=; for (int i=;i<=n;i++) head[i]=;
for (int i=;i<=n;i++) for (int j=;j<=n;j++) dis[i][j]=mst[i][j]=,dp[i][j]=0x3f3f3f3f;
for (int i=;i<=m;i++) {
scanf("%d%d%d",&e[i].x,&e[i].y,&e[i].z);
e[i].x++; e[i].y++;
dis[e[i].x][e[i].y]=dis[e[i].y][e[i].x]=e[i].z;
}
sum=; ans=;
Kruskal(); for (int i=;i<=n;i++) dfs(i,i,); //每个点做根节点dfs一次 int q; scanf("%d",&q);
for (int i=;i<=q;i++) {
int x,y,z; scanf("%d%d%d",&x,&y,&z);
x++; y++;
if (!mst[x][y]) ans+=sum; //不在MST上
else { //在MST上
LL tdis=sum-dis[x][y]+min(z,dp[x][y]);
ans+=tdis;
}
}
printf("%.4lf\n",(double)ans/q);
}
return ;
}

最新文章

  1. tp框架的增删改查
  2. 《Objective-C编程》部分示例
  3. linux -小记(1) 问题:&quot;linux ifconfig查看网卡名称与配置文件不否&quot; 或 启动网卡提示“ eth0 似乎不存在, 初始化操作将被延迟”。
  4. NHibernate系列文章十六:使用程序集管理NHibernate项目(附程序下载)
  5. [Bug FIX]安装 account_check_writing模块后采购收据打印报错的问题
  6. CentOS搭建Httpd Pyhton3 Django环境
  7. [svn] linux命令——svn分支创建、合并
  8. [分享] Code::Blocks Windows Console 中文亂碼解決
  9. Android 通用获取Ip的方法(判断手机是否联网的方法)!!!
  10. Codevs_1048_石子归并_(动态规划)
  11. oracle查看死锁和处理方法
  12. C#,新建的系统服务项目有些机器不能运行
  13. 原生的UITableViewCell高度自适应,textLabel自动换行显示
  14. kafka和strom集群的环境安装
  15. callback vs async.js vs promise vs async / await
  16. Qt+QGIS二次开发:QGIS中使用QgsRubberBand类创建临时图形
  17. quartz 任务时间调度入门使用
  18. Nginx配置项优化详解(转)
  19. webSQL 增删改查
  20. 沉淀再出发:再谈java的多线程机制

热门文章

  1. CSS实现三级菜单[转]
  2. 请求体中需要的true和requests包put请求冲突了
  3. 2.WCF学习--地址
  4. VTF/AMROC安装指南
  5. SpringBoot JSON文件读取
  6. shell脚本学习(5)join
  7. php fmod()函数 语法
  8. Delphi Base64编码/解码
  9. Delphi界面篇之ListView控件
  10. [杂题]:staGame(博弈论+Trie树+DFS)