【NOIP2018】保卫王国 动态dp
此题场上打了一个正确的$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的更多相关文章
- luogu5024 [NOIp2018]保卫王国 (动态dp)
可以直接套动态dp,但因为它询问之间相互独立,所以可以直接倍增记x转移到fa[x]的矩阵 #include<bits/stdc++.h> #define CLR(a,x) memset(a ...
- BZOJ 5466: [Noip2018]保卫王国 动态DP
Code: // luogu-judger-enable-o2 #include<bits/stdc++.h> #define ll long long #define lson (now ...
- JZOJ5966. [NOIP2018TGD2T3] 保卫王国 (动态DP做法)
题目大意 这还不是人尽皆知? 有一棵树, 每个节点放军队的代价是\(a_i\), 一条边连接的两个点至少有一个要放军队, 还有\(q\)次询问, 每次规定其中的两个一定需要/不可放置军队, 问这样修改 ...
- luoguP5024 保卫王国 动态dp
题目大意: emmmmm 题解: QAQ #include <cstdio> #include <cstring> #include <iostream> usin ...
- LuoguP5024 保卫王国(动态DP,LCT)
最小权覆盖集 = 全集 - 最大权独立集 强制取点.不取点可以使用把权值改成正无穷或负无穷实现 接下来就是经典的"动态最大权独立集"了 O(nlogn). 这不是我说的,是immo ...
- P5024 保卫王国(动态dp/整体dp/倍增dp)
做法(倍增) 最好写的一种 以下0为不选,1为选 \(f_{i,0/1}\)为\(i\)子树的最小值,\(g_{i,0/1}\)为除i子树外的最小值 \(fh_{i,j,0/1,0/1}\)为确定\( ...
- 竞赛题解 - NOIP2018 保卫王国
\(\mathcal{NOIP2018}\) 保卫王国 - 竞赛题解 按某一个炒鸡dalao名曰 taotao 的话说: \(\ \ \ \ \ \ \ \ \ "一道sb倍增题" ...
- [NOIP2018]保卫王国(树形dp+倍增)
我的倍增解法吊打动态 \(dp\) 全局平衡二叉树没学过 先讲 \(NOIP\) 范围内的倍增解法. 我们先考虑只有一个点取/不取怎么做. \(f[x][0/1]\) 表示取/不取 \(x\) 后,\ ...
- NOIP2018保卫王国
题目大意:给一颗有点权的树,每次规定两个点选还是不选,求这棵树的最小权点覆盖. 题解 ZZ码农题. 要用动态dp做,这题就是板子,然鹅并不会,留坑代填. 因为没有修改,所以可以静态倍增. 我们先做一遍 ...
随机推荐
- Java中的内存划分
Java程序在运行时,需要在内存中分配空间.为了提高运行效率,就对数据进行了不同的空间划分.因为每一片区域都有特定的数据处理方式和内存管理方式. 具体分为5种内存空间: 程序计数器:保证线程切换后能恢 ...
- 如何查看路由器的mac和计算机的mac
如何查看路由器的mac和计算机的mac 一.查看路由器的mac 方法一: 直接看路由器的背面,如下图,即可看到MAC地址 打开命令提示符窗口,输入ipconfig,找到网关地址,如192.168. ...
- 信息管理代码分析<二>读取二进制文件数据
first和end做为全局变量,分别指向链表的头和尾.建立链表的方式也比较简易,从二进制文件数据块中,依次从头到尾读取,每读取一个就建立一个结点. /*基本模型*/ EMP *emp1; while( ...
- (并查集 添加关系)How Many Answers Are Wrong --Hdu --3038
链接: http://acm.hdu.edu.cn/showproblem.php?pid=3038 http://acm.hust.edu.cn/vjudge/contest/view.action ...
- POJ2653判断线段相交
POJ2653 题目大意:按顺序放木棒,问最后所有的木棒中上面没有木棒的木棒的索引是…… 思路:按理说线段相交的题目做的听多了,这个应该不算新鲜,但是这个题,还是让我学到了认真读题,面对这个题很容易想 ...
- SoapUI5.1.2安装和破解教程
一.SoapUI简介 soapui提供一个工具通过soap/http来检查,调用,实现web service和web service的功能/负载/符合性测试. 该工具既可作为一个桌面应用软件使用,也可 ...
- .NET Core 运行时标识符 (RID) 目录
RID 是什么? RID 是运行时标识符的缩写. RID 用于标识其中将运行应用程序或资产(即程序集)的目标操作系统. 其外观类似如下:“ubuntu.14.04-x64”.“win7-x64”.“o ...
- lnmp下thinkphp 500错误指南
先在php.ini打开报错,display_errors: on: 如果是open_basedir的问题,修改nginx的配置文件fastcgi.conf 将fastcgi_param PHP_ADM ...
- HTML5 canvas 学习
一.canvas简介 <canvas> 是 HTML5 新增的,一个可以使用脚本(通常为JavaScript)在其中绘制图像的 HTML 元素.它可以用来制作照片集或者制作简单(也不是 ...
- C# DataTable导出EXCEL后身份证、银行卡号等长数字信息显示乱码解决
在DataTable导出EXCEL后发现有些格式显示有问题,比如身份证.银行卡号等大于11位的数字显示为科学计数法.13681-1等 带中划线的两段数字显示为日期格式等. 处理方法如下: public ...