题意:给你一颗树,然后这颗树有n*n条路径,a->b和b->a算是一条,然后路径的权值是 vi*(-1)^(i+1)  注意是点有权值。

从上头往下考虑是点分治,从下向上考虑就是树形DP,分成三类路径:1.指定点单点 2.跨过指定点3.没跨过指定点但不是单点

细节要好好打磨一下,然后还是用long long比较稳 ,1LL*还是有点危险.

点分治:

#include<cstdio>
#include<vector>
#include<cstring>
#include<algorithm>
using namespace std;
const int N=2e5+;
const int P=1e9+;
vector<int>G[N];
bool vis[N];
int SQ,SO,Q,O,q[N],o[N],rt,m,nq[N],no[N],sz[N],v[N],mx[N],ans;
void dfs(int u,int fa,int S){
sz[u]=mx[u]=;
q[u]=o[u]=no[u]=nq[u]=;
for(int i=;i<(int)G[u].size();++i) {
int v=G[u][i];
if(vis[v]||v==fa) continue;
dfs(v,u,S);
sz[u]+=sz[v];
mx[u]=max(mx[u],sz[v]);
}
mx[u]=max(mx[u],S-sz[u]);
if(m>mx[u]) rt=u,m=mx[u];
}
void solve(int u,int nv,int f,int fa,int last){
int t=(nv+f*v[u])%P;
if(f==-) o[fa]=(o[fa]+t)%P,O=(O+t)%P,++no[fa],++SO;
else q[fa]=(q[fa]+t)%P,Q=(Q+t)%P,++nq[fa],++SQ;
for(int i=;i<(int)G[u].size();++i) {
int v=G[u][i];
if(v==last||vis[v]) continue;
solve(v,t,-f,fa,u);
}
}
int work(int u,int S){
m=1e9+;
dfs(u,,S);
vis[rt]=;
Q=O=SQ=SO=;
for(int i=;i<(int)G[rt].size();++i) if(!vis[G[rt][i]]) solve(G[rt][i],v[rt],-,G[rt][i],rt);
for(int i=;i<(int)G[rt].size();++i) if(!vis[G[rt][i]]){
int t=G[rt][i];
ans=((ans-1LL*no[t]*(O-o[t])%P)%P-1LL*o[t]*(SO-no[t])%P)%P;
ans=(ans+1LL*no[t]*(SO-no[t])%P*v[rt]%P)%P;
ans=((ans+1LL*nq[t]*(Q-q[t])%P)%P+1LL*q[t]*(SQ-nq[t])%P)%P;
ans=(ans-1LL*nq[t]*(SQ-nq[t])%P*v[rt]%P)%P;
ans=(ans+(q[t]<<)%P)%P;
}
ans=((ans+v[rt])%P+P)%P;
int nt=rt;
for(int i=;i<(int)G[nt].size();++i) if(!vis[G[nt][i]])
work(G[nt][i],sz[G[nt][i]]>sz[nt]?S-sz[nt]:sz[G[nt][i]]);
}
int main(){
int n,x,y;
scanf("%d",&n);
for(int i=;i<=n;++i) scanf("%d",v+i);
for(int i=;i<n;++i) {
scanf("%d%d",&x,&y);
G[x].push_back(y);
G[y].push_back(x);
}
work(,n);
printf("%d\n",(ans+P)%P);
}

树形DP的思路是参考的这里

#include<cstdio>
#include<vector>
#include<cstring>
#include<algorithm>
using namespace std;
const int N=2e5+;
const int P=1e9+;
vector<int>G[N];
long long dp[N][],num[N][],ans,v[N];
void dfs(int u,int fa){
for(int i=;i<(int)G[u].size();++i) {
int vt=G[u][i];
if(vt==fa) continue;
dfs(vt,u);
dp[u][]=(((dp[u][]+dp[vt][])%P-1LL*v[u]*num[vt][]%P)%P+P)%P;
dp[u][]=(((dp[u][]+dp[vt][])%P+1LL*v[u]*num[vt][]%P)%P+P)%P;
num[u][]=(num[u][]+num[vt][])%P;
num[u][]=(num[u][]+num[vt][])%P;
}
for(int i=;i<(int)G[u].size();++i) {
int vt=G[u][i];
if(vt==fa) continue;
ans=(ans+2LL*(num[u][]-num[vt][])%P*dp[vt][]%P+1LL*(num[u][]-num[vt][])*v[u]%P*num[vt][]%P)%P;
ans=(ans+2LL*(num[u][]-num[vt][])%P*dp[vt][]%P-1LL*(num[u][]-num[vt][])*v[u]%P*num[vt][]%P)%P;
}
ans=(ans+(dp[u][]<<)%P+v[u])%P;
++num[u][];
num[u][]%=P;
dp[u][]=(dp[u][]+v[u])%P;
}
int main(){
int n,x,y;
scanf("%d",&n);
for(int i=;i<=n;++i) scanf("%lld",v+i);
for(int i=;i<n;++i) {
scanf("%d%d",&x,&y);
G[x].push_back(y);
G[y].push_back(x);
}
dfs(,);
printf("%lld\n",(ans+P)%P);
}

E. Alternating Tree 树点分治|树形DP的更多相关文章

  1. 算法笔记--树的直径 && 树形dp && 虚树 && 树分治 && 树上差分 && 树链剖分

    树的直径: 利用了树的直径的一个性质:距某个点最远的叶子节点一定是树的某一条直径的端点. 先从任意一顶点a出发,bfs找到离它最远的一个叶子顶点b,然后再从b出发bfs找到离b最远的顶点c,那么b和c ...

  2. (中等) HDU 5293 Tree chain problem,树链剖分+树形DP。

    Problem Description   Coco has a tree, whose vertices are conveniently labeled by 1,2,…,n.There are ...

  3. [集训队作业2018]蜀道难——TopTree+贪心+树链剖分+链分治+树形DP

    题目链接: [集训队作业2018]蜀道难 题目大意:给出一棵$n$个节点的树,要求给每个点赋一个$1\sim n$之内的权值使所有点的权值是$1\sim n$的一个排列,定义一条边的权值为两端点权值差 ...

  4. CF F - Tree with Maximum Cost (树形DP)给出你一颗带点权的树,dist(i, j)的值为节点i到j的距离乘上节点j的权值,让你任意找一个节点v,使得dist(v, i) (1 < i < n)的和最大。输出最大的值。

    题目意思: 给出你一颗带点权的树,dist(i, j)的值为节点i到j的距离乘上节点j的权值,让你任意找一个节点v,使得dist(v, i) (1 < i < n)的和最大.输出最大的值. ...

  5. [codeforces161D]Distance in Tree(点分治/树形dp)

    题意:求树上距离为k的点对个数: 解题关键:练习一下点分治不用容斥 而直接做的做法.注意先查询,后更新. 不过这个方法有个缺陷,每次以一个新节点为根,必须memset mp数组,或许使用map会好些, ...

  6. 『You Are Given a Tree 整体分治 树形dp』

    You Are Given a Tree Description A tree is an undirected graph with exactly one simple path between ...

  7. [BZOJ2152]聪聪可可 点分治/树形dp

    2152: 聪聪可可 Time Limit: 3 Sec  Memory Limit: 259 MB Submit: 3602  Solved: 1858 [Submit][Status][Discu ...

  8. BZOJ 2152 / Luogu P2634 [国家集训队]聪聪可可 (点分治/树形DP)

    题意 一棵树,给定边权,求满足两点之间的路径上权值和为3的倍数的点对数量. 分析 点分治板题,对每个重心求子树下面的到根的距离模3分别为0,1,2的点的个数就行了. O(3nlogn)O(3nlogn ...

  9. 2014 Super Training #9 E Destroy --树的直径+树形DP

    原题: ZOJ 3684 http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3684 题意: 给你一棵树,树的根是树的中心(到其 ...

随机推荐

  1. Font-Awesome使用教程

    何为Font-Awesome Font Awesome gives you scalable vector icons that can instantly be customized — size, ...

  2. Swift-Realm数据库的使用详解

    Swift-Realm数据库的使用详解 概述 Realm 是一个跨平台的移动数据库引擎,其性能要优于 Core Data 和 FMDB - 移动端数据库性能比较, 我们可以在 Android 端 re ...

  3. CF786B Legacy(线段树优化建边)

    模板题CF786B Legacy 先说算法 如果需要有n个点需要建图 给m个需要建边的信息,从单点(或区间内所有点)向一区间所有点连边 如果暴力建图复杂度\(mn^2\) 以单点连向区间为例,在n个点 ...

  4. Jenkins 项目构建

    一:新建项目 (1)点击新建,输入项目名称--构建一个自由风格的软件项目,点击ok (2)构建触发器-----设置每两分钟执行一次 其中有5个参数 (*****) 第一个是代表分钟  一小时内的分钟数 ...

  5. 使用JDBC连接oracle数据库

    需要jar包:ojdbc6.jar

  6. 【GISER&&规划】我这二三年

    从从参加工作到现在,已经接近三年了.在这不长不短的时间里,我的职业规划犹如正余弦函数一样变化,一直游离在前端和后端之间. 第一年入职,被安排维护和拓展一套基于JAVA实现的地图瓦片生产工艺程序,不算复 ...

  7. Mysql常用sql语句(八)- where 条件查询

    测试必备的Mysql常用sql语句,每天敲一篇,每次敲三遍,每月一循环,全都可记住!! https://www.cnblogs.com/poloyy/category/1683347.html 前言 ...

  8. Git管理修改、撤销修改、删除文件

    什么是修改?比如你新增了一行,这就是一个修改,删除了一行,也是一个修改,更改了某些字符,也是一个修改,删了一些又加了一些,也是一个修改,甚至创建一个新文件,也算一个修改. a.管理修改 对于提交修改, ...

  9. [hdu4713 Permutation]DP

    题意:将一个数拆成若干数的和使得它们的最小公倍数最大 思路:一个数x可以拆成p1k1 + p2k2 + ... + pnkn形式,其中pi是质数或1.对于最小公倍数最大的情况,一定可以表示成这种形式. ...

  10. 在dynamics 365 中,看字段的描述需要到系统字段设置里面才能看到,这里提供一种sql直接看字段和实体名描述的方法

    1.在crm对应的主数据库执行下面存储过程: -- ============================================= -- Author: <Author,,Name& ...