此题场上打了一个正确的$44pts$,接着看错题疯狂$rush$“正确”的$44pts$,后来没$rush$完没将之前的代码$copy$回去,直接变零分了。。。。。

这一题我们显然有一种$O(nm)$的做法

令$f[u][0]$表示在以$u$为根的子树内部署军队,且$u$不部署军队的最小代价。

令$f[u][1]$表示在以$u$为根的子树内部署军队,且$u$部署军队的最小代价。

结合题意(重要!)不难推出:

$f[u][0]=\sum_{v∈son[u]} f[v][1]$

$f[u][1]=val_u+\sum_{v∈son[u]} min(f[v][0],f[v][1])}$

考虑满足能选/不能选的要求,我们只需要临时将被选择点的权值赋值为$INF$或$-INF$即可。

考虑这棵树是一条链的情况:我们用一棵线段树,每个节点维护一个答案矩阵(该区间左端点驻军/可能不驻军,该区间右端点驻军/可能不驻军时的最小代价)。我们求一个节点的答案矩阵,显然可以通过其儿子的答案矩阵,通过大力分类讨论(详见代码)实现更新。

当出现修改权值的情况时,大力改一改然后$pushup$即可。

我们考虑将这一个链上做法扩展到树上。我们用树链剖分将整棵树剖成若干条链,对于轻边上的一对节点$(u,son[u])$,我们将$son[u]$所在链的信息加入节点$u$所对应答案矩阵内即可(详见代码)。

一遍过样例开了$O2$就过了美滋滋

 #include<bits/stdc++.h>
#define mid ((a[x].l+a[x].r)>>1)
#define L long long
#define INF (1LL<<50)
#define LOW (1LL<<40)
#define M 100005
using namespace std; struct edge{int u,next;}e[M*]={}; int head[M]={},use=;
void add(int x,int y){use++;e[use].u=y;e[use].next=head[x];head[x]=use;} int siz[M]={},dfn[M]={},rec[M]={},top[M]={},dn[M]={},son[M]={},fa[M]={},t=; void dfs(int x){
siz[x]=;
for(int i=head[x];i;i=e[i].next) if(e[i].u!=fa[x]){
fa[e[i].u]=x;
dfs(e[i].u);
siz[x]+=siz[e[i].u];
if(siz[son[x]]<siz[e[i].u]) son[x]=e[i].u;
}
}
void dfs(int x,int Top){
top[x]=Top; dfn[x]=++t; rec[t]=x;
if(son[x]) dfs(son[x],Top),dn[x]=dn[son[x]]; else dn[x]=x;
for(int i=head[x];i;i=e[i].next)
if(e[i].u!=fa[x]&&e[i].u!=son[x]) dfs(e[i].u,e[i].u);
} L f[M][]={},val[M]={};
void dp(int x){
f[x][]=val[x];
for(int i=head[x];i;i=e[i].next) if(e[i].u!=fa[x]){
dp(e[i].u);
f[x][]+=f[e[i].u][];
f[x][]+=min(f[e[i].u][],f[e[i].u][]);
}
} struct mat{
L a[][]; mat(){memset(a,,sizeof(a));}
mat(L x){a[][]=a[][]=a[][]=a[][]=x;}
mat(L a1,L a2,L a3,L a4){a[][]=a1; a[][]=a2; a[][]=a3; a[][]=a4;}
L ans(){ return min(min(a[][],a[][]),min(a[][],a[][]));}
void upd(L now){a[][]+=now; a[][]+=now; a[][]+=now;}
friend mat operator *(mat a,mat b){
mat c=INF;
for(int i=;i<;i++)
for(int j=;j<;j++){
for(int i1=;i1<;i1++)
for(int j1=;j1<;j1++)
if(!(i1==&&j1==))
c.a[i][j]=min(c.a[i][j],a.a[i][i1]+b.a[j1][j]);
}
return c;
}
}wei[M];
struct seg{int l,r; mat s;}a[M<<];
void pushup(int x){a[x].s=a[x<<].s*a[x<<|].s;} void build(int x,int l,int r){
a[x].l=l; a[x].r=r;
if(l==r){
int u=rec[l]; L g0=,g1=val[u];
for(int i=head[u];i;i=e[i].next)
if(e[i].u!=fa[u]&&e[i].u!=son[u]){
g0+=f[e[i].u][];
g1+=min(f[e[i].u][],f[e[i].u][]);
}
a[x].s=wei[l]=mat(g1,g1,g1,g0);
return;
}
build(x<<,l,mid); build(x<<|,mid+,r);
pushup(x);
}
mat query(int x,int l,int r){
if(l<=a[x].l&&a[x].r<=r) return a[x].s;
if(r<=mid) return query(x<<,l,r);
if(mid<l) return query(x<<|,l,r);
return query(x<<,l,r)*query(x<<|,l,r);
}
mat query(int x){return query(,dfn[top[x]],dfn[dn[x]]);}
L solve(){mat hh=query(); return hh.ans();} void updata(int x,int k){
if(a[x].l==a[x].r) return void(a[x].s=wei[k]);
if(k<=mid) updata(x<<,k); else updata(x<<|,k);
pushup(x);
}
void Updata(int x,L Val){
L chg=Val-val[x]; val[x]+=chg;
wei[dfn[x]].upd(chg);
while(x){
mat last=query(x);
L lg1=last.ans(),lg0=min(last.a[][],last.a[][]); updata(,dfn[x]); mat now=query(x);
L ng1=now.ans(),ng0=min(now.a[][],now.a[][]); x=fa[top[x]]; if(!x) return;
wei[dfn[x]].upd(ng1-lg1);
wei[dfn[x]].a[][]+=ng0-lg0;
}
} int n,m; char op[]; int main(){
//freopen("defense.in","r",stdin);
//freopen("defense.out","w",stdout);
scanf("%d%d%s",&n,&m,op);
for(int i=;i<=n;i++) scanf("%lld",val+i);
for(int i=,x,y;i<n;i++) scanf("%d%d",&x,&y),add(x,y),add(y,x);
dfs();
dfs(,);
dp();
build(,,n);
//cout<<solve()<<endl;
while(m--){
int A,x,B,y; scanf("%d%d%d%d",&A,&x,&B,&y);
L lastA=val[A],lastB=val[B],chg=; if(x==) Updata(A,INF);
else Updata(A,-INF); if(y==) Updata(B,INF);
else Updata(B,-INF); L res=solve();
if(x) res+=INF+lastA;
if(y) res+=INF+lastB; Updata(A,lastA);
Updata(B,lastB); if(res>LOW){printf("-1\n"); continue;}
printf("%lld\n",res);
}
}

【NOIP2018】保卫王国 动态dp的更多相关文章

  1. luogu5024 [NOIp2018]保卫王国 (动态dp)

    可以直接套动态dp,但因为它询问之间相互独立,所以可以直接倍增记x转移到fa[x]的矩阵 #include<bits/stdc++.h> #define CLR(a,x) memset(a ...

  2. BZOJ 5466: [Noip2018]保卫王国 动态DP

    Code: // luogu-judger-enable-o2 #include<bits/stdc++.h> #define ll long long #define lson (now ...

  3. JZOJ5966. [NOIP2018TGD2T3] 保卫王国 (动态DP做法)

    题目大意 这还不是人尽皆知? 有一棵树, 每个节点放军队的代价是\(a_i\), 一条边连接的两个点至少有一个要放军队, 还有\(q\)次询问, 每次规定其中的两个一定需要/不可放置军队, 问这样修改 ...

  4. luoguP5024 保卫王国 动态dp

    题目大意: emmmmm 题解: QAQ #include <cstdio> #include <cstring> #include <iostream> usin ...

  5. LuoguP5024 保卫王国(动态DP,LCT)

    最小权覆盖集 = 全集 - 最大权独立集 强制取点.不取点可以使用把权值改成正无穷或负无穷实现 接下来就是经典的"动态最大权独立集"了 O(nlogn). 这不是我说的,是immo ...

  6. P5024 保卫王国(动态dp/整体dp/倍增dp)

    做法(倍增) 最好写的一种 以下0为不选,1为选 \(f_{i,0/1}\)为\(i\)子树的最小值,\(g_{i,0/1}\)为除i子树外的最小值 \(fh_{i,j,0/1,0/1}\)为确定\( ...

  7. 竞赛题解 - NOIP2018 保卫王国

    \(\mathcal{NOIP2018}\) 保卫王国 - 竞赛题解 按某一个炒鸡dalao名曰 taotao 的话说: \(\ \ \ \ \ \ \ \ \ "一道sb倍增题" ...

  8. [NOIP2018]保卫王国(树形dp+倍增)

    我的倍增解法吊打动态 \(dp\) 全局平衡二叉树没学过 先讲 \(NOIP\) 范围内的倍增解法. 我们先考虑只有一个点取/不取怎么做. \(f[x][0/1]\) 表示取/不取 \(x\) 后,\ ...

  9. NOIP2018保卫王国

    题目大意:给一颗有点权的树,每次规定两个点选还是不选,求这棵树的最小权点覆盖. 题解 ZZ码农题. 要用动态dp做,这题就是板子,然鹅并不会,留坑代填. 因为没有修改,所以可以静态倍增. 我们先做一遍 ...

随机推荐

  1. Android窗口背景的优化

    视图有背景,每个窗口也是有背景的.每一Activity是一个窗口,每一个Activity都有不同得背景.界面的绘画顺序如下:窗口——跟视图 ——子视图.当我们的跟视图已经覆盖了整个窗口的时候 ,程序还 ...

  2. =default(c++11)

    1.概念 1)如果我们需要编译器默认的行为,则可以在参数列表后面加上=default来显式地要求编译器生成合成版本的默认构造函数和拷贝控制成员:合成的默认构造函数.合成拷贝构造函数.合成拷贝赋值运算符 ...

  3. Strut2的配置文件strust.xml报错:Package Action extends undefined package struts-default

    struts.xml的警告信息,是需要联网验证dtd是否符合规范,只要配置成本地的dtd就会消失, 配置方式请看: 详细请看http://www.cnblogs.com/liuyangfirst/p/ ...

  4. mac下svn无法上传.a文件的问题

    Xcode自带的svn和Versions以及一些其它工具都默认ignore".a"文件. 解决办法有两个: 方法一:使用命令行添加文件([转]原文在这) 1.打开终端,输入cd,空 ...

  5. HDU2212 DFS 2016-07-24 13:52 56人阅读 评论(0) 收藏

    DFS Problem Description A DFS(digital factorial sum) number is found by summing the factorial of eve ...

  6. chrome 概述

    Chrome, Firebug, Filddle 调试:http://www.zhihu.com/question/35667558/answer/63936233 Chrome开发者工具系列: ht ...

  7. 『IOS』 遇到问题记录(长期更新)

    遇到的很多问题,解决后都是自己记着,以为不会忘记,之后却会想不起来了. 所以把今后解决的问题记录在这. 一. 在二级页面设置了CAlayer的代理,在返回一级页面报错: EXC_BAD_ACCESS( ...

  8. Java IO流详尽解析(转)

    流的概念和作用 学习Java IO,不得不提到的就是JavaIO流. 流是一组有顺序的,有起点和终点的字节集合,是对数据传输的总称或抽象.即数据在两设备间的传输称为流,流的本质是数据传输,根据数据传输 ...

  9. cocos2dx lua 图片去色shader

    function FreeCardTool:setNodeGray(node) local vertShaderByteArray = "\n".. "attribute ...

  10. 一、配置etcd数据库

      etcd服务作为Kubernetes集群的主数据库,在安装Kubernetes各服务之前需要首先安装和启动. 1. 安装etcd yum -y install etcd 2. 修改etcd配置文件 ...