[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 ...
随机推荐
- excel计算日期天数和表格冻结首行
excel计算日期天数和表格冻结首行 1.在单元格E35中输入公式DATEDIF(A35.B35."MD")MD表起始日期.结束日期天数差."Y" 时间段中的整 ...
- Rust性能分析之测试及火焰图,附(lru,lfu,arc)测试
性能测试,在编写代码后,单元测试及性能测试是重要的验收点,好的性能测试可以让我们提前发现程序中存在的问题. 测试用例 在Rust中,测试通常有两部分,一部分是文档测试,一部分是模块测试. 通常我们在函 ...
- python globals()[]将字符串转化类,并通过反射执行方法
背景: 通过关键字设计ui自动化框架,将测试用例及其步骤存放到excel文件:其中步骤中包含了封装好的关键字方法,如打开浏览器.输入页面操作等,关键字保存的内容:具体类实例.方法 通过excel获取到 ...
- bugly进阶01-集成bugly时的相关参数
bugly进阶01-集成bugly时的相关参数 个人github CSDN博客 前言 bugly的集成十分的简单,在代码中只需要简单的一个语句就可以轻松集成: - (BOOL)application: ...
- 3D捕鱼大富翁源码分析
今天接受了一个捕鱼的源码,技术栈采用: 客户端:Unity 服务端:Java 数据库:mysql 缓存:redis 先来几张成品图 编辑编辑 编辑编辑 编辑 在代码中看到有腾讯推广渠道, ...
- .NET 认识日志系统-2
.NET 日志系统2 上一篇文章是将日志打印到控制台,这篇文章将日志写入到文本文件中. 文本日志一般按照日期区分 如何避免文本日志把磁盘撑爆? 限制日志总个数或者总大小 如何避免一个日志文件太大? 限 ...
- 个人网站接入Google Ads的一点心得
前言 前段时间花了一些精力尝试和摸索主题接入 Google Ads 的问题,算是阶段性成功了吧,这次简单分享一下,如果有缘看到这篇文章,应该会有些启发. 1. 展示效果 上篇文章说到,前两天我在我的两 ...
- Docker Harbor的安装配置
1.先安装docker-compose curl -L http://github.com/docker/compose/releases/download/1.21.2/docker-compose ...
- 超大容量 | 瑞芯微RK3588J工业核心板新增16GB DDR + 128GB eMMC配置!
作为瑞芯微的金牌合作伙伴,创龙科技在2023年9月即推出搭载瑞芯微旗舰级处理器RK3588J的全国产工业核心板--SOM-TL3588. SOM-TL3588工业核心板是基于瑞芯微RK3588J/RK ...
- QT 使用相对路径读取.txt文件
QT可以使用QFile来读取.txt文件,具体代码实现如下: 1 #include <QCoreApplication> 2 #include <QString> 3 #inc ...