\(\mathcal{Description}\)

  Link.

  求包含 \(n\) 个结点 \(m\) 条边的仙人掌的最大独立集。

  \(n\le5\times10^4\),\(m\le6\times10^4\)。

\(\mathcal{Solution}\)

  建出圆方树,考虑树上 DP。

  设状态 \(f(i,0/1)\) 表示该点不选择/不限制选择与父亲相邻的圆点(对于圆点,即它本身)时,子树内的最大独立集。转移分圆点和方点讨论:

  • 圆点:很显然,\(f(u,0)=\sum_{v\in son_u}f(v,1),f(u,1)=\max\{f(u,0),\sum_{v\in son_u}f(v,0)+1\}\)。

  • 方点:考虑把方点所代表的环展开成链。在链上做一个子 DP:令 \(g(i,0/1)\) 表示选择/不限制选择第 \(i\) 给点,前 \(i\) 个点的子树内的最大独立集大小。记 \(son_i\) 表示 \(u\) 的第 \(i\) 个儿子,和 \(f(u,0)\) 的转移类似,有:

    • \(g(i,0)=g(i-1,1)+f(son_i,0)\)。

    • \(g(i,1)=\max\{g(i,0),g(i-1,0)+f(son_i,1)\}\)。

  在求 \(f(u,0)\) 时,令 \(g(1,0)=g(1,1)=f(son_1,0)\);在求 \(f(u,1)\) 时,令 \(g(1,0)=f(son_1,0),g(1,1)=f(son_1,1)\),分别转移两次即可。

  最后,\(f(root,1)\) 就是答案。

  复杂度 \(\mathcal O(n)\)。

\(\mathcal{Code}\)

  读边只读 \(n\) 条调了半天 qwq。

#include <queue>
#include <cstdio>
#include <algorithm> #define adj( g, u, v ) \
for ( unsigned eid = 0, v; eid ^ g.vec[u].size () && ( v = g.vec[u][eid], 1 ); ++ eid ) const int MAXN = 2e5, MAXM = 2.4e5;
int n, m, q, snode;
int dfc, top, dfn[MAXN + 5], low[MAXN + 5], stk[MAXN + 5];
int f[MAXN + 5][2]; struct Graph {
// int ecnt, head[MAXN + 5], to[MAXM + 5], nxt[MAXM + 5];
std::vector<int> vec[MAXN + 5];
inline void link ( const int s, const int t ) {
// to[++ ecnt] = t, nxt[ecnt] = head[s];
// head[s] = ecnt;
vec[s].push_back ( t );
}
inline void add ( const int u, const int v ) {
link ( u, v ), link ( v, u );
}
} src, tre; inline int rint () {
int x = 0; char s = getchar ();
for ( ; s < '0' || '9' < s; s = getchar () );
for ( ; '0' <= s && s <= '9'; s = getchar () ) x = x * 10 + ( s ^ '0' );
return x;
} inline bool chkmin ( int& a, const int b ) { return b < a ? a = b, true : false; } inline bool chkmax ( int& a, const int b ) { return a < b ? a = b, true : false; } inline void Tarjan ( const int u, const int fa ) {
dfn[u] = low[u] = ++ dfc, stk[++ top] = u;
adj ( src, u, v ) if ( v ^ fa ) {
if ( ! dfn[v] ) {
Tarjan ( v, u ), chkmin ( low[u], low[v] );
if ( low[v] >= dfn[u] ) {
++ snode;
do tre.add ( snode, stk[top] ); while ( stk[top --] ^ v );
tre.link ( u, snode );
}
} else chkmin ( low[u], dfn[v] );
}
} inline void solve ( const int u, const int fa ) {
if ( u <= n ) {
f[u][0] = 0, f[u][1] = 1;
adj ( tre, u, v ) if ( v ^ fa ) {
solve ( v, u );
f[u][0] += f[v][1], f[u][1] += f[v][0];
}
chkmax ( f[u][1], f[u][0] );
} else {
static int tmp[MAXN + 5][2];
for ( int v: tre.vec[u] ) if ( v ^ fa ) solve ( v, u );
tmp[0][0] = tmp[0][1] = f[tre.vec[u][0]][0];
for ( int i = 1; i ^ tre.vec[u].size (); ++ i ) {
tmp[i][0] = tmp[i - 1][1] + f[tre.vec[u][i]][0];
tmp[i][1] = tmp[i - 1][0] + f[tre.vec[u][i]][1];
chkmax ( tmp[i][1], tmp[i][0] );
}
f[u][0] = tmp[tre.vec[u].size () - 1][0];
tmp[0][0] = f[tre.vec[u][0]][0], tmp[0][1] = f[tre.vec[u][0]][1];
for ( int i = 1; i ^ tre.vec[u].size (); ++ i ) {
tmp[i][0] = tmp[i - 1][1] + f[tre.vec[u][i]][0];
tmp[i][1] = tmp[i - 1][0] + f[tre.vec[u][i]][1];
chkmax ( tmp[i][1], tmp[i][0] );
}
f[u][1] = tmp[tre.vec[u].size () - 1][1];
}
} int main () {
n = snode = rint (), m = rint ();
for ( int i = 1, u, v; i <= m; ++ i ) {
u = rint (), v = rint ();
src.add ( u, v );
}
Tarjan ( 1, 0 );
solve ( 1, 0 );
printf ( "%d\n", f[1][1] );
return 0;
}

Solution -「BZOJ 4316」小C的独立集的更多相关文章

  1. Solution -「BZOJ 3812」主旋律

    \(\mathcal{Description}\)   Link.   给定含 \(n\) 个点 \(m\) 条边的简单有向图 \(G=(V,E)\),求 \(H=(V,E'\subseteq E)\ ...

  2. 「BZOJ 3280」小R的烦恼

    题目链接 戳我 \(Solution\) 这道题很像餐巾计划啊. 首先将每天拆成\(x\)和\(x'\),\(S->x\)流量为\(a_i\),费用为\(0\)表示一天下来有\(a_i\)个濒死 ...

  3. Solution -「BZOJ #3786」星系探索

    \(\mathcal{Description}\)   Link.   给定一棵含 \(n\) 个点的有根树,点有点权,支持 \(q\) 次操作: 询问 \(u\) 到根的点权和: 修改 \(u\) ...

  4. Solution -「BZOJ 3331」压力

    \(\mathcal{Description}\)   Link.   给定一个 \(n\) 个点 \(m\) 条边的连通无向图,并给出 \(q\) 个点对 \((u,v)\),令 \(u\) 到 \ ...

  5. 「BZOJ 4289」 PA2012 Tax

    「BZOJ 4289」 PA2012 Tax 题目描述 给出一个 \(N\) 个点 \(M\) 条边的无向图,经过一个点的代价是进入和离开这个点的两条边的边权的较大值,求从起点 \(1\) 到点 \( ...

  6. 「BZOJ 4228」Tibbar的后花园

    「BZOJ 4228」Tibbar的后花园 Please contact lydsy2012@163.com! 警告 解题思路 可以证明最终的图中所有点的度数都 \(< 3\) ,且不存在环长是 ...

  7. 「BZOJ 3645」小朋友与二叉树

    「BZOJ 3645」小朋友与二叉树 解题思路 令 \(G(x)\) 为关于可选大小集合的生成函数,即 \[ G(x)=\sum[i\in c ] x^i \] 令 \(F(x)\) 第 \(n\) ...

  8. 「BZOJ 4502」串

    「BZOJ 4502」串 题目描述 兔子们在玩字符串的游戏.首先,它们拿出了一个字符串集合 \(S\),然后它们定义一个字符串为"好"的,当且仅当它可以被分成非空的两段,其中每一段 ...

  9. 「BZOJ 2534」 L - gap字符串

    「BZOJ 2534」 L - gap字符串 题目描述 有一种形如 \(uv u\) 形式的字符串,其中 \(u\) 是非空字符串,且 \(v\) 的长度正好为 \(L\), 那么称这个字符串为 \( ...

随机推荐

  1. git clone 失败 ,提示 fatal: unable to access 'https://github.com/xxx.git/': OpenSSL SSL_read: Connection was reset, errno 10054

    怎么解决? 把原来的指令 $ git clone https://github.com/cen-xi/express.git 改成 $ git clone git://github.com/cen-x ...

  2. ssh到localhost或127.0.0.1拒绝连接

    通过ssh连接到本机报错 ssh: connect to host localhost port 22: Connection refused, 你能用ssh登录其它主机并不代表着本地有ssh服务,要 ...

  3. vue将HTML字符串解析为HTML标签

    如果返回的数据是html标签字符串的话,在vue里要通过v-html来渲染 <div v-html="rawHtml"></div> //v-html=&q ...

  4. [转]axios请求超时,设置重新请求的完美解决方法

    自从使用Vue2之后,就使用官方推荐的axios的插件来调用API,在使用过程中,如果服务器或者网络不稳定掉包了, 你们该如何处理呢? 下面我给你们分享一下我的经历. 具体原因 最近公司在做一个项目, ...

  5. Kube-OVN 0.6.0 发布,支持 IPv6、流量镜像及更多功能

    Kube-OVN 是一个基于 OVN 的 Kubernetes 开源网络系统. 本次更新主要包含了以下内容: 1. 支持流量镜像 在安装 Kube-OVN 时可以开启 mirror 选项,会自动在每个 ...

  6. 带你玩转Flink流批一体分布式实时处理引擎

    摘要:Apache Flink是为分布式.高性能的流处理应用程序打造的开源流处理框架. 本文分享自华为云社区<[云驻共创]手把手教你玩转Flink流批一体分布式实时处理引擎>,作者: 萌兔 ...

  7. Java基础(十)——枚举与注解

    一.枚举 1.介绍 枚举类:类的对象只有有限个,确定的.当需要定义一组常量时,强烈建议使用枚举类.如果枚举类中只有一个对象,则可以作为单例模式的实现. 使用 enum 定义的枚举类默认继承了 java ...

  8. WPF学习笔记(四):AvalonEdit 代码高亮编辑控件专题

    AvalonEdit 是一个基于 WPF 的文本编辑器组件.它是由 Daniel Grunwald 为 SharpDevelop 编写的.从 5.0 版开始,AvalonEdit 根据MIT许可证发布 ...

  9. AOP-底层原理

    AOP(底层原理) 1,AOP底层使用动态代理 (1)有两种情况动态代理 第一种 有接口情况,使用JDK动态代理 *创建接口实现类代理对象,增强类的方法 第二种 无接口情况,使用CGLIB动态代理 * ...

  10. 你可能不知道的Animation动画技巧与细节

    引言 在web应用中,前端同学在实现动画效果时往往常用的几种方案: css3 transition / animation - 实现过渡动画 setInterval / setTimeout - 通过 ...