[NOIP2018TG]保卫王国
[NOIP2018TG]保卫王国
BZOJ
luogu
当动态dp模板题写的,(全集-最大点权独立集)不能放军队的+inf,必须放军队-inf即可
注意矩阵乘法的顺序问题
#define ll long long
#define ls x<<1,l,mid
#define rs x<<1|1,mid+1,r
#include<bits/stdc++.h>
using namespace std;
const int _=1e5+5,inf=1e9;
int re(){
int x=0,w=1;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')w=-1;ch=getchar();}
while(ch>='0'&&ch<='9')x=x*10+ch-'0',ch=getchar();
return x*w;
}
int n,m,cnt,ts;
int h[_],p[_],fa[_],sz[_],son[_],top[_],dfn[_],id[_],ed[_];
ll sum,Ans,f[2][_];
string type;
struct edge{int to,next,w;}e[_<<1];
struct Mx{
ll g[2][2];
Mx(){memset(g,0,sizeof(g));}
}t[_<<2],val[_];
Mx operator *(Mx x,Mx y){
Mx z;
for(int i=0;i<2;i++)
for(int j=0;j<2;j++)
for(int k=0;k<2;k++)
z.g[i][j]=max(z.g[i][j],x.g[i][k]+y.g[k][j]);
return z;
}
void link(int u,int v){
e[++cnt]=(edge){v,h[u]};h[u]=cnt;
e[++cnt]=(edge){u,h[v]};h[v]=cnt;
}
void dfs(int u){
sz[u]=1;
for(int i=h[u];i;i=e[i].next){
int v=e[i].to;
if(v==fa[u])continue;
fa[v]=u;dfs(v);sz[u]+=sz[v];
if(sz[v]>sz[son[u]])son[u]=v;
}
}
void dfs(int u,int tp){
top[u]=tp;id[dfn[u]=++ts]=u;
if(son[u])dfs(son[u],tp);
else ed[top[u]]=ts;
for(int i=h[u];i;i=e[i].next){
int v=e[i].to;
if(v^fa[u]&&v^son[u])dfs(v,v);
}
}
void dp(int u){
f[1][u]=p[u];
for(int i=h[u];i;i=e[i].next){
int v=e[i].to;
if(v==fa[u])continue;
dp(v);
f[0][u]+=max(f[1][v],f[0][v]);
f[1][u]+=f[0][v];
}
}
void pu(int x){t[x]=t[x<<1]*t[x<<1|1];}
void bu(int x,int l,int r){
if(l==r){
t[x].g[0][0]=t[x].g[0][1]=f[0][id[l]]-max(f[0][son[id[l]]],f[1][son[id[l]]]);
t[x].g[1][0]=f[1][id[l]]-f[0][son[id[l]]];t[x].g[1][1]=-inf;val[l]=t[x];return;
}
int mid=(l+r)>>1;bu(ls);bu(rs);pu(x);
}
void upd(int x,int l,int r,int k){
if(l==r){t[x]=val[k];return;}
int mid=(l+r)>>1;
if(k<=mid)upd(ls,k);else upd(rs,k);pu(x);
}
Mx query(int x,int l,int r,int ql,int qr){
if(ql<=l&&r<=qr)return t[x];
int mid=(l+r)>>1;
if(qr<=mid)return query(ls,ql,qr);
if(ql>mid)return query(rs,ql,qr);
return query(ls,ql,qr)*query(rs,ql,qr);
}
void modify(int u,int v){
val[dfn[u]].g[1][0]+=v;
Mx od,nw;
while(u){
od=query(1,1,n,dfn[top[u]],ed[top[u]]);
upd(1,1,n,dfn[u]);
nw=query(1,1,n,dfn[top[u]],ed[top[u]]);
u=fa[top[u]];
val[dfn[u]].g[0][0]+=max(nw.g[0][0],nw.g[1][0])-max(od.g[0][0],od.g[1][0]);
val[dfn[u]].g[0][1]=val[dfn[u]].g[0][0];
val[dfn[u]].g[1][0]+=nw.g[0][0]-od.g[0][0];
}
}
int main(){
n=re(),m=re();cin>>type;
for(int i=1;i<=n;i++)p[i]=re(),sum+=p[i];
for(int i=1,u,v;i<n;i++){
u=re(),v=re();link(u,v);
}
dfs(1);dfs(1,1);dp(1);bu(1,1,n);
while(m--){
int a=re(),x=re(),b=re(),y=re();
if(!x&&!y&&(fa[a]==b||fa[b]==a)){puts("-1");continue;}
if(x)modify(a,-inf);
else modify(a,inf),sum+=inf;
if(y)modify(b,-inf);
else modify(b,inf),sum+=inf;
Mx ans=query(1,1,n,dfn[1],ed[1]);
Ans=sum-max(ans.g[1][0],ans.g[0][0]);
printf("%lld\n",Ans);
if(x)modify(a,inf);
else modify(a,-inf),sum-=inf;
if(y)modify(b,inf);
else modify(b,-inf),sum-=inf;
}
return 0;
}
[NOIP2018TG]保卫王国的更多相关文章
- noip2018 d2t3 保卫王国 解题报告
保卫王国 电脑卡懒得把题面挪过来了. 朴素 \[ dp_{i,0}=\sum dp_{s,1}\\ dp_{i,1}=\sum \min(dp_{s,0},dp_{s,1})+p_i \] 然后直接动 ...
- LG5024 保卫王国
题意 题目描述 Z 国有\(n\)座城市,\(n - 1\)条双向道路,每条双向道路连接两座城市,且任意两座城市 都能通过若干条道路相互到达. Z 国的国防部长小 Z 要在城市中驻扎军队.驻扎军队需要 ...
- 「NOIP2018」保卫王国
「NOIP2018保卫王国」 题目描述 有一棵 \(n\) 个点, 点有点权 \(a_i\),\(m\) 组询问, 每次求钦点两个节点必须选或者必须不选后的树上最小点覆盖. \(1 \leq n, m ...
- Uoj 441 保卫王国
Uoj 441 保卫王国 动态 \(dp\) .今天才来写这个题. 设 \(f[u][0/1]\) 表示子树 \(u\) 中不选/选 \(u\) 时的最小权值和,显然有:\(f[u][0]=\sum ...
- 竞赛题解 - NOIP2018 保卫王国
\(\mathcal{NOIP2018}\) 保卫王国 - 竞赛题解 按某一个炒鸡dalao名曰 taotao 的话说: \(\ \ \ \ \ \ \ \ \ "一道sb倍增题" ...
- 『保卫王国 树上倍增dp』
保卫王国 Description Z 国有n座城市,n - 1条双向道路,每条双向道路连接两座城市,且任意两座城市 都能通过若干条道路相互到达. Z 国的国防部长小 Z 要在城市中驻扎军队.驻扎军队需 ...
- 【比赛】NOIP2018 保卫王国
DDP模板题 #include<bits/stdc++.h> #define ui unsigned int #define ll long long #define db double ...
- luogu5024 [NOIp2018]保卫王国 (动态dp)
可以直接套动态dp,但因为它询问之间相互独立,所以可以直接倍增记x转移到fa[x]的矩阵 #include<bits/stdc++.h> #define CLR(a,x) memset(a ...
- NOIP2018保卫王国
题目大意:给一颗有点权的树,每次规定两个点选还是不选,求这棵树的最小权点覆盖. 题解 ZZ码农题. 要用动态dp做,这题就是板子,然鹅并不会,留坑代填. 因为没有修改,所以可以静态倍增. 我们先做一遍 ...
随机推荐
- 大型站点技术架构PDF阅读笔记(一):
1.数据库读写分离: 2.系统吞吐量和系统并发数以及系统响应时间之间的关系: 3.系统负载的概念: 4.反向代理的概念: 5.使用缓存来读取数据: 6.利用cookie来记录session: 利用co ...
- JS高程3:表单脚本
HTML和CSS对表单的操作还是比较乏力的,在表单操作中,JS势必会使用到. 基础知识 文本框 选择框 序列化 富文本编辑器 基础知识 HTMLFormElement接口可以创建或者修改<for ...
- iOS9 3DTouch、ShortcutItem、Peek And Pop技术一览
[iOS]iOS9 3DTouch.ShortcutItem.Peek And Pop技术一览 3DTouch UITouch类里API的变化 iOS9中添加的属性 altitudeAngle 当 ...
- ruby之各种概念
一.引言 刚开始接触ruby,遇到问题于是上网查资料,但是有时候却又看不懂,这很大一部分原因是我不知道一些关于ruby的概念名词是什么意思,所以看了别人的回答也理解不了. 二.各种名词 ruby:这个 ...
- Freemarker自定义方法
在项目中有一个需求,每个物品有一个guid,存在数据库中,而在页面上需要显示一个对应的业务数据值,暂且叫做serverId,serverId是通过guid移位计算得来.serverId只需要显示,后台 ...
- C#数组、ArrayList和List<T>
1.数组: 数组在内存中是连续的,索引速度快.赋值与修改简单. 数组的两个数据中间插入数据麻烦,且在声明数组的时候必须指定数组长度.数组长度过长,会浪费内存,过短会造成数据溢出. 2.ArrayLis ...
- hdu3879 Base Station 最大权闭合子图 边权有正有负
/** 题目:hdu3879 Base Station 最大权闭合子图 边权有正有负 链接:http://acm.hdu.edu.cn/showproblem.php?pid=3879 题意:给出n个 ...
- list<> 中find的使用
昨天要在 std::list<std::string> 中判断是否存在某一字符串 std::string . 我首先想到的是 list迭代+std::string重载的"==&q ...
- python 的简单抓取图片
在我们日常上网浏览网页的时候,经常会看到一些好看的图片,我们就希望把这些图片保存下载,或者用户用来做桌面壁纸,或者用来做设计的素材. 我们最常规的做法就是通过鼠标右键,选择另存为.但有些图片鼠标右键的 ...
- java 搭建web项目
从git到maven都是莫名其妙的装上了.... 然后看了下报错,是数据的事,把链接字符串一改,数据库一建,ok,跑起来了 基本上没任何问题,唯一的问题就是我的网速太慢,maven了一夜的样子....