一道基环树+树形\(DP\)

原题链接

显然输入的是内向基环树森林,且我们可以单独考虑每一棵基环树。

既然是基环树,自然先\(dfs\)找环,然后随便找环上的一点\(r\),将其与\(A[r]\)的边断开,建反边,这时就会形成一棵以\(r\)为根的树,且每个点的子节点都是能限制它的元素。

于是我们可以在这棵树上跑树形\(DP\)。

定义\(f[x][0]\)表示不投放元素\(x\)的时候,以\(x\)为根的子树中能投放元素的最大值;\(f[x][1]\)表示投放元素\(x\)的时候,以\(x\)为根的子树中能投放元素的最大值。

  1. 不投放\(x\)时,它的子节点可以投放,也可以不投放。

\(\qquad\qquad f[x][0]=\sum\limits_{A[y]=x}\max\{f[y][0],f[y][1]\}\)

  1. 投放\(x\)时,至少有一个子节点是不投放的,以限制\(x\)。

\(\qquad\qquad f[x][1]=\max\limits_{A[y]=x}\{f[y][0]+\sum\limits_{A[z]=y,z\ne y}\max\{f[z][0],f[z][1]\}\}\)

\(DP\)完成后,用\(\max\{f[r][0],f[r][1]\}\)来更新答案。

然后考虑断边的影响,即导致\(r\)无法限制\(A[r]\),所以我们强制令\(r\)限制\(A[r]\),然后再进行一遍树形\(DP\)。当\(DP\)中计算\(f[A[r]][1]\)时,其子节点可以投放,也可以不投放,因为\(A[r]\)已经被\(r\)限制,所以不需要再有子节点限制它,而对于其它点的转移方程依旧不变。最后再用\(f[r][0]\)去更新答案,因为我们强制让\(r\)去限制\(A[r]\),所以\(r\)不能被投放。

注意这题使用普通的递归会\(MLE\),所以要打手工栈。

#include<cstdio>
using namespace std;
const int N = 1e6 + 10;
int fi[N], di[N << 1], ne[N << 1], a[N], f[N][2], st_x[N], st_i[N], st_y[N], st_s[N], cb, l;
bool v[N];
inline int re()
{
int x = 0;
char c = getchar();
bool p = 0;
for (; c<'0' || c>'9'; c = getchar())
p |= c == '-';
for (; c >= '0'&&c <= '9'; c = getchar())
x = x * 10 + (c - '0');
return p ? -x : x;
}
inline void add(int x, int y)
{
di[++l] = y;
ne[l] = fi[x];
fi[x] = y;
}
inline int maxn(int x, int y)
{
return x > y ? x : y;
}
void dfs(int x)
{
int y;
v[x] = 1;
for (y = a[x]; y; y = a[y])
{
if (v[y])
{
cb = y;
return;
}
v[y] = 1;
}
}
void dp(int x)
{
int k = 1;
st_x[1] = x;
start:
st_s[k] = 0;
v[st_x[k]] = 1;
f[st_x[k]][0] = f[st_x[k]][1] = 0;
for (st_i[k] = fi[st_x[k]]; st_i[k]; st_i[k] = ne[st_i[k]])
{
st_y[k] = di[st_i[k]];
if (st_y[k] ^ cb)
{
st_x[k + 1] = st_y[k];
k++;
goto start;
end:
st_s[k] += maxn(f[st_y[k]][0], f[st_y[k]][1]);
}
}
f[st_x[k]][0] = st_s[k];
for (st_i[k] = fi[st_x[k]]; st_i[k]; st_i[k] = ne[st_i[k]])
{
st_y[k] = di[st_i[k]];
if (st_y[k] ^ cb)
f[st_x[k]][1] = maxn(f[st_x[k]][1], 1 + st_s[k] - maxn(f[st_y[k]][0], f[st_y[k]][1]) + f[st_y[k]][0]);
}
if (--k)
goto end;
}
void dp_2(int x)
{
int k = 1;
st_x[1] = x;
start:
st_s[k] = 0;
f[st_x[k]][0] = f[st_x[k]][1] = 0;
for (st_i[k] = fi[st_x[k]]; st_i[k]; st_i[k] = ne[st_i[k]])
{
st_y[k] = di[st_i[k]];
if (st_y[k] ^ cb)
{
st_x[k + 1] = st_y[k];
k++;
goto start;
end:
st_s[k] += maxn(f[st_y[k]][0], f[st_y[k]][1]);
}
}
f[st_x[k]][0] = st_s[k];
if (!(st_x[k] ^ a[cb]))
{
f[st_x[k]][1] = st_s[k] + 1;
if (--k)
goto end;
return;
}
for (st_i[k] = fi[st_x[k]]; st_i[k]; st_i[k] = ne[st_i[k]])
{
st_y[k] = di[st_i[k]];
if (st_y[k] ^ cb)
f[st_x[k]][1] = maxn(f[st_x[k]][1], 1 + st_s[k] - maxn(f[st_y[k]][0], f[st_y[k]][1]) + f[st_y[k]][0]);
}
if (--k)
goto end;
}
int main()
{
int i, n, ma, s = 0;
n = re();
for (i = 1; i <= n; i++)
{
a[i] = re();
add(a[i], i);
}
for (i = 1; i <= n; i++)
if (!v[i])
{
ma = 0;
dfs(i);
dp(cb);
ma = maxn(f[cb][0], f[cb][1]);
dp_2(cb);
ma = maxn(ma, f[cb][0]);
s += ma;
}
printf("%d", s);
return 0;
}

JoyOI1940 创世纪的更多相关文章

  1. Poetize4 创世纪

    3037: 创世纪 Time Limit: 5 Sec  Memory Limit: 128 MBSubmit: 123  Solved: 66[Submit][Status] Description ...

  2. 为创世纪图书馆(Library Genesis)作镜像

    简介 Library Genesis的Wikipedia条目中的介绍是: Library Genesis or LibGen is a search engine for articles and b ...

  3. 编程哲学之C#篇:01——创世纪

    我们能否像神一样地创建一个世界? 对于创建世界而言,程序员的创作能力最接近于神--相对于导演,作家,漫画家而言,他们创建的世界(作品)一旦完成,就再也不会变化,创建的角色再也不会成长.而程序员创建的世 ...

  4. 【BZOJ3037/2068】创世纪/[Poi2004]SZP 树形DP

    [BZOJ3037]创世纪 Description applepi手里有一本书<创世纪>,里面记录了这样一个故事……上帝手中有着N 种被称作“世界元素”的东西,现在他要把它们中的一部分投放 ...

  5. [bzoj3037/2068]创世纪[Poi2004]SZP_树形dp_并查集_基环树

    创世纪 SZP bzoj-3037/2068 Poi-2004 题目大意:给你n个物品,每个物品可以且仅可以控制一个物品.问:选取一些物品,使得对于任意的一个被选取的物品来讲,都存在一个没有被选取的物 ...

  6. CH6401 创世纪

    6401 创世纪 0x60「图论」例题 描述 上帝手中有 N(N≤10^6) 种世界元素,每种元素可以限制另外1种元素,把第 i 种世界元素能够限制的那种世界元素记为 A[i].现在,上帝要把它们中的 ...

  7. 图形学创世纪——写在SIGGRAPH 40年的边上

    40年的边上" title="图形学创世纪--写在SIGGRAPH 40年的边上"> 前言: SIGGRAPH是由ACM SIGGRAPH(美国计算机协会计算机图形 ...

  8. JZOJ 3929. 【NOIP2014模拟11.6】创世纪

    3929. [NOIP2014模拟11.6]创世纪 (Standard IO) Time Limits: 1000 ms Memory Limits: 65536 KB Description 上帝手 ...

  9. T1创世纪(原创)

    创世纪 这是我的第一道原创题 题解: 这道题的核心算法是:加维度的最短路+贪心 状态:\(dis[i][j][t][a]\)表示在 \(t\) 时,到达 \((i,j)\) ,当前共造\(a\)只&q ...

随机推荐

  1. Dapper使用总结

  2. jekins的一些配置

    GotPC_Excel_Branches丢弃旧的构建 svn http://devsvn.uuzuonline.com/GOT_PC_PRIVATE/config/trunk构建Execute she ...

  3. 微信小程序实现滚动分页加载更多

    参考网址:https://www.cnblogs.com/Smiled/p/8203306.html 1.wxml: <view class='myScroll' style='float:le ...

  4. scrapy 是指user_agent

    1.倒入random模块 2.在settings文件中添加useragent列表 MY_USER_AGENT = [ "Mozilla/4.0 (compatible; MSIE 6.0; ...

  5. harbor在centos7.4下面配置自签名证书(域名是端口映射)

    1.harbor安装,按常规安装. 注意事项,端口映射 端口要外网的与内网一至. 配置文件修改 vim docker-compose.yml proxy: image: vmware/nginx-ph ...

  6. poj1984(带权并查集)

    题目链接:http://poj.org/problem?id=1984 题意:给定n个farm,m条边连接farm,k组询问,询问根据前t3条边求t1到t2的曼哈顿距离,若不可求则输出-1. 思路:类 ...

  7. pta6-17(另类堆栈)

    题目链接:https://pintia.cn/problem-sets/1101307589335527424/problems/1101313244872126464 题意:一种新的堆栈,用Top表 ...

  8. C#创建cookie读写cookie

    一.创建cookie HttpCookie cookie = new HttpCookie("UserInfo");//创建多值cookie              cookie ...

  9. stm32中adc的常规通道和注入通道的区别

    STM32的每个ADC模块通过内部的模拟多路开关,可以切换到不同的输入通道并进行转换.STM32特别地加入了多种成组转换的模式,可以由程序设置好之后,对多个模拟通道自动地进行逐个地采样转换. 有2种划 ...

  10. stark组件之批量操作【模仿Django的admin】

    一.先看下django的admin是如何实现批量操作 首先在配置类中定义一个函数 然后我们为这个函数对象设置一个属性,这个属性主要用来显示在select标签中显示的文本内容 最后把函数对象放到一个ac ...