Codeforces Round #474-E(树形dp)
一、题目链接
http://codeforces.com/contest/960/problem/B
二、题意
给定一棵$N$个节点的树,每个节点的权值$V$。定义树中两点$u_1$和$u_m$的权值和为$A(u_1, u_m) = V_{u_1} - V{u_2} + V{u_3} - V{u_4} + \cdots + (-1)^{m+1}V{u_m}$。求$\sum\limits_{u_i=1}^{N}\sum\limits_{u_j=1}^{N}A(u_i, u_j)\ \%\ (10^9+7)$。
三、思路
显然的树形$dp$。采用"两遍扫描"法。
设$dp1[i]$表示:从$i$出发,在以$i$为根的子树中,可得到的权值和。那么,很容易想到一个式子:预处理$dp1[i]=V[i]$,表示从$i$走到$i$自己的$A$值。然后,对于$i$的所有子节点$j$,有$dp1[i]\ +=\ V[i] - dp1[j]$。在思路中,为表述简洁,我们不考虑取模。如果这样,那就大错特错了。你会发现,连样例2都过不去。画一棵链式树可以发现,其实正确的式子是,$dp1[i] += f[j] * V[i] - dp1[j]$,其中$f[j]$表示以$j$为根节点的子树中节点的个数。为什么要乘以$f[j]$,因为对于以$j$为根的子树中每一个节点$r$,节点$i$都要走一遍去计算$A(i, r)$,所以这个地方要乘以一个$f[j]$。我一开始就是没写,导致一直样例都过不去。
设$dp2[i]$表示:从$i$出发,可得到的权值和(最后累加$dp2[i]$即可)。一遍扫描完成之后,显然有$dp2[1] = dp1[1]$(当然了,要看你的dfs是从哪个节点开始的)。然后,对于$i$的所有子节点$j$,有
\[dp2[j] = 除去以j为根的子树中所有节点的个数*V[j] + dp1[j] - (dp2[i] - 以j为根的子树中所有节点的个数*V[i] + dp1[j])\]
形式化(规范化)表示就是:
\[dp2[j] = (N - f[j]) * V[j] + dp1[j] - (dp2[i] - f[j] * V[i] + dp1[j])\]
即\[dp2[j] = (N - f[j]) * V[j] - (dp2[i] - f[j] * V[i])\]
注意这些"+"、“-”号的意义哦。在第一次扫描中减(实际上是+负的)了的,这里要加(实际上是-正的)回去,就等于没动(没+负也没-正)。同时,第一次加了$f[j] * V[i]$,那么,这一次要减去这个值。
另外,要注意的就是,取模的问题。因为涉及负数和乘法,一次加模数再取模不一定能保证结果为正。所以,要对$dp1[i]$和$dp2[i]$分别做两次取模。
四、代码实现
#include<bits/stdc++.h> using namespace std; #define pb(x) push_back(x) #define mk(x, y) make_pair(x, y) typedef long long LL; typedef pair<int, int> PII; typedef pair<LL, LL> PLL; ; template <class T> inline void read(T &x) { int t; bool flag = false; ')) ; '; + t - '; if(flag) x = -x; } typedef struct { int to, next; } Edge; Edge tree[MAXN * ]; int head[MAXN], cnt; void add(int from, int to) { tree[cnt].to = to; tree[cnt].next = head[from]; head[from] = cnt++; } void init() { memset(head, -, sizeof(head)); cnt = ; } LL N, v[MAXN], dp0[MAXN], f0[MAXN], dp2[MAXN]; const LL MOD = 1000000007LL; void dfs0(int root, int par) { dp0[root] = v[root], f0[root] = ; for(int i = head[root]; ~i; i = tree[i].next) { int to = tree[i].to; if(to != par) { dfs0(to, root); dp0[root] = (f0[to] * v[root] + dp0[root] - dp0[to] + MOD) % MOD; dp0[root] = (dp0[root] + MOD) % MOD; f0[root] += f0[to]; } } } void dfs1(int root, int par) { for(int i = head[root]; ~i; i = tree[i].next) { int to = tree[i].to; if(to != par) { dp2[to] = ((N - f0[to]) * v[to] + dp0[to] - (dp2[root] + dp0[to] - f0[to] * v[root] + MOD) + MOD) % MOD; dp2[to] = (dp2[to] + MOD) % MOD; dfs1(to, root); } } } int main() { #ifndef ONLINE_JUDGE freopen("inputE.txt", "r", stdin); #endif // ONLINE_JUDGE init(); int a, b; read(N); ; i <= N; ++i)read(v[i]); ; i < N; ++i) { read(a), read(b); add(a, b); add(b, a); } dfs0(, -); dp2[] = dp0[]; dfs1(, -); LL ans = ; ; i <= N; ++i)ans = (ans + dp2[i]) % MOD; cout << ans << endl; ; }
Codeforces Round #474-E(树形dp)的更多相关文章
- Educational Codeforces Round 52F(树形DP,VECTOR)
#include<bits/stdc++.h>using namespace std;int n,k;vector<int>son[1000007];int dp[100000 ...
- codeforces 212E IT Restaurants(树形dp+背包思想)
题目链接:http://codeforces.com/problemset/problem/212/E 题目大意:给你一个无向树,现在用两种颜色去给这颗树上的节点染色.用(a,b)表示两种颜色分别染的 ...
- Codeforces 123E Maze(树形DP+期望)
[题目链接] http://codeforces.com/problemset/problem/123/E [题目大意] 给出一棵,给出从每个点出发的概率和以每个点为终点的概率,求出每次按照dfs序从 ...
- codeforces 709E E. Centroids(树形dp)
题目链接: E. Centroids time limit per test 4 seconds memory limit per test 512 megabytes input standard ...
- 2017 Wuhan University Programming Contest (Online Round) B Color 树形dp求染色方法数
/** 题目:Color 链接:https://oj.ejq.me/problem/23 题意:给定一颗树,将树上的点最多染成m种颜色,有些节点不可以染成某些颜色.相邻节点颜色不同.求染色方法数. 思 ...
- CodeForces 77C Beavermuncher-0xFF (树形dp)
不错的树形dp.一个结点能走多次,树形的最大特点是到达后继的路径是唯一的,那个如果一个结点无法往子结点走,那么子结点就不用考虑了. 有的结点不能走完它的子结点,而有的可能走完他的子节点以后还会剩下一些 ...
- bzoj 4424: Cf19E Fairy && codeforces 19E. Fairy【树形dp】
参考:https://blog.csdn.net/heheda_is_an_oier/article/details/51131641 这个找奇偶环的dp1真是巧妙,感觉像tarjan一样 首先分情况 ...
- Educational Codeforces Round 63-D(基础DP)
题目链接:https://codeforces.com/contest/1155/problem/D 题意:给定n个数,可以选择一段连续子段将其乘x,也可以不操作,求最大连续子段和. 思路:比赛时觉得 ...
- Educational Codeforces Round 62 E 局部dp + 定义状态取消后效性
https://codeforces.com/contest/1140/problem/E 局部dp + 定义状态取消后效性 题意 给你一个某些位置可以改变的字符串,假如字符串存在回文子串,那么这个字 ...
- Codeforces Round #544 (Div. 3) dp + 双指针
https://codeforces.com/contest/1133/problem/E 题意 给你n个数(n<=5000),你需要对其挑选并进行分组,总组数不能超过k(k<=5000) ...
随机推荐
- Bigtable阶段性总结(版本1)
Bigtable的角色:为大规模的结构化数据提供高效的存储.管理与查询. Bigtable的针对性: 大规模数据需要大规模集群支持,带来了存储.管理.查询.容错上的复杂性. 关系型数据库在数据规模较大 ...
- Spring IOC 源码简单分析 01 - BeanFactory
### 准备 ## 目标 了解 Spring IOC 的基础流程 ## 相关资源 Offical Doc:http://docs.spring.io/spring/docs/4.3.9.RELEASE ...
- 进入mac U盘文件夹命令
cd /Volumes 这条命令进入到你的u盘的上层文件夹,在这个文件夹里面包含了你的u盘,比如我的u盘名为SONG 则进入u盘的命令为 cd /Volumes/SONG
- ADSL拨号上网或者光纤上网设置概要(原创)
不管是在梧州设置光纤还是在太平设置ADSL拨号上网每次设置上网一体机的时候都是遇到各种麻烦...这次又是弄了N久,每次问题各不一样.总结一下操作过程,方便以后又遇问题回头查询自个微博.一.设置电话线的 ...
- Credentials(Rails5.2新) 很基础的知识点,具体还要实操。
Credentials(Rails5.2新) 增加config/credentials.yml.enc 憎加config/master.key 移除了5.1使用的config/secrets.yml, ...
- 爬虫框架:scrapy
一 介绍 Scrapy一个开源和协作的框架,其最初是为了页面抓取 (更确切来说, 网络抓取 )所设计的,使用它可以以快速.简单.可扩展的方式从网站中提取所需的数据.但目前Scrapy的用途十分广泛,可 ...
- 【转】 linux的网络接口之扫盲
[转] linux的网络接口之扫盲 转自:http://blog.csdn.net/zhangxinrun/article/details/6820433 (1)网络接口的命名 这里并不存在一定的命名 ...
- angular学习笔记系列一
首先我们要明确一点,angular在web应用的角色,在传统的多页面web应用程序中,服务器根据输出数据和html模板渲染好页面所需的html输出到页面(所谓的服务器装配程序),随着单页面应用程序和a ...
- 传智:自己简单实现一个struts2框架的demo
struts2的结构图: 代码实现: 组织结构: 主要代码: package cn.itcast.config; import org.apache.log4j.Logger; import org. ...
- 作业要求20181023-4 Alpha阶段第2周/共2周 Scrum立会报告+燃尽图 02
作业要求[https://edu.cnblogs.com/campus/nenu/2018fall/homework/2284] 版本控制:https://git.coding.net/liuyy08 ...