【思路】

题目给出一棵树。第\(i\)步拆的一定是第\(i\)层与第\(i+1\)层之间的连边,否则不是最优(自行证明即可),所以可以暴力枚举每一次拆哪一个节点与上一个节点的连边。

把所有节点所在的层数存下来,一号点在第\(1\)层,枚举每一层的每个节点(由于\(1\)号节点已经被感染,从第二层开始搜索就可以了)

大概可分为以下几步:

  • 存好一整棵树

  • 把每一层的节点都存在一个数组里面

  • 标记以ii号节点为根节点的子树的节点个数

  • 标记与回溯

  • 暴力搜索

【细节精讲】

1、树的存储

关于多叉树的存储,这里介绍一种简单有效的方法。考虑如下代码:

struct Node
{
int father;
int child[MAXN];
}tree[MAXN];

\(tree[i]\)存\(i\)号节点的所有信息:

\(father\)存父亲(在这题没有用) ; \(child[]\)存它所有的孩子 ; \(child[0]\)是它孩子的个数。

由于数据范围很小,我们不用担心造成空间过多的浪费。

结构体构建完成之后,我们就可以在读入的同时把整棵树存好。

n=read();p=read();
for(int i=1;i<=p;i++)
{
int x=read(),y=read();
if(x>y)swap(x,y);
tree[y].father=x;
tree[x].child[++tree[x].child[0]]=y;
}

2 、标记深度

如果能够理解,标记深度是比较简单的。

如图:我们令\(1\)号节点的深度为\(1\) ; 则\(2,3\)节点深度为\(2\) ; \(4,5,6,7\)节点的深度为\(3\); \(8\)节点的深度为\(4\)。这棵树一共有\(4\)层。

代码用\(deep[i][j]\)存第\(i\)层第\(j\)个节点的编号。\(deep[i][0]\)是第\(i\)层一共的节点数。

inline void getdeep(int now,int Nowdeep)//当前的节点标号是now,层数是Nowdeep
{
maxdeep=max(maxdeep,Nowdeep);//标记一共有几层
for(int i=1;i<=tree[now].child[0];i++)
{
deep[Nowdeep][++deep[Nowdeep][0]]=tree[now].child[i];//把这个节点放到第i层的数组中
getdeep(tree[now].child[i],Nowdeep+1);//以这个点为父节点继续标记它的儿子。每个节点的深度等于它父节点的深度+1
}
}

3、切断问题

我们知道,只要一个点与上层点的传播途径被切断,即这个点不会得传染病,那么以这个点为根节点的整个子树都应该被标记为安全。

这一段代码用来标记\(now\)这个节点为根节点的子树一共有多少节点,存在\(num[]\)中。

inline int getnum(int now)
{
for(int i=1;i<=tree[now].child[0];i++)
num[now]+=getnum(tree[now].child[i]);
return num[now];
}

4、回溯

接下来,我们切断了这个节点,相应地,以这个点为根节点的子树都应该被标记。(\(tag=1\)表示标记,\(tag=0\)表示删去标记,用于回溯)

inline void work(int now,bool tag)
{
vis[now]=tag;
for(int i=1;i<=tree[now].child[0];i++)
{
vis[tree[now].child[i]]=tag;
work(tree[now].child[i],tag);
}
}

5、搜索

做完上面这些铺垫操作之后,我们可以开始整个代码的核心:搜索了。

首先可以想到如下代码

inline void DFS(int now,int cnt)
{
if((now==maxdeep))
{
ans=min(ans,cnt);
return;
}
for(int i=1;i<=deep[now][0];i++)
{
if(vis[deep[now][i]])
continue;
work(deep[now][i],1);
DFS(now+1,cnt-num[deep[now][i]]);
work(deep[now][i],0);
}
}

但是提交这段代码的话只能得80分。为什么呢?

我们可以考虑这样一棵树:

它是一条链。我们第一次只能切断1号节点和2号节点之间的连边,这样第三层所有的节点就都被标记了。那么问题是什么呢?根本就搜不到最后一层的节点,导致答案根本没有更新!

于是我们优化一下搜索代码:

inline void DFS(int now,int cnt)
{
int tot=0;//记录总数
if((now==maxdeep))
{
ans=min(ans,cnt);
return;
}
for(int i=1;i<=deep[now][0];i++)
{
if(vis[deep[now][i]])
{
tot++;
continue;
}
work(deep[now][i],1);
DFS(now+1,cnt-num[deep[now][i]]);
work(deep[now][i],0);
}
if(tot==deep[now][0])//如果全部都被访问过了,那么直接更新答案
ans=min(ans,cnt);
}

【代码】

#include<bits/stdc++.h>
using namespace std;
const int MAXN=300+10;
int n,p;
struct Node
{
int father;
int child[MAXN];
}tree[MAXN];
int num[MAXN];
int deep[MAXN][MAXN];
int maxdeep=0;
bool vis[MAXN];
int ans=0x3f3f3f3f;
inline int read()
{
int tot=0;
char c=getchar();
while(c<'0'||c>'9')
c=getchar();
while(c>='0'&&c<='9')
{
tot=(tot<<1)+(tot<<3)+c-'0';
c=getchar();
}
return tot;
}
inline void getdeep(int now,int Nowdeep)
{
maxdeep=max(maxdeep,Nowdeep);
for(int i=1;i<=tree[now].child[0];i++)
{
deep[Nowdeep][++deep[Nowdeep][0]]=tree[now].child[i];
getdeep(tree[now].child[i],Nowdeep+1);
}
}
inline int getnum(int now)
{
for(int i=1;i<=tree[now].child[0];i++)
num[now]+=getnum(tree[now].child[i]);
return num[now];
}
inline void work(int now,bool tag)
{
vis[now]=tag;
for(int i=1;i<=tree[now].child[0];i++)
{
vis[tree[now].child[i]]=tag;
work(tree[now].child[i],tag);
}
}
inline void DFS(int now,int cnt)
{
int tot=0;
if((now==maxdeep))
{
ans=min(ans,cnt);
return;
}
for(int i=1;i<=deep[now][0];i++)
{
if(vis[deep[now][i]])
{
tot++;
continue;
}
work(deep[now][i],1);
DFS(now+1,cnt-num[deep[now][i]]);
work(deep[now][i],0);
}
if(tot==deep[now][0])
ans=min(ans,cnt);
}
int main()
{
n=read();p=read();
fill(num+1,num+1+n,1);
for(int i=1;i<=p;i++)
{
int x=read(),y=read();
if(x>y)swap(x,y);
tree[y].father=x;
tree[x].child[++tree[x].child[0]]=y;
}
/*for(int i=1;i<=n;i++)
{
for(int j=1;j<=tree[i].child[0];i++)cout<<tree[i].child[j]<<" ";
cout<<endl;
}*/
getdeep(1,2);
/*for(int i=2;i<=maxdeep;i++)
{
for(int j=1;j<=deep[i][0];j++)cout<<deep[i][j]<<" ";
cout<<endl;
}*/
getnum(1);
DFS(2,n);
printf("%d\n",ans);
return 0;
}

\[\color{red}\large\text{完结撒花}
\]

参考博客

洛谷 题解 P1041 【传染病控制】的更多相关文章

  1. 洛谷 题解 UVA572 【油田 Oil Deposits】

    这是我在洛谷上的第一篇题解!!!!!!!! 这个其实很简单的 我是一只卡在了结束条件这里所以一直听取WA声一片,详细解释代码里见 #include<iostream> #include&l ...

  2. 洛谷 题解 P1600 【天天爱跑步】 (NOIP2016)

    必须得说,这是一道难题(尤其对于我这样普及组205分的蒟蒻) 提交结果(NOIP2016 天天爱跑步): OJ名 编号 题目 状态 分数 总时间 内存 代码 / 答案文件 提交者 提交时间 Libre ...

  3. 洛谷题解P4314CPU监控--线段树

    题目链接 https://www.luogu.org/problemnew/show/P4314 https://www.lydsy.com/JudgeOnline/problem.php?id=30 ...

  4. 洛谷题解 CF777A 【Shell Game】

    同步题解 题目翻译(可能有童鞋没读懂题面上的翻译) 给你三张牌0,1,2. 最初选一张,然后依次进行n次交换,交换规则为:中间一张和左边的一张,中间一张和右边一张,中间一张和左边一张...... 最后 ...

  5. 洛谷题解 CF807A 【Is it rated?】

    同步题解 题目 好吧,来说说思路: 1.先读入啦~(≧▽≦)/~啦啦啦 2.判断a[i]赛前赛后是否同分数,如果分数不同,则输出,return 0 . 3.如果同分数,则判断a[i]赛前(或赛后)是否 ...

  6. 洛谷题解 P1138 【第k小整数】

    蒟蒻发题解了 说明:此题我用的方法为桶排(我翻了翻有人用了桶排只不过很难看出来,可能有些重复的,这个题只是作为一个专门的桶排来讲解吧) (不会算抄袭吧 ‘QWaWQ’) 简单来说(会的人跳过就行): ...

  7. 【洛谷题解】P2303 [SDOi2012]Longge的问题

    题目传送门:链接. 能自己推出正确的式子的感觉真的很好! 题意简述: 求\(\sum_{i=1}^{n}gcd(i,n)\).\(n\leq 2^{32}\). 题解: 我们开始化简式子: \(\su ...

  8. 洛谷题解 P2865 【[USACO06NOV]路障Roadblocks】

    链接:https://www.luogu.org/problemnew/show/P2865 题目描述 Bessie has moved to a small farm and sometimes e ...

  9. 洛谷题解:P1209 【[USACO1.3]修理牛棚 Barn Repair】

    原题传送门:https://www.luogu.org/problemnew/show/P1209 首先,这是一道贪心题.  我们先来分析它的贪心策略.  例如,样例:  4 50 18  3 4 6 ...

随机推荐

  1. 洛谷P1901发射站

    题目 一道单调栈裸题,主要是用单调栈维护单调性,和单调队列都可以在\(O(n)\)的时间内得出单调最大值或最小值,要比堆要快. 这个题可以用h来当做单调栈的使用对象,即用单调栈来维护高度,高度是越在栈 ...

  2. Day16:小前端

    回到顶部 <style type="text/css"> body { height: 8000px; } h1 { color: #000; } img { posi ...

  3. Linux文件与文件夹的权限问题

    0x01 修改文件可读写属性的方法 例如:把index.htm 文件修改为可写可读可执行: chmod 777 index.htm 要修改目录下所有文件属性可写可读可执行: chmod 777 *.* ...

  4. Chrome 浏览器中查看 webSocket 连接信息

      1.以下代码实现一个webSocket连接,在文本输入框中输入内容,点击发送,通过服务器,返回相同的内容显示在下方. 1 <!DOCTYPE html> 2 <html lang ...

  5. webpack4-用之初体验

    众所周知,webpack进入第4个大版本已经有2个月的时间了,而且webpack团队升级更新的速度也是非常的惊人 在写下如下内容的时候webpack已经出到了4.6的版本了,剑指5.0应该是指日可待了 ...

  6. selenium之 下拉选择框Select

    今天总结下selenium的下拉选择框.我们通常会遇到两种下拉框,一种使用的是html的标签select,另一种是使用input标签做的假下拉框. 后者我们通常的处理方式与其他的元素类似,点击或使用J ...

  7. vue+elementui搭建后台管理界面(4使用font-awesome)

    使用font-awesome npm install --save font-awesome 修改 src/main.js 增加 import 'font-awesome/scss/font-awes ...

  8. 开源JS图片裁剪插件

    开源JS图片裁剪插件 一.总结 一句话总结: 要用点赞最高的插件,这样适用性最好,效果最好,出问题的概率也最低,这里电脑端和手机端都可以用的建议用 cropper.js 二.5款好用的开源JS图片裁剪 ...

  9. PHP如何解决网站大流量与高并发的问题(一)

    高并发的相关概念 在某个时间点,有多少个访问量 如果一个系统的日PV在千万以上,有可能是一个高并发的系统 QPS: 每秒钟请求或者查询的数量,在互联网领域,指每秒相应请求数(指HTTP请求) 吞吐量: ...

  10. 【内功修炼】"裁员潮",“中年危机”,该如何战胜你的焦虑

    "裁员"."中年危机"这些曾经看上去比较遥远的词汇,最近开始频繁出现在各种文章和新闻中,个人觉得这主要由两方面原因造成: 近两年,国内外经济形势严峻(更有经济学 ...