【BZOJ1487】[HNOI2009]无归岛(动态规划)

题面

BZOJ

洛谷

题解

哪来的这么多废话啊,直接说一个仙人掌得了。

然后就是要你求仙人掌最大独立集了。(随便蒯份原来的代码就过了)

不过我还是重新整理一遍思路吧。

一种是裸的\(dp\),只需要额外考虑上环的影响就好了。

这种方法我们从树上的做法推广过来。

先考虑树的最大独立集,设\(f[i][0/1]\)表示当前考虑\(i\)及其子树,这个点一定不选,以及随意的最大独立集。转移的时候枚举这个点选还是不选即可。

推广到仙人掌上,相比于树,还多出来一条返祖边。所以额外维护一维\(0/1\),表示这条边所在的环的最底下的那个端点是否被选。分情况讨论转移即可,详细的解法戳这里

另外一种方法基于圆方树的思想。我们假装构建出来了圆方树(事实上不会构建出来的,只是利用了这个形式)。对于环而言,我们可以单独把环扣下来,显然除了环之外挂的子树对于这个环上的点是否选择是无关的。状态和树上的\(dp\)一样的设计。转移的时候这样子:如果这条边是一条树边,那么我们直接转移就好了,就把他当成一棵树来做。否则是一条返祖边,那么我们单独考虑这个环的贡献。现在的形式就是环上挂着一串的子树,子树内的答案我们已知,那么从一个方向开始遍历整个环,记录一下这个点选还是没有选的最大贡献,显然最终这个环的贡献只需要合并到深度最浅的那个点上面就可以了。那么枚举这个深度最浅的点是选还是不选,直接贪心抉择一下即可。更加具体的戳这里

代码一:直接\(dp\)

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<set>
#include<map>
#include<vector>
#include<queue>
using namespace std;
#define ll long long
#define RG register
#define MAX 100100
inline int read()
{
RG int x=0,t=1;RG char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=-1,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return x*t;
}
struct Line{int v,next;}e[MAX*3];
int h[MAX],cnt=1,n,m;
inline void Add(int u,int v){e[cnt]=(Line){v,h[u]};h[u]=cnt++;}
int dep[MAX],fa[MAX],a[MAX];
int tp[MAX],un[MAX];
void dfs(int u,int ff)
{
fa[u]=ff;dep[u]=dep[ff]+1;
for(int i=h[u];i;i=e[i].next)
if(!dep[e[i].v])dfs(e[i].v,u);
}
void jump(int u,int v){int x=v;while(x!=u)tp[x]=u,un[x]=v,x=fa[x];}
int f0[MAX],f1[MAX],g0[MAX],g1[MAX];
void dp(int u)
{
f1[u]=a[u];
if(u!=un[u])g1[u]=a[u];
for(int i=h[u];i;i=e[i].next)
{
int v=e[i].v;if(dep[u]+1!=dep[v])continue;
dp(v);
if(un[u]!=un[v])g0[u]+=f1[v],g1[u]+=g0[v];
else g0[u]+=g1[v],g1[u]+=g0[v];
if(tp[v]!=u)f0[u]+=f1[v],f1[u]+=f0[v];
else f0[u]+=f1[v],f1[u]+=g0[v];
}
f1[u]=max(f1[u],f0[u]);
g1[u]=max(g1[u],g0[u]);
}
int main()
{
n=read();m=read();
for(int i=1;i<=m;++i)
{
int u=read(),v=read();
Add(u,v);Add(v,u);
}
for(int i=1;i<=n;++i)a[i]=read();
dfs(1,0);
for(int u=1;u<=n;++u)
for(int i=h[u];i;i=e[i].next)
if(dep[u]<dep[e[i].v]&&fa[e[i].v]!=u)
jump(u,e[i].v);
dp(1);
printf("%d\n",f1[1]);
return 0;
}

类似圆方树的思想的做法:

#include<iostream>
#include<cstdio>
using namespace std;
#define MAX 100100
inline int read()
{
int x=0;bool t=false;char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=true,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return t?-x:x;
}
struct Line{int v,next;}e[MAX<<2];
int h[MAX],cnt=1;
inline void Add(int u,int v){e[cnt]=(Line){v,h[u]};h[u]=cnt++;}
int n,m,a[MAX];
int dfn[MAX],low[MAX],tim,f[MAX][2],fa[MAX];
void dp(int u,int y)
{
int f0=0,f1=0,t0,t1;
for(int i=y;i!=u;i=fa[i])
{
t0=f0+f[i][0];t1=f1+f[i][1];
f0=max(t0,t1);f1=t0;
}
f[u][0]+=f0;
f0=0;f1=-1e9;
for(int i=y;i!=u;i=fa[i])
{
t0=f0+f[i][0];t1=f1+f[i][1];
f0=max(t0,t1);f1=t0;
}
f[u][1]+=f1;
}
void Tarjan(int u,int ff)
{
dfn[u]=low[u]=++tim;fa[u]=ff;f[u][1]=a[u];
for(int i=h[u];i;i=e[i].next)
{
int v=e[i].v;
if(!dfn[v])Tarjan(v,u),low[u]=min(low[u],low[v]);
else if(v!=ff)low[u]=min(low[u],dfn[v]);
if(low[v]>dfn[u])
f[u][0]+=max(f[v][0],f[v][1]),f[u][1]+=f[v][0];
}
for(int i=h[u];i;i=e[i].next)
if(fa[e[i].v]!=u&&dfn[u]<dfn[e[i].v])
dp(u,e[i].v);
}
int main()
{
n=read();m=read();
for(int i=1;i<=m;++i)
{
int u=read(),v=read();
Add(u,v);Add(v,u);
}
for(int i=1;i<=n;++i)a[i]=read();
Tarjan(1,0);
printf("%d\n",max(f[1][0],f[1][1]));
return 0;
}

【BZOJ1487】[HNOI2009]无归岛(动态规划)的更多相关文章

  1. bzoj1487 [HNOI2009]无归岛

    Description Neverland是个神奇的地方,它由一些岛屿环形排列组成,每个岛上都生活着之中与众不同的物种.但是这些物种都有一个共同的生活习性:对于同一个岛 上的任意两个生物,他们有且仅有 ...

  2. BZOJ1487 [HNOI2009]无归岛 【仙人掌dp】

    题目链接 BZOJ1487 题解 就是一个简单的仙人掌最大权独立集 还是不会圆方树 老老实实地树形Dp + 环处理 #include<iostream> #include<cstdi ...

  3. 2019.02.07 bzoj1487: [HNOI2009]无归岛(仙人掌+树形dp)

    传送门 人脑转化条件过后的题意简述:给你一个仙人掌求最大带权独立集. 思路:跟这题没啥变化好吗?再写一遍加深记忆吧. 就是把每个环提出来分别枚举环在图中的最高点选还是不选分别dpdpdp一下即可,时间 ...

  4. P4410 [HNOI2009]无归岛

    P4410 [HNOI2009]无归岛 显然这还是一个仙人掌图 对于同一个岛上的任意两个生物,他们有且仅有一个公共朋友 要求求最大独立集,和树形dp一样,遇到环时单独提出来处理一下就好了 #inclu ...

  5. 【刷题】BZOJ 1487 [HNOI2009]无归岛

    Description Neverland是个神奇的地方,它由一些岛屿环形排列组成,每个岛上都生活着之中与众不同的物种.但是这些物种都有一个共同的生活习性:对于同一个岛上的任意两个生物,他们有且仅有一 ...

  6. 【BZOJ1487】[HNOI2009]无归岛(仙人掌 DP)

    题目: BZOJ1487 分析: 题目中给定的图一定是一棵仙人掌(每条边最多属于一个环),证明如下: 先考虑单独一个岛的情况.第一,一个岛一定是一张「弦图」,即任意一个大小超过 3 的环都至少有 1 ...

  7. [HNOI2009]无归岛

    Description Neverland是个神奇的地方,它由一些岛屿环形排列组成,每个岛上都生活着之中与众不同的物种.但是这些物种都有一个共同的生活习性:对于同一个岛上的任意两个生物,他们有且仅有一 ...

  8. 【题解】HNOI2009无归岛

    这题真的是无语了,在哪个岛上根本就没有任何的用处……不过我是画了下图,感受到一定是仙人掌,并不会证.有谁会证的求解…… 如果当做仙人掌来做确实十分的简单.只要像没有上司的舞会一样树形dp就好了,遇到环 ...

  9. Luogu-4410 [HNOI2009]无归岛

    裸的仙人掌最大独立子集,结果一个zz的错误让我调了好久... \(-inf\)开始设为\(0x7fffffff\)结果\(A_i\)有负数一加就炸了 #include<cstdio> #i ...

随机推荐

  1. 【webstorm】免费使用

    http://idea.imsxm.com/       测试过ok 后期追加(20180316更新为) http://idea.codebeta.cn/ 后期追加(20180502更新为) http ...

  2. 【H5】移动端页面根font-size设置

    h5rem.js   (配置写法①) (function (doc, win) { var docEl = doc.documentElement, resizeEvt = 'orientationc ...

  3. Luogu P2577 [ZJOI2005]午餐

    一道贪心+类背包DP的好题 首先发现一个十分显然的性质,没有这个性质整道题目都难以下手: 无论两队的顺序如何,总是让吃饭慢的人先排队 这是一个很显然的贪心,因为如果让吃饭慢的排在后面要更多的时间至少没 ...

  4. python删除文件与目录的方法

    python内置方法删除目录(空目录与非空目录)及文件 1.os.remove(file_path):删除文件 #PPTV是文件夹,xen.txt是文件 >>> os.remove( ...

  5. linux之 sed 基础

    转载:https://www.cnblogs.com/chensiqiqi/p/6382080.html sed 介绍 Sed命令是操作,过滤和转换文本内容的强大工具.常用功能有增删改查(增加,删除, ...

  6. zabbix设置微信报警的配置过程

    zabbix设置微信报警的配置过程 转发:https://blog.csdn.net/qq_31613055/article/details/78831607 微信企业号的申请 注册的地址https: ...

  7. PBFT_拜占庭容错算法

    根据论文<Practical Byzantine Fault Tolerance and Proactive Recovery>整理 Practical byzantine fault t ...

  8. PHP Laravel 连接并访问数据库

    第一次连接数据库 数据库配置位于config/database.php数据库用户名及密码等敏感信息位于.env文件创建一个测试表laravel_course <?php namespace Ap ...

  9. (第十二周)新功能WBS

    项目名:食物链教学工具 组名:奋斗吧兄弟 组长:黄兴 组员:李俞寰.杜桥.栾骄阳.王东涵 新增加的功能:背景音乐 功能 子功能 二级子功能 预计花费时间(小时) 实际花费时间(小时) 背景音乐 界面组 ...

  10. Linux内核分析:完成一个简单的时间片轮转多道程序内核代码

    PS.贺邦   原创作品转载请注明出处  <Linux内核分析>MOOC课程    http://mooc.study.163.com/course/USTC-1000029000 1.m ...