【题意】n个点的树,1为根,要求删除一些点使得截断根节点和所有叶子结点的路径(不能删根,可以删叶子)。有m支军队在m个点上,每时刻所有军队可以走一步,最终走到的地方就是删除的点,求最短时间。

【算法】二分,贪心,倍增

【题解】

所有点同时走路,求最短时间,这样的询问通常考虑二分转化为判定性问题。(实际上,这题用二分确实没有想到,如果能想到二分整道题就好写一些了)

容易发现,每支军队贪心地往上走最优。

那么对于二分的时间,有一部分军队可以到达根,A数组记录这些军队到达根后的剩余时间,待会可以走到第二层覆盖其它节点。

有一部分军队不能到达根,处理出这些军队能覆盖多少二层节点,B数组不能覆盖的二层节点到根的路径。

AB各自排序之后,对应匹配,若A能将B全部匹配就可以满足要求,否则不能。

还有一个问题,一个军队虽然不能到达根后返回来覆盖自己,但可以直接不去根。解决方法是从小到大枚举A时,如果该点本身的二层节点还没覆盖就直接覆盖(因为该点本来就是最劣的,只要能覆盖一个二层结点就不亏)。

最后的问题是处理出不能到达根的军队能覆盖多少二层结点?可以对每个军队倍增,也可以直接一遍dfs。

dfs的具体做法是:c[x]表示x被覆盖,t[x]表示x子树的军队到x的最大剩余时间。c[x] = c[son[x]]=1 || t[x]>=0。son[x]表示x的所有儿子。

#include<cstdio>
#include<cstring>
#include<cctype>
#include<algorithm>
#define ll long long
using namespace std;
int read(){
char c;int s=,t=;
while(!isdigit(c=getchar()))if(c=='-')t=-;
do{s=s*+c-'';}while(isdigit(c=getchar()));
return s*t;
}
const int maxn=;
int n,m,tot,first[maxn],top[maxn],a[maxn];
ll dis[maxn],t[maxn];
bool b[maxn],c[maxn];
struct edge{int v,w,from;}e[maxn*];
void insert(int u,int v,int w){tot++;e[tot].v=v;e[tot].w=w;e[tot].from=first[u];first[u]=tot;}
void DFS(int x,int fa,int tp){
top[x]=tp;
for(int i=first[x];i;i=e[i].from)if(e[i].v!=fa){
dis[e[i].v]=dis[x]+e[i].w;//yu ju shun xu
DFS(e[i].v,x,tp);
}
}
void dfs(int x,int fa){
c[x]=;
for(int i=first[x];i;i=e[i].from)if(e[i].v!=fa){c[x]=;break;}
for(int i=first[x];i;i=e[i].from)if(e[i].v!=fa){
dfs(e[i].v,x);
c[x]&=c[e[i].v];
t[x]=max(t[x],t[e[i].v]-e[i].w);
}
if(t[x]>=)c[x]=;
}
int totA,totB;
struct cyc{ll num;int id;}A[maxn],B[maxn];//
bool cmp(cyc a,cyc b){return a.num<b.num;}
bool check(ll time){
memset(t,-,sizeof(t));
totA=;totB=;
for(int i=;i<=m;i++)if(dis[a[i]]<=time)b[i]=,A[++totA].num=time-dis[a[i]],A[totA].id=top[a[i]];
else b[i]=,t[a[i]]=time;
dfs(,);
for(int i=first[];i;i=e[i].from)if(!c[e[i].v])B[++totB].num=e[i].w,B[totB].id=e[i].v;
sort(A+,A+totA+,cmp);sort(B+,B+totB+,cmp);
int now=;
for(int i=;i<=totA;i++){
while(now<=totB&&c[B[now].id])now++;
if(!c[A[i].id]){c[A[i].id]=;continue;}
if(now<=totB&&A[i].num>=B[now].num){c[B[now++].id]=;}
}
while(now<=totB&&c[B[now].id])now++;//
if(now>=totB+)return ;
return ;
}
int main(){
n=read();
for(int i=;i<n;i++){
int u=read(),v=read(),w=read();
insert(u,v,w);insert(v,u,w);
}
memset(dis,,sizeof(dis));
for(int i=first[];i;i=e[i].from)dis[e[i].v]=e[i].w,DFS(e[i].v,,e[i].v);
m=read();
for(int i=;i<=m;i++)a[i]=read();
ll l=,r=1ll*n*1e9+,mid;
while(l<r){
mid=(l+r)>>;
if(check(mid))r=mid;else l=mid+;
}
if(r>1ll*n*1e9)printf("-1");else printf("%lld",l);
return ;
}

【NOIP】提高组2012 疫情控制的更多相关文章

  1. 洛谷P1084 [NOIP2012提高组Day2T3]疫情控制

    P1084 疫情控制 题目描述 H 国有 n 个城市,这 n 个城市用 n-1 条双向道路相互连通构成一棵树,1 号城市是首都,也是树中的根节点. H 国的首都爆发了一种危害性极高的传染病.当局为了控 ...

  2. 【noip 2012】提高组Day2T3.疫情控制

    Description H国有n个城市,这n个城市用n-1条双向道路相互连通构成一棵树,1号城市是首都,也是树中的根节点. H国的首都爆发了一种危害性极高的传染病.当局为了控制疫情,不让疫情扩散到边境 ...

  3. luogu P1038借教室【Noip提高组2012】

    这道题我读完题目的第一感觉是: 这不就是个线段树??用线段树维护区间最小值,检查是否满足订单要求即可判断. 对于修改操作直接在区间上进行. 据说会卡一卡线段树,但是貌似写一个懒标记,连zkw线段树都不 ...

  4. NOIP提高组2004 合并果子题解

    NOIP提高组2004 合并果子题解 描述:在一个果园里,多多已经将所有的果子打了下来,而且按果子的不同种类分成了不同的堆.多多决定把所有的果子合成一堆. 每一次合并,多多可以把两堆果子合并到一起,消 ...

  5. 计蒜客 NOIP 提高组模拟竞赛第一试 补记

    计蒜客 NOIP 提高组模拟竞赛第一试 补记 A. 广场车神 题目大意: 一个\(n\times m(n,m\le2000)\)的网格,初始时位于左下角的\((1,1)\)处,终点在右上角的\((n, ...

  6. 1043 方格取数 2000 noip 提高组

    1043 方格取数  2000 noip 提高组 题目描述 Description 设有N*N的方格图(N<=10,我们将其中的某些方格中填入正整数,而其他的方格中则放入数字0.如下图所示(见样 ...

  7. [NOIP提高组2018]货币系统

    [TOC] 题目名称:货币系统 来源:2018年NOIP提高组 链接 博客链接 CSDN 洛谷博客 洛谷题解 题目链接 LibreOJ(2951) 洛谷(P5020) 大视野在线评测(1425) 题目 ...

  8. NOIP提高组初赛难题总结

    NOIP提高组初赛难题总结 注:笔者开始写本文章时noip初赛新题型还未公布,故会含有一些比较老的内容,敬请谅解. 约定: 若无特殊说明,本文中未知数均为整数 [表达式] 表示:在表达式成立时它的值为 ...

  9. 津津的储蓄计划 NOIp提高组2004

    这个题目当年困扰了我许久,现在来反思一下 本文为博客园ShyButHandsome的原创作品,转载请注明出处 右边有目录,方便快速浏览 题目描述 津津的零花钱一直都是自己管理.每个月的月初妈妈给津津\ ...

随机推荐

  1. TCP 的有限状态机

    TCP 有限状态机的图中每一个方框都是 TCP 可能具有的状态. 每个方框中的大写英文字符串是 TCP 标准所使用的 TCP 连接状态名. 状态之间的箭头表示可能发生的状态变迁. 箭头旁边的字,表明引 ...

  2. mysql8基本配置,差点被各种坑蒙圈

    1. 下载免安装版mysql地址 https://dev.mysql.com/downloads/mysql/ 2. 基本配置 (1)解压zip包,将bin目录添加到环境变量 (2)在mysql根目录 ...

  3. perf record -c

    如果perf record -c -c后面接的是sample_period,也就是说你让这个事件没 我的loop进程一直在执行,我的CPU的频率是2.6G hz,也就是说每一秒会有2,600,000, ...

  4. FullCalendar(日程管理控件)

    (以下是我学习FullCalendar控件时,网络上收集的一些资料) jQuery.fullCalendar官方网址: http://arshaw.com/fullcalendar/ http://a ...

  5. Oracle数据库表被锁定以及去除方式

    select t2.username, t2.sid, t2.serial#, t3.object_name, t2.OSUSER, t2.MACHINE, t2.PROGRAM, t2.LOGON_ ...

  6. 【周记:距gdoi43天】

    这个星期切了几道题吧,虽然说还是想让自己搏一搏,但是毕竟自己弱嘛,而且很多东西都还没熟透&不像rausen大神都屠进前100了. 加油吧.

  7. [NOIP2012]疫情控制 贪心 二分

    题面:[NOIP2012]疫情控制 题解: 大体思路很好想,但是有个细节很难想QAQ 首先要求最大时间最小,这种一般都是二分,于是我们二分一个时间,得到一个log. 然后发现一个军队,越往上走肯定可以 ...

  8. BZOJ3165 & 洛谷4097:[HEOI2013]Segment——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=3165 https://www.luogu.org/problemnew/show/P4097 要求 ...

  9. MySQL中数据表的基本操纵

    本文基于对国家863中部软件孵化器编著的<MySQL从入门到精通>一书的操作实践.  一.创建数据表 数据表属于数据库,在创建数据表之前,应该使用语句 USE 数据库名  指定操作是在那个 ...

  10. UVA.122 Trees on the level(二叉树 BFS)

    UVA.122 Trees on the level(二叉树 BFS) 题意分析 给出节点的关系,按照层序遍历一次输出节点的值,若树不完整,则输出not complete 代码总览 #include ...