题目链接

观察题目,答案明显具有单调性。

因为如果用$x$小时能够控制疫情,那么用$(x+1)$小时也一定能控制疫情。

由此想到二分答案,将问题转换为判断用$x$小时是否能控制疫情。

对于那些在$x$小时内不能够走到根节点的子节点上的军队,让他们尽量往上走即可,走到哪里是哪里,这样显然不会更劣。

对于那些在$x$小时内能走到根节点的子节点上的军队,就让他们先走到根节点的子节点上。

然后搞搞贪心即可。

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<vector>
#include<fstream>
using namespace std;
struct edge
{
int last;
int end;
int weight;
}e[100005];
int n=0,m=0,ne=0;
int a[50005],note[50005],f[50005][20];
bool c[50005];
long long k1[50005],k2[50005],dis[50005];
vector<int> s[50005];
void NewEdge(int u,int v,int w)
{
ne++;
e[ne].last=note[u];
e[ne].end=v;
e[ne].weight=w;
note[u]=ne;
}
void dfs(int x,int fx)
{
for(int i=note[x];i;i=e[i].last)
if(e[i].end!=fx)
{
dis[e[i].end]=dis[x]+e[i].weight;
f[e[i].end][0]=x;
dfs(e[i].end,x);
}
}
void calc()
{
for(int j=1;j<=16;j++)
for(int i=1;i<=n;i++)
f[i][j]=f[f[i][j-1]][j-1];
}
bool dfss(int x,int fx)
{
if(c[x]) return false;
bool flag=false;
for(int i=note[x];i;i=e[i].last)
if(e[i].end!=fx)
{
flag=true;
if(dfss(e[i].end,x)) return true;
}
if(!flag) return true;
return false;
}
bool check(long long x)
{
for(int i=1;i<=n;i++)
c[i]=false,s[i].clear();
for(int i=1;i<=m;i++)
{
int y=a[i];
long long w=0;
for(int j=16;j>=0;j--)
if(f[y][j]>1&&w+dis[y]-dis[f[y][j]]<=x)
w+=dis[y]-dis[f[y][j]],y=f[y][j];
if(f[y][0]==1) s[y].push_back(a[i]);
c[y]=true;
}
int cnt1=0,cnt2=0;
for(int i=note[1];i;i=e[i].last)
{
bool h=c[e[i].end];
c[e[i].end]=false;
bool t=dfss(e[i].end,1);
c[e[i].end]=h;
int len=s[e[i].end].size();
for(int j=0;j<len;j++)
{
if(t&&dis[s[e[i].end][j]]+e[i].weight>x)
{t=false;continue;}
k1[++cnt1]=x-dis[s[e[i].end][j]];
}
if(t) k2[++cnt2]=e[i].weight;
}
sort(k1+1,k1+cnt1+1);
sort(k2+1,k2+cnt2+1);
for(int i=1,j=1;i<=cnt2;i++)
{
while(j<=cnt1&&k1[j]<k2[i]) j++;
if(j>cnt1) return false;
j++;
}
return true;
}
int main()
{
scanf("%d",&n);
long long hi=0;
for(int i=1;i<=n-1;i++)
{
int u=0,v=0,w=0;
scanf("%d%d%d",&u,&v,&w),hi+=w;
NewEdge(u,v,w);
NewEdge(v,u,w);
}
scanf("%d",&m);
for(int i=1;i<=m;i++) scanf("%d",&a[i]);
dfs(1,0),calc();
long long l=-1,r=hi;
while(l+1<r)
{
long long mid=(l+r)>>1;
if(check(mid)) r=mid;
else l=mid;
}
if(!check(r)) printf("-1");
else printf("%lld",r);
return 0;
}

Luogu P1084

Luogu P1084 疫情控制 | 二分答案 贪心的更多相关文章

  1. LUOGU P1084 疫情控制(二分+贪心+树上倍增)

    传送门 解题思路 比较神的一道题.首先发现是最小值问题,并且具有单调性,所以要考虑二分答案.其次有一个性质是军队越靠上越优,所以我们要将所有的军队尽量向上提,这一过程我们用倍增实现.发现这时有两种军队 ...

  2. luogu P1084疫情控制 二分

    链接 loj luogu太水不要去了. 思路 二分. 每个军队在一定的时间内越往上越好. 注意一个军队可以跨过1去帮别的. 把能到1脚下的点都存下来特判. 有一种情况是这个子树内只有一个军队,但这个军 ...

  3. NOIP2012疫情控制(二分答案+树上贪心)

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

  4. Luogu1084 NOIP2012D2T3 疫情控制 二分答案、搜索、贪心、倍增

    题目传送门 题意太长就不给了 发现答案具有单调性(额外的时间不会对答案造成影响),故考虑二分答案. 贪心地想,在二分了一个时间之后,军队尽量往上走更好.所以我们预处理倍增数组,在二分时间之后通过倍增看 ...

  5. 洛谷 P1084 疫情控制 —— 二分+码力

    题目:https://www.luogu.org/problemnew/show/P1084 5个月前曾经写过一次,某个上学日的深夜,精疲力竭后只有区区10分,从此没管... #include< ...

  6. luogu P1084 疫情控制

    传送门 首先,所有军队又要尽量往上走,这样才能尽可能的封锁更多的到叶子的路径 而随着时间的增加,能封锁的路径也就越来越多,所以可以二分最终的时间 然后对于每个时间,就让能走到根的军队走到根,记录到根上 ...

  7. luogu1084 [NOIp2012]疫情控制 (二分答案+倍增+dfs序)

    先二分出一个时间,把每个军队倍增往上跳到不能再跳 然后如果它能到1号点,就记下来它跳到1号点后剩余的时间:如果不能,就让它就地扎根,记一记它覆盖了哪些叶节点(我在这里用了dfs序+差分,其实直接dfs ...

  8. 疫情控制 2012年NOIP全国联赛提高组(二分答案+贪心)

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

  9. BZOJ_2196_[Usaco2011 Mar]Brownie Slicing_二分答案+贪心

    BZOJ_2196_[Usaco2011 Mar]Brownie Slicing_二分答案+贪心 Description Bessie烘焙了一块巧克力蛋糕.这块蛋糕是由R*C(1 <= R,C ...

随机推荐

  1. nodejs安装 Later version of Node.js is already installed. Setup will now exit 及 node与npm版本不符

    暴力删除nodejs导致无法重新安装  Later version of Node.js is already installed. Setup will now exit 1.电脑全局搜索nodej ...

  2. 【简单数据结构】链表--洛谷P1160

    题目描述 一个学校里老师要将班上NN个同学排成一列,同学被编号为1\sim N1∼N,他采取如下的方法: 先将11号同学安排进队列,这时队列中只有他一个人: 2-N2−N号同学依次入列,编号为i的同学 ...

  3. 5UCMS判断当前栏目高亮(用于当前所在栏目加背景图片或颜色)

    5UCMS判断当前栏目高亮标签 比较简单的是频道页(channel.html): 大类代码: <!--menu:{ $row=10 $table=channel }--> <li { ...

  4. Linux系列(31) - rpm命令管理之升级与卸载命令(3)

    升级命令 rpm -Uvh 包全名(要升级到的软件版本),如果没有安装这个软件的任何版本,这个命令可以代替rpm -ivh. rpm -Uvh 包全名 选项: - -U(upgrade):升级 卸载命 ...

  5. CI框架 core

    https://blog.csdn.net/admin_admin/article/details/51769805 1.扩展控制器 1.在application/core新建一个自己的控制器(MY_ ...

  6. jmeter跑脚本的注意事项

    指标主要看以下几点: 1.jmeter性能测试的报告,不要看平均响应时间,而是看90%响应时间,一般不能超过3s,超过3s则不符合标准2.响应时间超过3s就要优化,但不是平均响应时间,因为最小响应时间 ...

  7. 基于深度学习的建筑能耗预测01——Anaconda3-4.4.0+Tensorflow1.7+Python3.6+Pycharm安装

    基于深度学习的建筑能耗预测-2021WS-02W 一,安装python及其环境的设置 (写python代码前,在电脑上安装相关必备的软件的过程称为环境搭建) · 完全可以先安装anaconda(会自带 ...

  8. 初探webpack之编写plugin

    初探webpack之编写plugin webpack通过plugin机制让其使用更加灵活,以适应各种应用场景,当然也大大增加了webpack的复杂性,在webpack运行的生命周期中会广播出许多事件, ...

  9. CentOS 7.9+19c单实例静默安装

    一.环境准备 二.解压文件 三.文件配置 四.安装 五.相关调整 六.打补丁 一.环境准备0.依赖包安装 rpm -q --qf '%{NAME}-%{VERSION}-%{RELEASE} (%{A ...

  10. 微服务Cloud整体聚合工程创建过程

    1.父工程创建及使用 使用idea开发工具,选择File-new- project ,在选项中选择Maven工程,选择jdk版本1.8,勾选maven-archetype-site,点击next,输入 ...