http://www.lydsy.com/JudgeOnline/problem.php?id=2286

题意:n个点的边加权树,m个询问,每次询问给出的k个点与结点1分离的最小代价。(n<=250000, sum{ki}<=500000)

  1. #include <bits/stdc++.h>
  2. using namespace std;
  3. typedef long long ll;
  4. const int N=250005, oo=0x7f7f7f7f;
  5. int dep[N], FF[N], f[N][21], mn[N][21], n, m, a[N], s[N], top, tot, nd[N];
  6. struct Gr {
  7. int ihead[N], cnt;
  8. struct E { int next, to, w; }e[N<<1];
  9. void add(int x, int y, int c=0) {
  10. e[++cnt]=(E){ihead[x], y, c}; ihead[x]=cnt;
  11. e[++cnt]=(E){ihead[y], x, c}; ihead[y]=cnt;
  12. }
  13. void dfs(int x) {
  14. FF[x]=++tot;
  15. for(int i=1; i<=20; ++i) f[x][i]=f[f[x][i-1]][i-1], mn[x][i]=min(mn[x][i-1], mn[f[x][i-1]][i-1]);
  16. for(int i=ihead[x]; i; i=e[i].next) if(e[i].to!=f[x][0]) {
  17. f[e[i].to][0]=x; dep[e[i].to]=dep[x]+1; mn[e[i].to][0]=e[i].w;
  18. dfs(e[i].to);
  19. }
  20. }
  21. int LCA(int x, int y) {
  22. if(dep[x]<dep[y]) swap(x, y);
  23. int d=dep[x]-dep[y];
  24. for(int i=20; i>=0; --i) if((d>>i)&1) x=f[x][i];
  25. if(x==y) return x;
  26. for(int i=20; i>=0; --i) if(f[x][i]!=f[y][i]) x=f[x][i], y=f[y][i];
  27. return f[x][0];
  28. }
  29. int dist(int x, int y) {
  30. int ret=oo;
  31. if(dep[x]<dep[y]) swap(x, y);
  32. int d=dep[x]-dep[y];
  33. for(int i=20; i>=0; --i) if((d>>i)&1) ret=min(ret, mn[x][i]), x=f[x][i];
  34. return ret;
  35. }
  36. ll dp(int x, int fa=0) {
  37. ll ret=0;
  38. for(int i=ihead[x]; i; i=e[i].next) if(e[i].to!=fa)
  39. ret+=min(nd[e[i].to]?(ll)oo:dp(e[i].to, x), (ll)dist(x, e[i].to));
  40. return ret;
  41. }
  42. void clr(int x, int fa=0) { for(int i=ihead[x]; i; i=e[i].next) if(e[i].to!=fa) clr(e[i].to, x); ihead[x]=0; }
  43. }g, G;
  44. bool cmp(const int &a, const int &b) { return FF[a]<FF[b]; }
  45. int main() {
  46. scanf("%d", &n);
  47. for(int i=0; i<n-1; ++i) { int x, y, c; scanf("%d%d%d", &x, &y, &c); G.add(x, y, c); }
  48. memset(mn, 0x7f, sizeof mn);
  49. G.dfs(1);
  50. int T; scanf("%d", &T);
  51. while(T--) {
  52. scanf("%d", &m);
  53. for(int i=0; i<m; ++i) scanf("%d", &a[i]), nd[a[i]]=1;
  54. sort(a, a+m, cmp);
  55. s[top=1]=1; g.clr(1); g.cnt=0;
  56. for(int i=0; i<m; ++i) {
  57. int x=a[i], lca=G.LCA(x, s[top]);
  58. while(FF[lca]<FF[s[top]]) {
  59. if(FF[lca]>=FF[s[top-1]]) {
  60. g.add(lca, s[top--]);
  61. if(lca!=s[top]) s[++top]=lca;
  62. break;
  63. }
  64. g.add(s[top], s[top-1]); --top;
  65. }
  66. if(s[top]!=x) s[++top]=x;
  67. }
  68. while(--top) g.add(s[top], s[top+1]);
  69. printf("%lld\n", g.dp(1));
  70. for(int i=0; i<m; ++i) nd[a[i]]=0;
  71. }
  72. return 0;
  73. }

  

学习了下虚树= =(为何啥玩意都喜欢加上一个名词?其实虚树 = dfs序 + lca + 单调栈

首先容易想到树dp:$f(x) = \sum_{y是孩子} min(y点必需割?w(x, y):f(y), w(x, y))$

可是复杂度$O(nm)$无法承受...

发现每一次计算很多点是不需要计算的= =比如说两个点之间的链= =用倍增就能得到链的最小值辣...

于是将图重建一个包括x个点以及他们两两之间的lca的树。【可是这样最坏也是n个点都被加入了啊QAQ比如完全二叉树= =...感觉能分分钟卡啊....】(博主纯属sb,泥忘记了有sum ki<=500000的条件么= =

方法就是先求出dfs序然后对于每次询问先对点进行dfs序排序,然后再维护一个dfs序递增的一条链(1点到x点的链)的栈,顺序加入每个点,维护以下性质(包括得到的性质):

1、对任意$i>j$,$dfn(s[i]) > dfn(s[j])$,而排序后可知任意$x$都有$dfn(x)>dfn(s[top])$

2、$f = lca(x, s[top])$在$s[1]~s[top]$中

3、当$f \neq s[top]$时,$f$到$s[top]$之间的点以及$s[top]$的子树都不再对要加入的$lca$有贡献了,因为可以用$f$替代。由前两点性质可证。

所以具体算法就是先按dfs序排序给出的节点$a$,然后依次加入每个点。令$f = lca(s[top], a[now])$,如果$f$在$s[top]$和$s[top-1]$之间,那么$f$和$s[top]$连边,删掉$s[top]$。否则一直删栈顶且连边,直到刚刚满足$f$在$s[top]$和$s[top-1]$之间。最后再加入节点$a[now]$即可。

然后注意最后树dp的时候不要忘记是新图啊!不是原来的father啊!一开始没判re了好多次啊!

(还有写完代码发现和别人的代码相似度为99%是什么鬼啊!

最坏复杂度$O(nm)$,但是数据你懂的= =(所以说虚树属于卡常啊!(博主纯属sb

【BZOJ】2286: [Sdoi2011消耗战的更多相关文章

  1. BZOJ 2286: [Sdoi2011]消耗战

    2286: [Sdoi2011消耗战 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 2082  Solved: 736[Submit][Status] ...

  2. bzoj 2286: [Sdoi2011]消耗战 虚树+树dp

    2286: [Sdoi2011]消耗战 Time Limit: 20 Sec  Memory Limit: 512 MB[Submit][Status][Discuss] Description 在一 ...

  3. bzoj 2286 [Sdoi2011]消耗战(虚树+树上DP)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2286 [题意] 给定一棵树,切断一条树边代价为ci,有m个询问,每次问使得1号点与查询 ...

  4. bzoj 3611: [Heoi2014]大工程 && bzoj 2286: [Sdoi2011消耗战

    放波建虚树的模板. 大概是用一个栈维护根节点到当前关键点的一条链,把其他深度大于lca的都弹出去. 每次做完记得复原. 还有sort的时候一定要加cmp!!! bzoj 3611 #include&l ...

  5. BZOJ 2286: [Sdoi2011]消耗战 虚树 树形dp 动态规划 dfs序

    https://www.lydsy.com/JudgeOnline/problem.php?id=2286 wa了两次因为lca犯了zz错误 这道题如果不多次询问的话就是裸dp. 一棵树上多次询问,且 ...

  6. BZOJ.2286.[SDOI2011]消耗战(虚树 树形DP)

    题目链接 BZOJ 洛谷P2495 树形DP,对于每棵子树要么逐个删除其中要删除的边,要么直接断连向父节点的边. 如果当前点需要删除,那么直接断不需要再管子树. 复杂度O(m*n). 对于两个要删除的 ...

  7. BZOJ 2286 [Sdoi2011]消耗战(虚树+树形DP)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2286 [题目大意] 出一棵边权树,每次给出一些关键点,求最小边割集, 使得1点与各个关 ...

  8. bzoj 2286: [Sdoi2011消耗战

    #include<cstdio> #include<iostream> #define M 1000009 #define N 250009 #define ll long l ...

  9. bzoj 2286 [Sdoi2011]消耗战 虚树+dp

    题目大意:多次给出关键点,求切断边使所有关键点与1断开的最小费用 分析:每次造出虚树,dp[i]表示将i和i子树与父亲断开费用 对于父亲x,儿子y ①y为关键点:\(dp[x]\)+=\(dismn( ...

  10. BZOJ 2286: [Sdoi2011消耗战 [DP 虚树]

    传送门 题意: 删除价值和最小的边使得$1$号点与$k$个关键点不连通 一个树形DP...但是询问多次,保证总的关键点数为$O(n)$ 先说一下这个$DP$ $f[i]$表示子树$i$中的关键点与$1 ...

随机推荐

  1. C++ 基础 构造函数的使用

  2. [Qcon] 百姓网开发总结

    拿到的PPT看了之后,发现给出的很简洁,但每个步骤用处却非常有用,我们一个个来分析: 1. 集中开发环境,这些方法看似简单,但是都是很实用的方法,在我开发中看的出来,SVN无分支就能解决我现有部门的部 ...

  3. W-数据库基础

    数据库系统由三部分组成:数据库(DB).数据库管理系统(DBMS)和数据库应用系统 数据加是用来存储数据的,里面存储两大类数据:用户数据及系统数据/数据字典,具体为系统中的用户以及用户孤权限,各种统计 ...

  4. CQRS及.NET中的参考资料

    (此文章同时发表在本人微信公众号"dotNET每日精华文章",欢迎右边二维码来关注.) 题记:CQRS作为一种设计模式,其实一点都不新鲜了.不过今天有朋友感叹.NET朋友也关注CQ ...

  5. WebRTC音视频引擎研究(1)--整体架构分析

    WebRTC技术交流群:234795279 原文地址:http://blog.csdn.net/temotemo/article/details/7530504     1.WebRTC目的     ...

  6. Windows phone 8.0 本地化遇到的两个问题

    基本上来说,按照msdn来讲的,本地化和全球化没有太多的问题,链接如下: http://msdn.microsoft.com/zh-cn/library/windowsphone/develop/ff ...

  7. JMockit

    [TOC] 简介 JMockit是基于JavaSE5中的java.lang.instrument包开发,内部使用ASM库来动态修改java的字节码,使得java这种静态语言可以想动态脚本语言一样动态设 ...

  8. ☆ ☆ VMware9虚拟机安装MAC OS X Mountain Lion 10.8.2详细图文教程 (转)

    参考  http://diybbs.zol.com.cn/1/34037_699.html 然后对安装的Mac系统进行升级到最新版本. 安装mac系统之后,再安装VMTOOLS darwin. 方法可 ...

  9. 在Salesforce中通过编写C#程序调用dataloadercliq的bat文件取触发调用data loader来批量处理数据

    通过这篇文章 http://www.cnblogs.com/mingmingruyuedlut/p/3413903.html 我们已经知道了Data Loader可以对Salesforce的Objec ...

  10. Laravel系列2入门使用

    最好的教程是官方文档! homestead安装好,就可以使用了. 安装Laravel composer create-project --prefer-dist laravel/laravel blo ...