E. Alternating Tree 树点分治|树形DP
题意:给你一颗树,然后这颗树有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的更多相关文章
- 算法笔记--树的直径 && 树形dp && 虚树 && 树分治 && 树上差分 && 树链剖分
树的直径: 利用了树的直径的一个性质:距某个点最远的叶子节点一定是树的某一条直径的端点. 先从任意一顶点a出发,bfs找到离它最远的一个叶子顶点b,然后再从b出发bfs找到离b最远的顶点c,那么b和c ...
- (中等) HDU 5293 Tree chain problem,树链剖分+树形DP。
Problem Description Coco has a tree, whose vertices are conveniently labeled by 1,2,…,n.There are ...
- [集训队作业2018]蜀道难——TopTree+贪心+树链剖分+链分治+树形DP
题目链接: [集训队作业2018]蜀道难 题目大意:给出一棵$n$个节点的树,要求给每个点赋一个$1\sim n$之内的权值使所有点的权值是$1\sim n$的一个排列,定义一条边的权值为两端点权值差 ...
- 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)的和最大.输出最大的值. ...
- [codeforces161D]Distance in Tree(点分治/树形dp)
题意:求树上距离为k的点对个数: 解题关键:练习一下点分治不用容斥 而直接做的做法.注意先查询,后更新. 不过这个方法有个缺陷,每次以一个新节点为根,必须memset mp数组,或许使用map会好些, ...
- 『You Are Given a Tree 整体分治 树形dp』
You Are Given a Tree Description A tree is an undirected graph with exactly one simple path between ...
- [BZOJ2152]聪聪可可 点分治/树形dp
2152: 聪聪可可 Time Limit: 3 Sec Memory Limit: 259 MB Submit: 3602 Solved: 1858 [Submit][Status][Discu ...
- BZOJ 2152 / Luogu P2634 [国家集训队]聪聪可可 (点分治/树形DP)
题意 一棵树,给定边权,求满足两点之间的路径上权值和为3的倍数的点对数量. 分析 点分治板题,对每个重心求子树下面的到根的距离模3分别为0,1,2的点的个数就行了. O(3nlogn)O(3nlogn ...
- 2014 Super Training #9 E Destroy --树的直径+树形DP
原题: ZOJ 3684 http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3684 题意: 给你一棵树,树的根是树的中心(到其 ...
随机推荐
- springboot docker jenkins 自动化部署并上传镜像
springboot + docker + jenkins自动化部署项目,jenkins.mysql.redis都是docker运行的,并且没有使用虚拟机,就在阿里云服务器(centos7)运行 1. ...
- 27.rm命令
rm命令可以删除指定的文件或目录.也可以将某个目录及其下属的所有文件及其子目录均删除掉.对于链接文件,只是删除整个链接文件,而原有文件保持不变. 选项:-f:强制删除. -r:递归处理,将指定目录下的 ...
- 项目Alpha冲刺 Day12
1)站立式会议: 2)今日安排: 项目演示. 3)项目情况 项目进展:系统已实现预期的所有的功能.问题困难:系统测试不够全面,主要做功能测试,对于非功能测试,如压力测试.效能测试.安全性等并未测试.心 ...
- Vue项目中实现图片懒加载
个人网站 https://iiter.cn 程序员导航站 开业啦,欢迎各位观众姥爷赏脸参观,如有意见或建议希望能够不吝赐教! ---对于图片过多的页面,为了加速页面加载速度,所以很多时候我们需要将页面 ...
- 由JS数组去重说起
一.问题描述: var array=[1,45,3,1,4,67,45],请编写一个函数reDup来去掉其中的重复项,即 reDup(array); console.log(array);//[1,4 ...
- C#对象初始化器
1.对象初始化器 Student objStu2 = new Student() { StudentId=, //属性之间使用","分隔 StudentName="小明& ...
- C. K-Complete Word(小小的并查集啦~)
永久打开的传送门 \(\color{Pink}{-------------分割-------------}\) \(n最大有2e5,那么暴力一定不行,找规律\) \(我们发现第i位的字符一定和第i+k ...
- python学习笔记-零碎知识点
1. 绝对值 abs(-4) 结果: 4 2.
- spring的后台数据校验
数据校验对于开发项目来说是必须的.校验一般分为前台校验和后台校验,前台校验是必须要做的,后台校验是可选的.后台校验相对前台校验来说配置起来一般更复杂.前台校验通过js做,前台校验一般非常容易绕过.sp ...
- 【2020Java最新学习路线】写了很久,这是一份最适合普通大众、科班、非科班的路线
点赞再看,养成习惯,微信搜索[三太子敖丙]关注这个互联网苟且偷生的工具人. 本文 GitHub https://github.com/JavaFamily 已收录,有一线大厂面试完整考点.资料以及我的 ...