做法(倍增)

最好写的一种

以下0为不选,1为选

\(f_{i,0/1}\)为\(i\)子树的最小值,\(g_{i,0/1}\)为除i子树外的最小值

\(fh_{i,j,0/1,0/1}\)为确定\(i\)与\(i\)的\(2^j\)级祖先的状态,\(i\)的\(2^j\)祖先不包括i子树的最小值

这个转移挺好想的,故不赘述

  • 查询

    考虑边倍增爬上去,边处理即可
#include<bits/stdc++.h>
typedef long long LL;
const LL maxn=1e6+9,inf=0x3f3f3f3f3f3f3f3f;
inline LL Read(){
LL x(0),f(1); char c=getchar();
while(c<'0' || c>'9'){
if(c=='-') f=-1; c=getchar();
}
while(c>='0' && c<='9'){
x=(x<<3)+(x<<1)+c-'0'; c=getchar();
}return x*f;
}
struct node{
LL to,nxt;
}dis[maxn<<1];
LL n,q,num;
LL head[maxn],f[maxn][2],inc[maxn][21],g[maxn][2],fh[maxn][21][2][2],dep[maxn],p[maxn];
char s[3];
inline void Add(LL u,LL v){
dis[++num]=(node){v,head[u]}; head[u]=num;
}
void Dfs1(LL u,LL fa){
inc[u][0]=fa; dep[u]=dep[fa]+1; f[u][1]=p[u];
for(LL i=1;i<=20;++i) inc[u][i]=inc[inc[u][i-1]][i-1];
for(LL i=head[u];i;i=dis[i].nxt){
LL v(dis[i].to); if(v==fa) continue;
Dfs1(v,u);
f[u][0]+=f[v][1]; f[u][1]+=std::min(f[v][0],f[v][1]);
}
}
void Dfs2(LL u,LL fa){
fh[u][0][0][1]=fh[u][0][1][1]=f[fa][1]-std::min(f[u][0],f[u][1]);
fh[u][0][0][0]=inf; fh[u][0][1][0]=f[fa][0]-f[u][1];
for(LL i=1;i<=20;++i)
for(LL x=0;x<=1;++x)
for(LL y=0;y<=1;++y){
fh[u][i][x][y]=inf;
for(LL xx=0;xx<=1;++xx)
fh[u][i][x][y]=std::min(fh[u][i][x][y],fh[inc[u][i-1]][i-1][xx][y]+fh[u][i-1][x][xx]);
}
for(LL i=head[u];i;i=dis[i].nxt){
LL v(dis[i].to); if(v==fa) continue;
g[v][0]=g[u][1]+f[u][1]-std::min(f[v][0],f[v][1]);
g[v][1]=std::min(g[u][0]+f[u][0]-f[v][1],g[v][0]);
Dfs2(v,u);
}
}
inline LL Solve(LL a,LL x,LL b,LL y){
if(dep[a]<dep[b]) std::swap(a,b),std::swap(x,y);
LL aa[2]={inf,inf},ab[2]={inf,inf};
LL ta[2]={inf,inf},tb[2]={inf,inf};
aa[x]=f[a][x]; ab[y]=f[b][y];
for(LL i=20;i>=0;--i){
if(dep[inc[a][i]]>=dep[b]){
for(LL ii=0;ii<=1;++ii){
ta[ii]=inf;
for(LL jj=0;jj<=1;++jj)
ta[ii]=std::min(ta[ii],aa[jj]+fh[a][i][jj][ii]);
}
a=inc[a][i]; aa[1]=ta[1],aa[0]=ta[0];
}
}
if(a==b) return aa[y]+g[a][y];
for(LL i=20;i>=0;--i){
if(inc[a][i]!=inc[b][i]){
for(LL ii=0;ii<=1;++ii){
ta[ii]=tb[ii]=inf;
for(LL jj=0;jj<=1;++jj){
ta[ii]=std::min(ta[ii],aa[jj]+fh[a][i][jj][ii]);
tb[ii]=std::min(tb[ii],ab[jj]+fh[b][i][jj][ii]);
}
}
a=inc[a][i], b=inc[b][i]; aa[1]=ta[1],aa[0]=ta[0]; ab[1]=tb[1],ab[0]=tb[0];
}
}
LL fa=inc[a][0];
LL ans0(aa[1]+ab[1]+(g[fa][0]+f[fa][0]-f[a][1]-f[b][1]));
LL ans1(g[fa][1]+f[fa][1]-std::min(f[a][1],f[a][0])-std::min(f[b][1],f[b][0])+std::min(aa[1],aa[0])+std::min(ab[1],ab[0]));
return std::min(ans0,ans1);
}
int main(){
// freopen("testdata.in","r",stdin);
n=Read(); q=Read(); scanf(" %s",s+1);
for(LL i=1;i<=n;++i) p[i]=Read();
for(LL i=1;i<n;++i){
LL u(Read()),v(Read()); Add(u,v); Add(v,u);
}
Dfs1(1,0); Dfs2(1,0);
while(q--){
LL a(Read()),x(Read()),b(Read()),y(Read());
LL ret(Solve(a,x,b,y));
if(ret>=inf) puts("-1");
else printf("%lld\n",ret);
}
return 0;
}

整体

思维较大,代码细节有点多

f与g与方法一同理得出

考虑类似tarjan离线的处理方式,用并查集,并查集按秩压缩

\(fa_x\)为x暂时在并查集里的父亲,\(h_{x,0/1,0/1}\)表示确定x父亲与x的状态,x父亲这棵子树的最小值,状态转移压缩时处理

当我们得出(x,y)的祖先时,把查询再丢到祖先u去

  • 查询

    对于u存在于(x,y)的情况单独讨论,因为这样u的状态唯一

    否则枚举u状态,\(min(g[u][0]+h[a][0][x]+h[b][0][y]-f[u][0],g[u][1]+h[a][1][x]+h[b][1][y]-f[u][1])\)
#include<bits/stdc++.h>
typedef long long LL;
const LL maxn=1e6+9,inf=0x3f3f3f3f3f3f3f3f;
inline LL Read(){
LL x(0),f(1); char c=getchar();
while(c<'0' || c>'9'){
if(c=='-') f=-1; c=getchar();
}
while(c>='0' && c<='9'){
x=(x<<3)+(x<<1)+c-'0'; c=getchar();
}return x*f;
}
struct node{
LL to,nxt;
}dis[maxn<<1];
struct qy{
LL a,x,b,y,id;
};
LL n,q,num;
LL head[maxn],f[maxn][2],g[maxn][2],h[maxn][2][2],dep[maxn],p[maxn],fa[maxn],nx[2][2],ans[maxn];
std::vector<qy> Qy[maxn],Ans[maxn];
char s[3];
inline void Add(LL u,LL v){
dis[++num]=(node){v,head[u]}; head[u]=num;
}
void Dfs1(LL u,LL ff){
f[u][1]=p[u];
for(LL i=head[u];i;i=dis[i].nxt){
LL v(dis[i].to); if(v==ff) continue;
Dfs1(v,u);
f[u][0]+=f[v][1]; f[u][1]+=std::min(f[v][0],f[v][1]);
}
}
void Dfs2(LL u,LL ff){
for(LL i=head[u];i;i=dis[i].nxt){
LL v(dis[i].to); if(v==ff) continue;
g[v][0]=g[u][1]+f[u][1]-std::min(f[v][0],f[v][1]);
g[v][1]=std::min(g[u][0]+f[u][0]-f[v][1],g[v][0]);
Dfs2(v,u);
}
}
inline void Union(LL x,LL y){
fa[x]=y; LL tmp(f[y][1]-std::min(f[x][0],f[x][1]));
h[x][0][0]=inf; h[x][0][1]=f[y][0]; h[x][1][1]=tmp+f[x][1]; h[x][1][0]=tmp+f[x][0];
}
void Find(LL x){
if(fa[x]==fa[fa[x]]) return;
Find(fa[x]);
LL up(fa[x]);
for(LL i=0;i<=1;++i)
for(LL j=0;j<=1;++j){
nx[i][j]=h[x][i][j]; h[x][i][j]=inf;
}
for(LL i=0;i<=1;++i)
for(LL j=0;j<=1;++j)
for(LL k=0;k<=1;++k)
h[x][i][j]=std::min(h[x][i][j],nx[k][j]+h[up][i][k]-f[up][k]);
fa[x]=fa[fa[x]];
}
void Dfs3(LL u,LL ff){
for(LL i=head[u];i;i=dis[i].nxt){
LL v(dis[i].to); if(v==ff) continue;
Dfs3(v,u);
Union(v,u);
}
for(LL i=0;i<Qy[u].size();++i){
LL a(Qy[u][i].a),b(Qy[u][i].b);
if(a==u) a=b;
Find(a);
if(a!=fa[a]){
Ans[fa[a]].push_back(Qy[u][i]);
}
}
for(LL i=0;i<Ans[u].size();++i){
LL a(Ans[u][i].a),b(Ans[u][i].b),x(Ans[u][i].x),y(Ans[u][i].y),id(Ans[u][i].id);
Find(a); Find(b);
if(u!=a && u!=b) ans[id]=std::min(g[u][0]+h[a][0][x]+h[b][0][y]-f[u][0],g[u][1]+h[a][1][x]+h[b][1][y]-f[u][1]);
else{
if(u!=a) std::swap(a,b),std::swap(x,y);
ans[id]=g[a][x]+h[b][x][y];
}
}
}
int main(){
// freopen("testdata.in","r",stdin);
n=Read(); q=Read(); scanf(" %s",s+1);
for(LL i=1;i<=n;++i) p[i]=Read();
for(LL i=1;i<n;++i){
LL u(Read()),v(Read()); Add(u,v); Add(v,u);
}
Dfs1(1,0); Dfs2(1,0);
for(LL i=1;i<=q;++i){
LL a(Read()),x(Read()),b(Read()),y(Read());
Qy[a].push_back((qy){a,x,b,y,i}); Qy[b].push_back((qy){a,x,b,y,i});
}
for(LL i=1;i<=n;++i) fa[i]=i;
Dfs3(1,0);
for(LL i=1;i<=q;++i){
if(ans[i]>=inf) puts("-1");
else printf("%lld\n",ans[i]);
}
return 0;
}

P5024 保卫王国(动态dp/整体dp/倍增dp)的更多相关文章

  1. P5024 保卫王国[倍增+dp]

    窝当然不会ddp啦,要写这题当然是考虑优化裸dp啦,但是这题非常麻烦,于是变成了黑题. 首先,这个是没有上司的舞会模型,求图的带权最大独立集. 不考虑国王的限制条件,有 \[ dp[x][0]+=dp ...

  2. [倍增][换根DP]luogu P5024 保卫王国

    题面 https://www.luogu.com.cn/problem/P5024 分析 可以对有限制的点对之间的链进行在倍增上的DP数组合并. 需要通过一次正向树形DP和一次换根DP得到g[0][i ...

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

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

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

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

  5. 【NOIP2018】保卫王国 动态dp

    此题场上打了一个正确的$44pts$,接着看错题疯狂$rush$“正确”的$44pts$,后来没$rush$完没将之前的代码$copy$回去,直接变零分了..... 这一题我们显然有一种$O(nm)$ ...

  6. luoguP5024 保卫王国 动态dp

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

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

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

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

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

  9. 【洛谷】P5024 保卫王国 (倍增)

    前言 传送门 很多人写了题解了,我就懒得写了,推荐一篇博客 那就分享一下我的理解吧(说得好像有人看一样 对于每个点都只有选与不选两种情况,所以直接用倍增预处理出来两种情况的子树之内,子树之外的最值,最 ...

随机推荐

  1. 深入理解JVM(六) -- GC执行原则和方案

    上篇文章中,我们了解了Java虚拟机垃圾回收的思路和策略,这篇文章我们将了解Java是如何实现高效的回收算法的. 我们需要了解,内存回收必须要保证“一致性”,意思就是在执行GC分析的时候,系统看起来要 ...

  2. VBA编程图表(二十一)

    使用VBA,可以根据特定标准生成图表.下面通过一个例子来看看它如何实现. 第1步 - 输入要生成图形的数据. 第2步 - 创建3个按钮 - 一个生成条形图,另一个生成饼图,另一个生成柱形图. 第3步 ...

  3. HTML知识整理

    以下是自己对以前所学的部分HTML相关知识进行的简单的梳理,水平有限,若有问题的地方,还请见谅. 1. 常用的浏览器及浏览器内核分别是什么? IE:Trident 内核 Firefox:gecko 内 ...

  4. django路由系统及分发路由的本质

    路由系统 当我们启动一个django项目后,想要通过浏览器访问到django项目中的资源 就需要在django中的urls项目中进行路由配置 urlpatterns = [ url(r'^admin/ ...

  5. 获取项目中所有URL--获取swagger上展示的接口信息

    有时我们需要接口的一些基本信息,比如接口请求路径,接口请求方式等,我们用这些信息来做判断,或者入库. 我在开发接口权限的时候就遇到了这个问题,之前写的接口很多,现在需要将这些接口信息存到数据库中, 用 ...

  6. 【spark】spark应用(分布式估算圆周率+基于Spark MLlib的贷款风险预测)

    注:本章不涉及spark和scala原理的探讨,详情见其他随笔 一.分布式估算圆周率 计算原理:假设正方形的面积S等于x²,而正方形的内切圆的面积C等于Pi×(x/2)²,因此圆面积与正方形面积之比C ...

  7. Android笔记(五十五) Android四大组件之一——ContentProvider,使用系统提供的ContentProvider

    因为在Android中,存储系统联系人姓名和电话是存在与不同的ContentProvider中的,具体如何查找,可以从Android的源代码中查看,在android.providers包中列出了所有系 ...

  8. [算法竞赛][2018][蓝桥杯][LanqiaoCA]第九届蓝桥杯A组

    题目1 标题:分数 1/1 + 1/2 + 1/4 + 1/8 + 1/16 + .... 每项是前一项的一半,如果一共有20项,求这个和是多少,结果用分数表示出来.类似:3/2当然,这只是加了前2项 ...

  9. 每个程序员都应该知道延迟数—Latency Numbers Every Programmer Should Know

    每个程序员都应该知道延迟数 Latency Numbers Every Programmer Should Know https://people.eecs.berkeley.edu/~rcs/res ...

  10. suse12安装详解

    1.部署步骤 1.1.启动安装程序 在启动页面上选择Installation,然后按Enter键,这将载入SUSE Linux服务器安装程序并以普通模式安装. 1.2.选择安装语言 Language和 ...