HDU - 4809 树形dp
找了半天bug 发现把q打成了p。。。
思路:用dp[ i ][ j ][ k ] 表示在 i 这个点 这个点的状态为 j (0:不选 1:属于奇联通块 2:属于偶联通块) 且 奇联通块 - 偶联通块 = k 的方案数, 这个统计的方案数不包括i这个点的联通块。
dp[u][0][x+y]=dp[u][0][x]*dp[v][0][y]+dp[u][0][x]*dp[v][1][y-1]+dp[u][0][x]*dp[v][2][y+1]
dp[u][1][x+y]=dp[u][1][x]*dp[v][0][y]+dp[u][1][x]*dp[v][2][y]+dp[u][2][x]*dp[v][1][y]
dp[u][2][x+y]=dp[u][2][x]*dp[v][0][y]+dp[u][1][x]*dp[v][1][y]+dp[u][2][x]*dp[v][2][y]
#include<bits/stdc++.h>
#define LL long long
#define fi first
#define se second
#define mk make_pair
#define pii pair<int,int>
#define piii pair<int, pair<int,int> > using namespace std; const int N = + ;
const int M = + ;
const int inf = 0x3f3f3f3f;
const LL INF = 0x3f3f3f3f3f3f3f3f;
const int mod = 1e9 + ;
const int zero = ; int n, dp[N][][], tmp[][], down[N], up[N];
vector<int> edge[N]; void inline add(int &x, int y) {
x += y; if(x >= mod) x -= mod;
} void dfs(int u, int p) {
dp[u][][zero] = ;
dp[u][][zero] = ;
up[u] = down[u] = ;
for(int v : edge[u]) {
if(v == p) continue;
dfs(v, u);
memset(tmp, , sizeof(tmp));
for(int p = down[u]; p <= up[u]; p++) {
for(int q = down[v] - ; q <= up[v] + ; q++) {
if((p + q > n) || (p + q < (-n) / - )) continue;
add(tmp[][p+q+zero], 1ll * dp[u][][p+zero] * dp[v][][q+zero] % mod);
add(tmp[][p+q+zero], 1ll * dp[u][][p+zero] * dp[v][][q+zero-] % mod);
add(tmp[][p+q+zero], 1ll * dp[u][][p+zero] * dp[v][][q+zero+] % mod); add(tmp[][p+q+zero], 1ll * dp[u][][p+zero] * dp[v][][q+zero] % mod);
add(tmp[][p+q+zero], 1ll * dp[u][][p+zero] * dp[v][][q+zero] % mod);
add(tmp[][p+q+zero], 1ll * dp[u][][p+zero] * dp[v][][q+zero] % mod); add(tmp[][p+q+zero], 1ll * dp[u][][p+zero] * dp[v][][q+zero] % mod);
add(tmp[][p+q+zero], 1ll * dp[u][][p+zero] * dp[v][][q+zero] % mod);
add(tmp[][p+q+zero], 1ll * dp[u][][p+zero] * dp[v][][q+zero] % mod); if(tmp[][p + q + zero]) down[u] = min(down[u], p + q), up[u] = max(up[u], p + q);
if(tmp[][p + q + zero]) down[u] = min(down[u], p + q), up[u] = max(up[u], p + q);
if(tmp[][p + q + zero]) down[u] = min(down[u], p + q), up[u] = max(up[u], p + q);
}
} for(int i = down[u]; i <= up[u]; i++)
for(int j = ; j < ; j++)
dp[u][j][i + zero] = tmp[j][i + zero];
}
} void init() {
for(int i = ; i <= n; i++)
edge[i].clear();
memset(dp, , sizeof(dp));
} int main() {
while(scanf("%d", &n) != EOF) {
init();
for(int i = ; i < n; i++) {
int u, v; scanf("%d%d", &u, &v);
edge[u].push_back(v);
edge[v].push_back(u);
}
dfs(, );
int ans = ;
for(int i = down[]; i <= up[]; i++) {
add(ans, 1ll * max(, i) * dp[][][i + zero] % mod);
add(ans, 1ll * max(, i + ) * dp[][][i + zero] % mod);
add(ans, 1ll * max(, i - ) * dp[][][i + zero] % mod);
}
add(ans, * ans % mod);
printf("%d\n", ans);
}
return ;
}
/*
*/
HDU - 4809 树形dp的更多相关文章
- hdu 4123 树形DP+RMQ
http://acm.hdu.edu.cn/showproblem.php? pid=4123 Problem Description Bob wants to hold a race to enco ...
- HDU 1520 树形dp裸题
1.HDU 1520 Anniversary party 2.总结:第一道树形dp,有点纠结 题意:公司聚会,员工与直接上司不能同时来,求最大权值和 #include<iostream> ...
- HDU 1561 树形DP入门
The more, The Better Time Limit: 6000/2000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Oth ...
- HDU 2196树形DP(2个方向)
HDU 2196 [题目链接]HDU 2196 [题目类型]树形DP(2个方向) &题意: 题意是求树中每个点到所有叶子节点的距离的最大值是多少. &题解: 2次dfs,先把子树的最大 ...
- HDU 1520 树形DP入门
HDU 1520 [题目链接]HDU 1520 [题目类型]树形DP &题意: 某公司要举办一次晚会,但是为了使得晚会的气氛更加活跃,每个参加晚会的人都不希望在晚会中见到他的直接上司,现在已知 ...
- codevs 1380/HDU 1520 树形dp
1380 没有上司的舞会 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题解 查看运行结果 回到问题 题目描述 Description Ural大学有N个职员 ...
- HDU 5834 [树形dp]
/* 题意:n个点组成的树,点和边都有权值,当第一次访问某个点的时候获得利益为点的权值 每次经过一条边,丢失利益为边的权值.问从第i个点出发,获得的利益最大是多少. 输入: 测试样例组数T n n个数 ...
- hdu 4267 树形DP
思路:先dfs一下,找出1,n间的路径长度和价值,回溯时将该路径长度和价值清零.那么对剩下的图就可以直接树形dp求解了. #include<iostream> #include<al ...
- hdu 4607 (树形DP)
当时比赛的时候我们找出来只要求出树的最长的边的节点数ans,如果要访问点的个数n小于ans距离直接就是n-1 如果大于的话就是(n-ans)*2+ans-1,当时求树的直径难倒我们了,都不会树形dp ...
随机推荐
- (转) 经典功率谱估计及Matlab仿真
原文出自:http://www.cnblogs.com/jacklu/p/5140913.html 功率谱估计在分析平稳各态遍历随机信号频率成分领域被广泛使用,并且已被成功应用到雷达信号处理.故障诊断 ...
- MacBook Air网络问题
自从买了本本之后,一直觉得无线网连接不能正常使用,最开始觉得是网络不给力,因为图标都没有满格.后来搬家,网速家里的window,iphone设备都能正常使用,就我的mac 本本图标显示满格,但是网页打 ...
- android onActivityResult的执行
1.如果activity中重写了onActivityResult函数,同时添加在该activity的fragment也重写了onActivtyResult函数,那么会执行Activity的onActi ...
- 关于thinkpad安装win10操作系统
thinkpad预装的是win8或者win10,会有自己的分区方式是GPT,所以会出现两个引导分区. F2进入tinkpad的bios,F12进入启动选项 我们用pe进入后,用分区工具删除两个分区,然 ...
- 给Ubuntu替换阿里的源
1. 阿里巴巴镜像源站点 有所有linux的源的镜像加速. 点击查看介绍 2. 具体配置方法在这里 copy: ubuntu 18.04(bionic) 配置如下 创建自己的配置文件,比如创建文件 / ...
- 并发编程(三) IO模型
五 IO模型 常用的IO模型有4种: 阻塞IO 非阻塞IO IO多路复用 异步IO 不常用的有: 驱动信号 5.1 阻塞IO.非阻塞IO 阻塞IO:进程不能做其他的事情 非阻塞IO:等待数据无阻塞 阻 ...
- host映射方法
host映射: host是根据TCP/IP for Windows 的标准来工作的,它的作用是包含IP地址和Host name(主机名)的映射关系,是一个映射IP地址和Host name(主机名)的规 ...
- python3学习笔记.4.turtle绘图
先放上参考 https://docs.python.org/3/library/turtle.html //********************************************** ...
- 01-Coredump核心转存&&Linux程序地址分析【转】
转自:http://www.itwendao.com/article/detail/404132.html 目录(?)[-] 一Core Dump核心转存 二Linux程序地址分析 一Core Dum ...
- overridePendingTransition()使用
实现两个 Activity 切换时的动画.在Activity中使用有两个参数:进入动画和出去的动画. 注意1.必须在 StartActivity() 或 finish() 之后立即调用.2.而且在 ...