[COCI2015-2016#1] UZASTOPNI 题解
前言
题目链接:洛谷。
题意简述
一棵有根树,节点数 \(n \leq 10^5\),每个点有权值 \(v_i \leq 2000\),现在选出一些点,满足:
- 一个点的父亲点若未被选择则其不能被选择。
- 所选点的集合内不能有相同的权值。
- 对于每一个选择的点,其子树中所有被选择点的权值必须可以构成公差为 \(1\) 的等差数列。
求满足上述条件构成的等差数列的方案数。
题目分析
凭直觉是树形 DP。考虑记 \(f[i][l][r]\) 表示在 \(i\) 子树,\(i\) 必选,能否构成 \(l \sim r\) 的等差数列。发现有冗余状态,即由于 \(i\) 必选,一定有 \(l \leq v_i \leq r\),其他状态是不合法的。
进一步发现,对于 \(yzh \in \operatorname{son}(i)\),她对 \(i\) 的贡献要么全在 \(v_i\) 左边,要么全在右边,否则会因为已经选中了一个权值为 \(v_i\) 的点不符合要求。进一步发现,如果 \(v_{yzh} < v_i\),她只能贡献左边,反之如果 \(v_{yzh} > v_i\) 贡献右边。即 \(yzh\) 不会对 \(i\) 两边都产生贡献。
所以将状态砍半,记 \(L[i][l]\) 和 \(R[i][r]\) 分别表示在 \(i\) 子树,\(i\) 必选,能否构成 \(l \sim v_i\) 的等差数列或 \(v_i \sim r\) 的等差数列。边界为 \(L[i][v_i] = R[i][v_i] = \text{true}\)。由于讨论过一个孩子不可能对左右多产生贡献,所以 \(L\) 和 \(R\) 是独立的,答案就是乘法原理,\(L[1]\) 中为 \(\text{true}\) 的个数与 \(R[1]\) 中为 \(\text{true}\) 的个数之积即为答案。
说了这么多,考虑转移。考虑 \(v_{yzh} < v_i\) 的情况,反之同理。如果存在一个值 \(k\),满足在 \(yzh\) 中能够得到 \(v_{yzh} \sim k\),并且在之前的 \(i\) 中能够得到 \(k + 1 \sim v_i\),那么,就可以把 \(L[yzh]\) 或到 \(L[i]\) 上。需要注意的是,转移的顺序也要考虑。即我们要先考虑 \(v_{yzh}\) 靠近 \(v_i\) 的孩子,这样后续可能会使用到这次更新过来的信息。
时间复杂度:\(\Theta(n(V + \log n))\),如果使用 bitset 优化,则是:\(\Theta(\cfrac{nV}{w} + n \log n)\)。
代码
// #pragma GCC optimize(3)
// #pragma GCC optimize("Ofast", "inline", "-ffast-math")
// #pragma GCC target("avx", "sse2", "sse3", "sse4", "mmx")
#include <iostream>
#include <cstdio>
#define debug(a) cerr << "Line: " << __LINE__ << " " << #a << endl
#define print(a) cerr << #a << "=" << (a) << endl
#define file(a) freopen(#a".in", "r", stdin), freopen(#a".out", "w", stdout)
#define main Main(); signed main() { return ios::sync_with_stdio(0), cin.tie(0), Main(); } signed Main
using namespace std;
#include <algorithm>
#include <vector>
#include <bitset>
int n, val[100010];
vector<int> edge[100010];
bitset<2010> L[100010], R[100010];
void dfs(int now) {
L[now][val[now]] = R[now][val[now]] = 1;
for (const auto& to: edge[now]) dfs(to);
for (int i = 0; i < (int)edge[now].size(); ++i) {
int to = edge[now][i];
if (val[to] > val[now]) {
if (((R[now] << 1) & L[to]).any()) {
R[now] |= R[to];
}
}
}
for (int i = (int)edge[now].size() - 1; i >= 0; --i) {
int to = edge[now][i];
if (val[to] < val[now]) {
if (((L[now] >> 1) & R[to]).any()) {
L[now] |= L[to];
}
}
}
}
signed main() {
scanf("%d", &n);
for (int i = 1; i <= n; ++i) scanf("%d", &val[i]);
for (int i = 1, u, v; i <= n - 1; ++i) {
scanf("%d%d", &u, &v);
edge[u].push_back(v);
}
for (int i = 1; i <= n; ++i)
sort(edge[i].begin(), edge[i].end(), [] (const int &a, const int &b) {
return val[a] < val[b];
});
dfs(1);
printf("%llu", L[1].count() * R[1].count());
return 0;
}
[COCI2015-2016#1] UZASTOPNI 题解的更多相关文章
- P6666 [清华集训2016] 数据交互 题解
## P6666 [清华集训2016] 数据交互 题解 ### 简要题意: n个点的树,m次操作,分别为添加一条路径$(u_i,v_i,w_i)$,和撤消一条路径,每一次操作后求出一条路径使得与这条路 ...
- [COCI]coci2015/2016 nekameleoni
题意: 初始数列,每个数都在1~k以内 支持两种操作:1.修改一个数,修改后的数在1~k内 2.查询一个最短包含1~k的序列的长度 查询100000 ...
- COCI 2015、2016 1st round 题解(官方)
官方题解: 官方代码: Code-KARTE: #include <cstdio> #include <iostream> #include <cstring> u ...
- CODE FESTIVAL 2016 Grand Final 题解
传送门 越学觉得自己越蠢--这场除了\(A\)之外一道都不会-- \(A\) 贪心从左往右扫,能匹配就匹配就好了 //quming #include<bits/stdc++.h> #def ...
- noip 2016 提高组题解
前几天写的那个纯属搞笑.(额,好吧,其实这个也不怎么正经) 就先说说day2吧: T1:这个东西应该叫做数论吧. 然而我一看到就照着样例在纸上推了大半天(然而还是没有看出来这东西是个杨辉三角) 然后就 ...
- NCPC 2016:简单题解
A .Artwork pro:给定N*M的白色格子,然后Q次黑棒,输出每次加黑棒后白色连通块的数量.(N,M<1e3, Q<1e4) sol:倒着离线做,并查集即可. (在线做法:http ...
- 2016 ACM-ICPC EC-Final题解
题目链接 A. Number Theory Problem 题意:给你一个数N,求形如2k-1且小于2N的数中有多少能被7整除. 解法:观察二进制位找规律,答案是N/3. #include<bi ...
- NOIP 2016 组合数问题 题解
一道sb题目,注意范围,可打表解决,打出杨辉三角,在用前缀和求解即可 代码(一维前缀和) #include<bits/stdc++.h> using namespace std; int ...
- CODE FESTIVAL 2016 qual C题解
传送门 \(A\) 什么玩意儿-- const int N=105; char s[N];int n,f1,f2; int main(){ scanf("%s",s+1),n=st ...
- CODE FESTIVAL 2016 qual B题解
传送门 \(A\) 什么玩意儿-- const char t[]={"0CODEFESTIVAL2016"}; char s[25];int res; int main(){ sc ...
随机推荐
- SpringBoot指标监控功能
SpringBoot指标监控功能 随时查看SpringBoot运行状态,将状态以josn格式返回 添加Actuator功能 Spring Boot Actuator可以帮助程序员监控和管理Spring ...
- 用 Python 绘制现金流量图
目录 用 Python 绘制现金流量图 Python 实现 实现原理 具体代码 使用示例 1:根据现金流量表绘制现金流量图 使用示例 2:绘制等额.等差.等比序列现金流量图 等额序列现金流量图 等差序 ...
- 聊一聊 C# 弱引用 底层是怎么玩的
一:背景 1. 讲故事 最近在分析dump时,发现有程序的卡死和WeakReference有关,在以前只知道怎么用,但不清楚底层逻辑走向是什么样的,借着这个dump的契机来简单研究下. 二:弱引用的玩 ...
- 开源日志组件Sejil--附带日志管理界面
1.开源日志组件源码: https://github.com/alaatm/Sejil 2.下载下来发现里面对于不同的.net core 版本的配置提供了对应的示例 .Net Core 3.1 Pr ...
- Linux历史管理命令
history管理历史命令 [1].history命令 history命令用于显示历史记录和执行过的命令,登录系统时,会读取~./bash_history历史文件中记录的命令,当我们退出shell时, ...
- 使用gzexe加密shell脚本
使用 gzexe 加密 shell 脚本是一个相对简单的过程.以下是具体的步骤: 编写你的 shell 脚本:首先,你需要有一个 shell 脚本文件,比如 myscript.sh. 确保脚本可执行: ...
- WebGL管网展示(及TubeGeometry优化)
前言 管路展示在三维场景中很常见.比如地下管网,建筑里面的水果,暖通管道等等的展示. 建立管路的方式主要两种: 通过3DMax C4D Blender等建模工具进行建模. 通过路径数据,程序生成三维管 ...
- [oeasy]python0094_视频游戏_双人网球_pong_atari_mos_6502_雅达利_米洛华
编码进化 回忆上次内容 上次 我们回顾了 微软之前的 比尔盖茨和保罗艾伦 mits 迎来的 是帮手 还是隐患? intel-8080 遇到了 mos-6502 底层硬件 驱动 游戏行业进化 不光是扑克 ...
- oeasy 教您玩转linux 010303文件管理器 nautilus
我们来回顾一下 上一部分我们都讲了什么? 讲了火狐 火狐的位置 用命令行打开多个网址 火狐的升级 火狐桌面建立快捷方式 我们可以知道桌面快捷方式文件的名称么? 从文件管理器到命令行 按住文件 拖动到t ...
- oeasy教您玩转vim - 37 - # 删除字符
通过十进制的 ascii 值输入字符 在输入模式下 输入 ctrl + v 然后再输入 065 通过十六进制的 unicode 在输入模式下 输入 ctrl + v 然后再输入 u2642 就可以 ...