一、题目链接

  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)的更多相关文章

  1. Educational Codeforces Round 52F(树形DP,VECTOR)

    #include<bits/stdc++.h>using namespace std;int n,k;vector<int>son[1000007];int dp[100000 ...

  2. codeforces 212E IT Restaurants(树形dp+背包思想)

    题目链接:http://codeforces.com/problemset/problem/212/E 题目大意:给你一个无向树,现在用两种颜色去给这颗树上的节点染色.用(a,b)表示两种颜色分别染的 ...

  3. Codeforces 123E Maze(树形DP+期望)

    [题目链接] http://codeforces.com/problemset/problem/123/E [题目大意] 给出一棵,给出从每个点出发的概率和以每个点为终点的概率,求出每次按照dfs序从 ...

  4. codeforces 709E E. Centroids(树形dp)

    题目链接: E. Centroids time limit per test 4 seconds memory limit per test 512 megabytes input standard ...

  5. 2017 Wuhan University Programming Contest (Online Round) B Color 树形dp求染色方法数

    /** 题目:Color 链接:https://oj.ejq.me/problem/23 题意:给定一颗树,将树上的点最多染成m种颜色,有些节点不可以染成某些颜色.相邻节点颜色不同.求染色方法数. 思 ...

  6. CodeForces 77C Beavermuncher-0xFF (树形dp)

    不错的树形dp.一个结点能走多次,树形的最大特点是到达后继的路径是唯一的,那个如果一个结点无法往子结点走,那么子结点就不用考虑了. 有的结点不能走完它的子结点,而有的可能走完他的子节点以后还会剩下一些 ...

  7. bzoj 4424: Cf19E Fairy && codeforces 19E. Fairy【树形dp】

    参考:https://blog.csdn.net/heheda_is_an_oier/article/details/51131641 这个找奇偶环的dp1真是巧妙,感觉像tarjan一样 首先分情况 ...

  8. Educational Codeforces Round 63-D(基础DP)

    题目链接:https://codeforces.com/contest/1155/problem/D 题意:给定n个数,可以选择一段连续子段将其乘x,也可以不操作,求最大连续子段和. 思路:比赛时觉得 ...

  9. Educational Codeforces Round 62 E 局部dp + 定义状态取消后效性

    https://codeforces.com/contest/1140/problem/E 局部dp + 定义状态取消后效性 题意 给你一个某些位置可以改变的字符串,假如字符串存在回文子串,那么这个字 ...

  10. Codeforces Round #544 (Div. 3) dp + 双指针

    https://codeforces.com/contest/1133/problem/E 题意 给你n个数(n<=5000),你需要对其挑选并进行分组,总组数不能超过k(k<=5000) ...

随机推荐

  1. install ros-indigo-filters

    CMake Warning at /opt/ros/indigo/share/catkin/cmake/catkinConfig.cmake: (find_package): Could not fi ...

  2. python 浮点数取绝对值

    import math print(math.fabs(-2.1)) print(math.fabs(-0.0)) print(math.fabs(10.1)) print(math.fabs(0.0 ...

  3. vim 将文件从dos格式转换到unix格式

    dos格式文件传输到unix系统时,会在每行的结尾多一个^M(/r),当然也有可能看不到.但是在vim的时候,会在下面显示此文件的格式,比如 "dos.txt" [dos] 120 ...

  4. Rails 5 Test Prescriptions 倒数第2章spring gem 如何让测试变快。分离rails(只有原理)

    Spring speeds up development by keeping your application running in the background Rails程序自动增加:sprin ...

  5. android------引导页两种实现方式(原生和WebView网页实现)

    有的App当你第一次打开的是和常常会有引导页来描述一些App信息(功能,特点),当然也要做验证,验证第二次进入不进入引导页,直接进入App,此博客借助ViewPager来实现引导页, ViewPage ...

  6. Python在七牛云平台的应用(一)

    七牛云:(引用百度的介绍)七牛云是国内领先的企业级公有云服务商,致力于打造以数据为核心的场景化PaaS服务.围绕富媒体场景,七牛先后推出了对象存储,融合CDN加速,数据通用处理,内容反垃圾服务,以及直 ...

  7. iOS UILabel两端对齐的实现(可包括中英文/数字)

    - (void)conversionCharacterInterval:(NSInteger)maxInteger current:(NSString *)currentString withLabe ...

  8. New Concept English Two 29 79

    $课文77  一例成功的手术 829. The mummy of an Egyptian woman who died in 800 B.C. has just had an operation. 死 ...

  9. springmvc的表单标签

    1. Spring提供的轻量级标签库 2.可在JSP页面中渲染HTML元素的标签 3 用法 1)必须在JSP页面的开头处声明taglib指令 <%@ taglib prefix="fm ...

  10. 记录一些js框架用途

    accounting.min.js 货币格式化alertify.min.js 提示信息库amd.loader.js 按需动态加载js文件angular-cookies.js 处理cookieangul ...