传送门


$\sum k \leq 100000$虚树套路题

设$f_{i,0/1}$表示处理完$i$以及其所在子树的问题,且处理完后$i$所在子树内是否存在$1$个关键点满足它到$i$的路径上不存在任何点被封的最小代价。

转移考虑$i$是否是关键点以及是否封$i$号点。

注意判断相邻两个点同时是关键点的情况。同时需要注意:如果虚树上两个点不相邻,可以封这两个点之间的点。

 #include<bits/stdc++.h>
 //This code is written by Itst
 using namespace std;

 inline int read(){
     ;
     ;
     char c = getchar();
     while(c != EOF && !isdigit(c)){
         if(c == '-')
             f = ;
         c = getchar();
     }
     while(c != EOF && isdigit(c)){
         a = (a << ) + (a << ) + (c ^ ');
         c = getchar();
     }
     return f ? -a : a;
 }

 ;
 struct Edge{
     int end , upEd;
 }Ed[MAXN << ] , newEd[MAXN];
 ] , newHead[MAXN] , s[MAXN] , dfn[MAXN] , num[MAXN] , dp[MAXN][] , dep[MAXN];
 int N , headS , cntEd , cntNewEd , ts , cnt;
 bool imp[MAXN];

 inline void addEd(Edge* Ed , int* head , int& cntEd , int a , int b){
     Ed[++cntEd].end = b;
     Ed[cntEd].upEd = head[a];
     head[a] = cntEd;
 }

 void init(int now , int fa){
     dfn[now] = ++ts;
     dep[now] = dep[fa] + ;
     jump[now][] = fa;
      ; jump[now][i - ] ; ++i)
         jump[now][i] = jump[jump[now][i - ]][i - ];
     for(int i = head[now] ; i ; i = Ed[i].upEd)
         if(Ed[i].end != fa)
             init(Ed[i].end , now);
 }

 inline int jumpToLCA(int x , int y){
     if(dep[x] < dep[y])
         swap(x , y);
      ; i >=  ; --i)
          << i) >= dep[y])
             x = jump[x][i];
     if(x == y)
         return x;
      ; i >=  ; --i)
         if(jump[x][i] != jump[y][i]){
             x = jump[x][i];
             y = jump[y][i];
         }
     ];
 }

 inline void create(){
     imp[] = ;
     dp[][] = dp[][] = ;
      ; i <= cnt ; ++i){
         imp[num[i]] = ;
         dp[num[i]][] = ;
         dp[num[i]][] = N + ;
     }
      ; i <= cnt ; ++i)
         if(!headS)
             s[++headS] = num[i];
         else{
             int t = jumpToLCA(s[headS] , num[i]);
             if(t != s[headS]){
                 ]] > dep[t]){
                     addEd(newEd , newHead , cntNewEd , s[headS - ] , s[headS]);
                     --headS;
                 }
                 addEd(newEd , newHead , cntNewEd , t , s[headS]);
                 if(s[--headS] != t)
                     s[++headS] = t;
             }
             s[++headS] = num[i];
         }
     ){
         addEd(newEd , newHead , cntNewEd , s[headS - ] , s[headS]);
         --headS;
     }
     )
         addEd(newEd , newHead , cntNewEd ,  , s[headS]);
     --headS;
 }

 void dfs(int now){
     ;
     for(int i = newHead[now] ; i ; i = newEd[i].upEd){
         int t = newEd[i].end;
         dfs(t);
         if(imp[now])
             ] == now)
                 dp[now][] += dp[t][];
             else
                 dp[now][] += min(dp[t][] , dp[t][] + );
         else{
             ] == now)
                 dp[now][] = min(sum + dp[t][] , dp[now][] + dp[t][]);
             else
                 dp[now][] = min(sum + dp[t][] , dp[now][] + min(dp[t][] , dp[t][] + ));
             ] == now)
                 sum += dp[t][];
             else
                 sum += min(dp[t][] , dp[t][] + );
         }
         ] > N)
             dp[now][] = N + ;
         )
             sum = N + ;
     }
     sum = ;
     for(int i = newHead[now] ; i ; i = newEd[i].upEd){
         int t = newEd[i].end;
         if(!imp[now]){
             sum += min(dp[t][] , dp[t][]);
             dp[now][] += dp[t][];
         }
         dp[t][] = dp[t][] = imp[t] = ;
     }
     if(!imp[now])
         dp[now][] = min(dp[now][] , sum + );
     newHead[now] = ;
 }

 bool cmp(int a , int b){
     return dfn[a] < dfn[b];
 }

 int main(){
 #ifndef ONLINE_JUDGE
     freopen("613D.in" , "r" , stdin);
     //freopen("613D.out" , "w" , stdout);
 #endif
     N = read();
      ; i < N ; ++i){
         int a = read() , b = read();
         addEd(Ed , head , cntEd , a , b);
         addEd(Ed , head , cntEd , b , a);
     }
     init( , );
     for(int M = read() ; M ; --M){
         cnt = read();
          ; i <= cnt ; ++i)
             num[i] = read();
         sort(num +  , num + cnt +  , cmp);
         create();
         dfs();
         ][] , dp[][]);
         printf( ? - : t);
     }
     ;
 }

CF613D Kingdom and its Cities 虚树的更多相关文章

  1. CF613D Kingdom and its Cities 虚树 树形dp 贪心

    LINK:Kingdom and its Cities 发现是一个树上关键点问题 所以考虑虚树刚好也有标志\(\sum k\leq 100000\)即关键点总数的限制. 首先当k==1时 答案显然为0 ...

  2. CF613D Kingdom and its Cities 虚树 + 树形DP

    Code: #include<bits/stdc++.h> #define ll long long #define maxn 300003 #define RG register usi ...

  3. 【CF613D】Kingdom and its Cities 虚树+树形DP

    [CF613D]Kingdom and its Cities 题意:给你一棵树,每次询问给出k个关键点,问做多干掉多少个非关键点才能使得所有关键点两两不连通. $n,\sum k\le 10^5$ 题 ...

  4. CF613D:Kingdom and its Cities(树形DP,虚树)

    Description 一个王国有n座城市,城市之间由n-1条道路相连,形成一个树结构,国王决定将一些城市设为重要城市. 这个国家有的时候会遭受外敌入侵,重要城市由于加强了防护,一定不会被占领.而非重 ...

  5. CF613D Kingdom and its Cities(虚树+贪心)

    很休闲的一个题啊 其实一看到关于\(\sum k\)的限制,就知道是个虚树的题了 首先我们把虚树建出来,然后考虑怎么计算个数呢? 我们令\(f[x]\)表示以\(x\)的子树中,剩余了多少个还没有切断 ...

  6. [CF613D]Kingdom and its Cities

    description 题面 data range \[n, q,\sum k\le 10^5\] solution 还是虚树的练手题 \(f[0/1][u]\)表示\(u\)的子树内,\(u\)是否 ...

  7. 【CF613D】Kingdom and its Cities(虚树,动态规划)

    [CF613D]Kingdom and its Cities(虚树,动态规划) 题面 洛谷 CF 翻译洛谷上有啦 题解 每次构建虚树,首先特判无解,也就是关键点中存在父子关系. 考虑\(dp\),设\ ...

  8. CodeForces - 613D:Kingdom and its Cities(虚树+DP)

    Meanwhile, the kingdom of K is getting ready for the marriage of the King's daughter. However, in or ...

  9. Codeforces Round #613 Div.1 D.Kingdom and its Cities 贪心+虚树

    题目链接:http://codeforces.com/contest/613/problem/D 题意概述: 给出一棵树,每次询问一些点,计算最少删除几个点可以让询问的点两两不连通,无解输出-1.保证 ...

随机推荐

  1. 理解jQuery中$.get、$.post、$.getJSON和$.ajax的用法

    ajax的4种方法:$.get.$.post.$getJSON.$ajax. 1.$.get $.get()方法使用GET方式来进行异步请求,它的语法结构为: $.get( url [, data] ...

  2. 安卓开发之自定义一个view弹出框

    https://www.cnblogs.com/muyuge/p/6152167.html

  3. 我的博客:C# PHP J2ee Java Android js WP Asp.net mvc Python

    <p><A target="_blank" href="http://blog.163.com/hr_company_product/" &g ...

  4. c#所有部门及其下所部门生成树形图(递归算法获取或键值对方式获取)

    部门数据库的设计: 代码: /// <summary> /// 获取部门(入口) /// </summary> /// <returns></returns& ...

  5. 洗礼灵魂,修炼python(29)--装饰器(1)—>利用经典案例解析装饰器概念

    前提必备 不急着进入正题,在前面函数作用域那一章介绍了闭包,全局变量局部变量,这里再看几个简单的闭包案例: 1):不带参数 注意: 1.这里的name属性是每个函数都有的,可以反馈函数名 2.temp ...

  6. 找Maven --> pom.xml --> groupId和artifactId的网站

    http://mvnrepository.com/ 在此记录

  7. 校园网下对VMware网络的配置

    1.参考:https://jingyan.baidu.com/article/8275fc8675c22646a03cf6b8.html 2.因为校园网的每一次连接都是自动获得IP地址的,所以虚拟机要 ...

  8. Burp Suite 抓取http、https流量配置+CA证书安装

    HTTPS协议是为了数据传输安全的需要,在HTTP原有的基础上,加入了安全套接字层SSL协议,通过CA证书来验证服务器的身份,并对通信消息进行加密.基于HTTPS协议这些特性,我们在使用Burp Pr ...

  9. 个人技术博客——linux服务器配置以及flask框架

    本次的软件工程实践,我负责我们组后台服务的搭建,我选用了bandwagon的服务器,安装的是Debian GNU/Linux,全程在root用户下操作,后端服务是用python的flask框架,数据库 ...

  10. ConcurrentLinkedQueue源码解读

    1.简介 ConcurrentLinkedQueue是JUC中的基于链表的无锁队列实现.本文将解读其源码实现. 2. 论文 ConcurrentLinkedQueue的实现是以Maged M. Mic ...