P1041 传染病控制(noip2003)(搜索)
呃呃呃。。。真的是惨烈啊。。。
今天的模拟赛是真的惨。。。。。
本题,正解居然是搜索!!!!!!
蒟蒻自己歪歪了一个貌似是正解但是却连一半都没过的错解。
先解释一下自己的dp思路把。
$f[i][u][v][0/1]$表示第i秒(等效于深度)uv之间的边断还是不断。
转移有两个:
$f[dep[u]][u][v][1]=size[v];$表示此边断,然后下面的点全对答案有贡献
然后找到v的儿子
$f[dep[u]][u][v][0]=max(f[dep[u]][u][v][0],max(f[dep[v]][u][j][0],f[dep[v]][u][j][1]));$
临收卷前发现貌似第二维可以压掉,也就是:$f[dep[v]][v][0/1]$表示v上面那个边要不要删。
然后发现随手hack....
于是,我发现答案统计的时候,会有惊人的重复计算。
于是,我歪歪了好久,怎么标记统计过了,怎么返回,怎么....
然后....发现最后好像被我打成了贪心???
我在每一层去最大值,然后标记,返回,从别的子树里找最大值,然后一直重复到底层。
于是...我抱着能A掉这题的心情,笑了起来。。。
然后我rand了好多的的数据,都没有hack掉,我更开心了。
(先放dp代码:)
#include<bits/stdc++.h>
#define g(x) for(int i=head[x];i;i=e[i].next)
#define l(x) for(int l=head[x];l;i=e[l].next)
using namespace std;
const int maxn=;
int n,p;
struct edge
{
int to,next;
}e[maxn];
int head[maxn],cnt;
void addedge(int from,int to)
{
e[++cnt]=(edge){to,head[from]};
head[from]=cnt;
}
int dep[maxn];
int size[maxn];
int fa[maxn];
void dfs(int u,int f)
{
dep[u]=dep[f]+;
size[u]=;
fa[u]=f;
g(u)
{
int v=e[i].to;
if(v==f)
continue;
dfs(v,u);
size[u]+=size[v];
}
}
int f[maxn][maxn][];
int mp;
void dp(int u)
{
g(u)
{
int v=e[i].to;
if(v==fa[u])
continue;
dp(v);
f[dep[u]][u][v][]=size[v];
for(int j=;j<=n;j++)
if(fa[j]==v)
f[dep[u]][v][]=max(f[dep[u]][v][],max(f[dep[v]][j][],f[dep[v]][j][]));
}
}
int ans;
int vis[maxn];
void getans(int u)
{
int m=;
for(int i=head[u];i;i=e[i].next)
{
int v=e[i].to;
if(v==fa[u])
continue;
m=max(m,max(f[dep[u]][v][],f[dep[u]][v][]));
}
for(int i=head[u];i;i=e[i].next)
{
int v=e[i].to;
if(v==fa[u])
continue;
if(m==f[dep[u]][v][]&&vis[u]==)
{
ans+=m;
vis[u]=;
continue;
}
else
{
getans(v);
}
}
} int main()
{
scanf("%d%d",&n,&p);
for(int i=;i<=p;i++)
{
int x,y;
scanf("%d%d",&x,&y);
addedge(x,y);
addedge(y,x);
}
addedge(,);
dfs(,);
dp();
for(int i=;i<=n;i++)
mp=max(mp,dep[i]);
getans();
printf("%d",n-ans);
return ;
}
dp错在于:它会漏状态,也就是说:
我的dp是从子树开始更新,然鹅,即使我在答案统计上避免了重复,它也从一个dp变成了一个贪心,也就是说,它貌似是等效于每次在一层之内找一个子树最大的点加上,这个贪心是错的。
从而得出:dp都是错的。
因为它总会漏状态,如果从儿子更新,同深度的状态没法记录,如果从孙子更新,它的曾孙子没法更新。总而言之就是没法更新完全,也就是有后效性。
所以,如果想得出正解,就得从最下开始更新子树,而这点dp是没法做到的,所以,能够猜到正解是什么:
暴力
怎么暴力怎么写,直接更新,不需要记录状态。
这个复杂度简直玄学。。。
如果是分叉很多,这个dfs能被卡成皮皮,但是这题的数据貌似非常水....
所以,暴力的思路就是:
记录每个深度的所有点,然后对于这些同深度的点进行更新,找出最大值,一次一次向上更新,一直到根节点。
感觉这个深搜巨tm像广搜啊....但是....好吧我错了
代码:
#include <bits/stdc++.h>
using namespace std;
const int maxn = ;
int n, p;
vector<int> g[maxn], dep[maxn];
int vis[maxn];
int ans, s;
void dfs(int u, int d)
{
dep[d].push_back(u);
for (int i = ; i < g[u].size(); i++)
{
dfs(g[u][i], d + );
}
}
int work(int u, int t)
{
int ss = ;
vis[u] = t;
for (int i = ; i < g[u].size(); i++)
{
ss += work(g[u][i], t);
}
return ss;
}
void getans(int d)
{
for (int i = ; i < dep[d].size(); i++)
{
if (vis[dep[d][i]])
continue;
s += work(dep[d][i], );
ans = max(s, ans);
getans(d + );
s -= work(dep[d][i], );
}
}
int main()
{
scanf("%d%d", &n, &p);
for (int i = ; i <= p; i++)
{
int x, y;
scanf("%d%d", &x, &y);
if (x > y)
swap(x, y);
g[x].push_back(y);
}
dfs(, );
getans();
printf("%d", n - ans);
return ;
}
P1041 传染病控制(noip2003)(搜索)的更多相关文章
- Luogu P1041 传染病控制(搜索)
P1041 传染病控制 题意 题目背景 近来,一种新的传染病肆虐全球.蓬莱国也发现了零星感染者,为防止该病在蓬莱国大范围流行,该国政府决定不惜一切代价控制传染病的蔓延.不幸的是,由于人们尚未完全认识这 ...
- 【搜索】P1041 传染病控制
题目链接:P1041 传染病控制 题解: 这个题目是看别人的博客做出来的,其实挺不错的一个题目,考察的东西挺多的, 一个dfs可以处理5个东西: 1.找出父亲 2.找出深度 3.每一层的节点,存进Ve ...
- P1041 传染病控制(dfs)
P1041 传染病控制 题目背景 近来,一种新的传染病肆虐全球.蓬莱国也发现了零星感染者,为防止该病在蓬莱国大范围流行,该国政府决定不惜一切代价控制传染病的蔓延.不幸的是,由于人们尚未完全认识这种传染 ...
- 洛谷 P1041 传染病控制
P1041 传染病控制 题目背景 近来,一种新的传染病肆虐全球.蓬莱国也发现了零星感染者,为防止该病在蓬莱国大范围流行,该国政府决定不惜一切代价控制传染病的蔓延.不幸的是,由于人们尚未完全认识这种传染 ...
- [luogu]P1041 传染病控制[dfs]
[luogu]P1041 传染病控制 ——!x^n+y^n=z^n 题目背景 近来,一种新的传染病肆虐全球.蓬莱国也发现了零星感染者,为防止该病在蓬莱国大范围流行,该国政府决定不惜一切代价控制传染病的 ...
- P1041 传染病控制——暴力遍历所有相同深度的节点
P1041 传染病控制 说实话这种暴力我还是头一次见,每次病毒都会往下传染一层: 数据范围小,我们可以直接枚举当前层保护谁就好了: 用vector 记录相同层数的节点:维护已经断了的点: 如果超出最底 ...
- P1041 传染病控制【暴搜】
P1041 传染病控制 提交 10.78k 通过 3.74k 时间限制 1.00s 内存限制 125.00MB 题目提供者CCF_NOI 难度提高+/省选- 历史分数100 提交记录 查看题解 标签 ...
- Luogu1041 NOIP2003传染病控制(搜索)
暴搜加个最优性剪枝即可.一直觉得正式比赛出这种不能一眼看出来暴搜就行了的搜索题的出题人都是毒瘤. #include<iostream> #include<cstdio> #in ...
- [NOIP2003] 提高组 洛谷P1041 传染病控制
题目背景 近来,一种新的传染病肆虐全球.蓬莱国也发现了零星感染者,为防止该病在蓬莱国大范围流行,该国政府决定不惜一切代价控制传染病的蔓延.不幸的是,由于人们尚未完全认识这种传染病,难以准确判别病毒携带 ...
随机推荐
- gorilla/mux类库解析
golang自带的http.SeverMux路由实现简单,本质是一个map[string]Handler,是请求路径与该路径对应的处理函数的映射关系.实现简单功能也比较单一: 不支持正则路由, 这个是 ...
- F#周报2019年第41期
新闻 .NET架构指南 美妙的WebSharper:学术刊物 .NET Core 3.0中Blazor Server的方案与性能 Mono 6.4.0发布说明 CapitolFSharp召集发言人 视 ...
- Hyper-V虚拟机win7网络红叉,无法上网解决方法
之前一直都是玩Vmware虚拟机,后来win8之后的系统有Hyper-V虚拟机就开始接触了. Windows 中内置的Hyper-V管理器可以说是给很多人带来了惊喜!至少运行的流畅程度要比Vmware ...
- Hexo 博客快速整合gitalk组件,给静态博客添加动态评论功能!
什么是 hexo-plugin-gitalk
- GUI tkinter (bind)事件篇
"""事件:1.我们的很多操作,比如我们点击了一下鼠标,这就是一 个事件,而操作系统会根据我们的相应的事件产生相应的消息, 操作系统把消息传递给我们的应用程序,然后我们的 ...
- Jenkins源代码管理(SVN)
Subversion 安装插件 1.首先将本地的自动化用例打包上传svn 2.配置jenkins源代码管理(每次执行jenkins时,会自动check-out配置地址中的代码到Jenkins的工作空间 ...
- java架构之路-(Redis专题)Redis的主从、哨兵和集群
我们使用的redis,单机的绝对做不到高可用的,万一单机的redis宕机了,就没有备用的了,我们可以采用集群的方式来保证我们的高可用操作. 主从架构 大致就是这样的,一个主节点,两个从节点(一般两个就 ...
- opencv::AKAZE检测与匹配
AKAZE局部匹配 AKAZE局部匹配介绍 AOS 构造尺度空间 Hessian矩阵特征点检测 方向指定基于一阶微分图像 描述子生成 与SIFT.SUFR比较 更加稳定 非线性尺度空间 AKAZE速度 ...
- unittest断言
assertEquals(expected,actual,msg=msg) # 判断 expected,actual是否一致,msg类似备注,可以为空
- spring boot 中通过CORS实现跨域
一.跨域问题出现的原因 出现跨域问题的原因是同源策略. 同源策略 主要是三同:同协议.同域名.同端口, 同源策略目的 保证用户信息安全,防止恶意网站窃取数据.同源策略是必须的,否则cookie可以共享 ...