Codeforces 486D Valid Sets (树型DP)
题目链接 Valid Sets
题目要求我们在一棵树上计符合条件的连通块的个数。
满足该连通块内,点的权值极差小于等于d
树的点数满足 n <= 2000
首先我们先不管这个限制条件,也就是先考虑d为正无穷大的时候的情况。
我们要求出树上所有连通块的个数。
这个时候我们令f[i]为以i为根的子树中的连通块的数目。
此时状态转移方程为 f[x] = f[x] * (f[u] + 1)
其中f[x]初始值为1,u为x的儿子
最后f[1]的值(我们假设1为根结点)即为答案
时间复杂度为O(n)
注意到n只有2000,说明这题的时间复杂度不止O(n)
那么我们对于每一个点,以他的权值作为连通块的权值最小值。
于是就可以以他为根做一次DFS。
若DFS的过程中碰到权值比他小的点,或者权值减他的权值大于d的点,我们就不往这个点DFS下去。
但是有一种特殊情况
这样做可能导致重复计算
因为这样的方法会导致两个权值相同切且相连的点组成的连通块被计算多次。
于是我们对那些权值相同切且相连的点的边,定一个方向。
规定编号小的点能DFS到编号大,和他相连且权值和他相等的点
但是反过来就不行了。
这样规定了一个方向之后我们就消除了重复计算的问题。
时间复杂度 $O(n^{2})$
- #include <bits/stdc++.h>
- using namespace std;
- #define rep(i, a, b) for (int i(a); i <= (b); ++i)
- #define dec(i, a, b) for (int i(a); i >= (b); --i)
- typedef long long LL;
- const int N = 2010;
- const LL mod = 1e9 + 7;
- vector <int> v[N];
- int n, d, et, cnt;
- int a[N];
- LL f[N];
- LL ans = 0;
- void dfs(int x, int fa){
- LL now = 0;
- f[x] = 1;
- for (auto u : v[x]){
- if (u == fa) continue;
- if (a[u] > cnt + d || a[u] < cnt) continue;
- if (a[u] == cnt && u < et) continue;
- dfs(u, x);
- (f[x] *= f[u] + 1) %= mod;
- }
- }
- int main(){
- scanf("%d%d", &d, &n);
- rep(i, 1, n) scanf("%d", a + i);
- rep(i, 1, n - 1){
- int x, y;
- scanf("%d%d", &x, &y);
- v[x].push_back(y);
- v[y].push_back(x);
- }
- rep(i, 1, n){
- cnt = a[i]; et = i;
- memset(f, 0, sizeof f);
- dfs(i, 0);
- (ans += f[i]) %= mod;
- }
- printf("%lld\n", ans);
- return 0;
- }
Codeforces 486D Valid Sets (树型DP)的更多相关文章
- Codeforces 486D Valid Sets:Tree dp【n遍O(n)的dp】
题目链接:http://codeforces.com/problemset/problem/486/D 题意: 给你一棵树,n个节点,每个节点的点权为a[i]. 问你有多少个连通子图,使得子图中的ma ...
- Codeforces 486D. Valid Sets
D. Valid Sets time limit per test 1 second memory limit per test 256 megabytes input standard input ...
- Codeforces 23E Tree(树型DP)
题目链接 Tree $dp[x][i]$表示以x为根的子树中x所属的连通快大小为i的时候 答案最大值 用$dp[x][j]$ * $dp[y][k]$ 来更新$dp[x][j + k]$. (听高手说 ...
- Codeforces 149D Coloring Brackets(树型DP)
题目链接 Coloring Brackets 考虑树型DP.(我参考了Q巨的代码还是略不理解……) 首先在序列的最外面加一对括号.预处理出DFS树. 每个点有9中状态.假设0位不涂色,1为涂红色,2为 ...
- Codeforces 581F Zublicanes and Mumocrates(树型DP)
题目链接 Round 322 Problem F 题意 给定一棵树,保证叶子结点个数为$2$(也就是度数为$1$的结点),现在要把所有的点染色(黑或白) 要求一半叶子结点的颜色为白,一半叶子结点的 ...
- 【题解】codeforces 219D Choosing Capital for Treeland 树型dp
题目描述 Treeland国有n个城市,这n个城市连成了一颗树,有n-1条道路连接了所有城市.每条道路只能单向通行.现在政府需要决定选择哪个城市为首都.假如城市i成为了首都,那么为了使首都能到达任意一 ...
- POJ3659 Cell Phone Network(树上最小支配集:树型DP)
题目求一棵树的最小支配数. 支配集,即把图的点分成两个集合,所有非支配集内的点都和支配集内的某一点相邻. 听说即使是二分图,最小支配集的求解也是还没多项式算法的.而树上求最小支配集树型DP就OK了. ...
- POJ 3342 - Party at Hali-Bula 树型DP+最优解唯一性判断
好久没写树型dp了...以前都是先找到叶子节点.用队列维护来做的...这次学着vector动态数组+DFS回朔的方法..感觉思路更加的清晰... 关于题目的第一问...能邀请到的最多人数..so ea ...
- 【XSY1905】【XSY2761】新访问计划 二分 树型DP
题目描述 给你一棵树,你要从\(1\)号点出发,经过这棵树的每条边至少一次,最后回到\(1\)号点,经过一条边要花费\(w_i\)的时间. 你还可以乘车,从一个点取另一个点,需要花费\(c\)的时间. ...
随机推荐
- A*和IDA*介绍
\(A*\)算法是一种很神奇的搜索方法,它属于启发式搜索中的一种.A*最主要的功能当然就是用来剪枝,提高搜索的效率.A*主要的实现方法是通过一个估价函数,每次对下一步进行一个估价,根据估价出的值来决定 ...
- iOS UI 设计
优设 http://www.uisdc.com Sketch http://www.sketchcn.com
- C# 读App.config配置文件[2]: .Net Core框架
C# 读App.config配置文件[1]:.Net Framework框架 C# 读App.config配置文件[2]: .Net Core框架 网上都是.net framework读取配置文件的方 ...
- Clover启动mbr的win7/win8
对以传统bios安装在mbr分区的win7/WIN8也可以使用EFI引导直接进入win.首先进win提取EFI引导文件,以管理员员身份运行cmd,输入如下命令 bcdboot c:\windows / ...
- HDU-1241-油藏
这题一道深搜的简单题目,其实题目的思路就只是向八个方向搜索,然后把整个油田遍历一遍即可. #include <cstdio> #include <cstring> int ma ...
- RuntimeError: Failed to init API, possibly an invalid tessdata path: E:\python36\报错
OCR:光学识别符,tesserocr是python中一个OCR识别库,是对tesseract做的一个python的 API封装,所以它的核心是tesseract 在这里我安装的版本是:tessera ...
- 三、Pandas速查手册中文版
本文翻译自文章:Pandas Cheat Sheet - Python for Data Science,同时添加了部分注解. 对于数据科学家,无论是数据分析还是数据挖掘来说,Pandas是一个非常重 ...
- 【14】PNG,GIF,JPG的区别及如何选
[14]PNG,GIF,JPG的区别及如何选 GIF: 8位像素,256色 无损压缩 支持简单动画 支持boolean透明 适合简单动画 JPEG: 颜色限于256 有损压缩 可控制压缩质量 不支持透 ...
- [转]python 多线程threading简单分析
多线程和多进程是什么自行google补脑 对于python 多线程的理解,我花了很长时间,搜索的大部份文章都不够通俗易懂.所以,这里力图用简单的例子,让你对多线程有个初步的认识. 单线程 在好些年前的 ...
- CentOS 下通过命令登录Mysql
CentOS 下通过命令登录Mysql: mysql -uroot -p 按回车键后输入密码