[学习笔记]整体DP
问题:
有一些问题,通常见于二维的DP,另一维记录当前x的信息,但是这一维过大无法开下,O(nm)也无法通过。
但是如果发现,对于x,在第二维的一些区间内,取值都是相同的,并且这样的区间是有限个,就可以批量处理。
思想:
通过动态开点线段树维护第二维,
如果某个节点没有儿子,那么这个节点区间都是同一个权值。
也即,一个节点是空节点,那么这个节点所有的值和父亲的值都一致。(其实它的兄弟也是空节点的)
对于序列的问题,
可以直接扫过去,修改某些位置的点。
或者线段树合并。
对于树上的问题,
线段树合并。
实现:
主要考虑什么时候线段树合并停止。以及pushdown的标记问题。
当x都没有儿子或者y都没有儿子时候,整个x的区间或整个y的区间都是同一个值,可以直接计算贡献转移过来(这个必须支持,否则不能整体DP)。
否则,pushdown,进行递归
pushdown时候建立新的儿子(如果之前没有)。
空间复杂度和时间复杂度基本一致。O(nlogn)
只要满足,在x都没有儿子或者y都没有儿子时候,可以快速合并然后return,那么就可以整体DP了。
例题2:
【PKUSC 2019】D2T1 树染色
$dp[x][c]=\Pi (sumy-dp[y][c])$sumy表示y的所有dp[y][*]的和
在x都没有儿子或者y都没有儿子时候,我们要么知道每个x的值,要么知道每个y的值。
在x都没有儿子时候,把y的节点内每个数乘-1再加sumy,再乘上x区间的值。
y都没有儿子时候,直接用(sumy-val)乘给x即可。
code:
#include<bits/stdc++.h>
#define reg register int
#define il inline
#define fi first
#define se second
#define mk(a,b) make_pair(a,b)
#define numb (ch^'0')
#define pb push_back
#define solid const auto &
#define enter cout<<endl
#define pii pair<int,int>
using namespace std;
typedef long long ll;
template<class T>il void rd(T &x){
char ch;x=;bool fl=false;while(!isdigit(ch=getchar()))(ch=='-')&&(fl=true);
for(x=numb;isdigit(ch=getchar());x=x*+numb);(fl==true)&&(x=-x);}
template<class T>il void output(T x){if(x/)output(x/);putchar(x%+'');}
template<class T>il void ot(T x){if(x<) putchar('-'),x=-x;output(x);putchar(' ');}
template<class T>il void prt(T a[],int st,int nd){for(reg i=st;i<=nd;++i) ot(a[i]);putchar('\n');}
namespace Modulo{
const int mod=;
int ad(int x,int y){return (x+y)>=mod?x+y-mod:x+y;}
void inc(int &x,int y){x=ad(x,y);}
int mul(int x,int y){return (ll)x*y%mod;}
void inc2(int &x,int y){x=mul(x,y);}
int qm(int x,int y=mod-){int ret=;while(y){if(y&) ret=mul(x,ret);x=mul(x,x);y>>=;}return ret;}
}
using namespace Modulo;
namespace Miracle{
const int N=2e5+;
int n,m,k;
struct node{
int nxt,to;
}e[*N];
int hd[N],cnt;
void add(int x,int y){
e[++cnt].nxt=hd[x];
e[cnt].to=y;
hd[x]=cnt;
}
#define mid ((l+r)>>1)
struct tr{
int sum,mul,ad;
int ls,rs,val;
void op(){
cout<<"SUM "<<sum<<" MUL "<<mul<<" AD "<<ad<<endl;
}
}t[+];
int tot,S;
vector<int>no[N];
int rt[N];
int nc(){
++tot;
t[tot].sum=;t[tot].mul=;t[tot].ad=;
t[tot].ls=t[tot].rs=;t[tot].val=;
return tot;
}
void tag(int x,int l,int r,int ml,int aa){
// cout<<" tag "<<x<<" l "<<l<<" r "<<r<<" ml "<<ml<<" ad "<<aa<<endl;
// t[x].op();
t[x].sum=mul(t[x].sum,ml);
t[x].sum=ad(t[x].sum,mul(r-l+,aa));
t[x].val=ad(mul(t[x].val,ml),aa);
t[x].mul=mul(t[x].mul,ml);
t[x].ad=ad(mul(t[x].ad,ml),aa);
}
void pushup(int x){
t[x].sum=ad(t[t[x].ls].sum,t[t[x].rs].sum);
}
void pushdown(int x,int l,int r){
if(!t[x].ls) t[x].ls=nc();
if(!t[x].rs) t[x].rs=nc();
tag(t[x].ls,l,mid,t[x].mul,t[x].ad);
tag(t[x].rs,mid+,r,t[x].mul,t[x].ad);
t[x].mul=;t[x].ad=;
}
void upda(int &x,int l,int r,int p){
// cout<<" pp "<<p<<" x "<<x<<" l "<<l<<" r "<<r<<" sm "<<t[x].sum<<" mul "<<t[x].mul<<" ad "<<t[x].ad<<endl;
// cout<<" ls "<<t[x].ls<<" rs "<<t[x].rs<<endl;
if(!x) x=nc();
if(l==r){
// cout<<" ss "<<t[x].sum<<endl;
t[x].sum=;
t[x].val=;
return;
}
pushdown(x,l,r);
if(p<=mid) upda(t[x].ls,l,mid,p);
else upda(t[x].rs,mid+,r,p);
pushup(x);
}
int merge(int x,int y,int l,int r){
if(!t[x].ls&&!t[x].rs){
swap(x,y);
int v=t[y].val;
tag(x,l,r,mod-,S);
tag(x,l,r,v,);
}else if(!t[y].ls&&!t[y].rs){
int v=t[y].val;
tag(x,l,r,ad(S,mod-v),);
}else{
pushdown(x,l,r);pushdown(y,l,r);
t[x].ls=merge(t[x].ls,t[y].ls,l,mid);
t[x].rs=merge(t[x].rs,t[y].rs,mid+,r);
pushup(x);
}
return x;//warining!!
}
void dfs(int x,int fa){
rt[x]=nc();
tag(rt[x],,m,,);
for(reg i=hd[x];i;i=e[i].nxt){
int y=e[i].to;
if(y==fa) continue;
dfs(y,x);
S=t[rt[y]].sum;
rt[x]=merge(rt[x],rt[y],,m);
// cout<<y<<" back "<<x<<" : "<<" sum "<<t[rt[x]].sum<<endl;
}
for(solid c:no[x]){
upda(rt[x],,m,c);
}
// cout<<x<<" : "<<" sum "<<t[rt[x]].sum<<endl;
}
int main(){
rd(n);rd(m);rd(k);
int x,y;
for(reg i=;i<n;++i){
rd(x);rd(y);
add(x,y);add(y,x);
}
for(reg i=;i<=k;++i){
rd(x);rd(y);
no[x].push_back(y);
}
dfs(,);
printf("%d",t[rt[]].sum);
return ;
} }
signed main(){
Miracle::main();
return ;
} /*
Author: *Miracle*
*/
[学习笔记]整体DP的更多相关文章
- [学习笔记] 数位DP的dfs写法
跟着洛谷日报走,算法习题全都有! 嗯,没错,这次我也是看了洛谷日报的第84期才学会这种算法的,也感谢Mathison大佬,素不相识,却写了一长篇文章来帮助我学习这个算法. 算法思路: 感觉dfs版的数 ...
- [学习笔记]区间dp
区间 \(dp\) 1.[HAOI2008]玩具取名 \(f[l][r][W/I/N/G]\) 表示区间 \([l,r]\) 中能否压缩成 \(W/I/N/G\) \(Code\ Below:\) # ...
- [学习笔记]树形dp
最近几天学了一下树形\(dp\) 其实早就学过了 来提高一下打开树形\(dp\)的姿势. 1.没有上司的晚会 我的人生第一道树形\(dp\),其实就是两种情况: \(dp[i][1]\)表示第i个人来 ...
- [学习笔记]插头dp
基于连通性的状压dp 巧妙之处:插头已经可以表示内部所有状态了. 就是讨论麻烦一些. 简介 转移方法:逐格转移,分类讨论 记录状态方法:最小表示法(每次要重新编号,对于一类没用“回路路径”之类的题,可 ...
- 【学习笔记】dp基础
知识储备:dp入门. 好了,完成了dp入门,我们可以做一些稍微不是那么裸的题了. dp基础,主要是做题,只有练习才能彻底掌握. 洛谷P1417 烹调方案 分析:由于时间的先后会对结果有影响,所以c[i ...
- 【学习笔记】dp入门
知识点 动态规划(简称dp),可以说是各种程序设计中遇到的第一个坎吧,这篇博文是我对dp的一点点理解,希望可以帮助更多人dp入门. 先看看这段话 动态规划(dynamic programming) ...
- [学习笔记]动态dp
其实就过了模板. 感觉就是带修改的dp [模板]动态dp 给定一棵n个点的树,点带点权. 有m次操作,每次操作给定x,y表示修改点x的权值为y. 你需要在每次操作之后求出这棵树的最大权独立集的权值大小 ...
- [BZOJ4011][HNOI2015] 落忆枫音(学习笔记) - 拓扑+DP
其实就是贴一下防止自己忘了,毕竟看了题解才做出来 Orz PoPoQQQ 原文链接 Description 背景太长了 给定一个DAG,和一对点(x, y), 在DAG中由x到y连一条有向边,求生成树 ...
- webpack学习笔记--整体配置结构
之前的章节分别讲述了每个配置项的具体含义,但没有描述它们所处的位置和数据结构,下面通过一份代码来描述清楚: const path = require('path'); module.exports = ...
随机推荐
- day36 03-Hibernate检索方式:排序、参数绑定、投影查询
排序之后是分页查询. 检索单个对象 还可以进行参数的绑定. HQL的参数绑定,按参数名称绑定或者是按参数位置绑定. 还可以用投影的操作,投影的操作是只查询这里面的某几个属性.只查询某一个属性,查询多个 ...
- ubuntu和win10设置双显示器
ubuntu:最右上角那个图标,点开找到系统设置,系统设置中找到“显示”中,在其中可以调节双屏显示或者只显示一个屏,图等会补... win10:现在是ubuntu系统所以操作忘记了写不出来,等下换系统 ...
- 模拟7题解 T1方程的解
方程的解 [扩展欧几里德] 首先进行特判,两个小时基本想到了,除了a!=0,b==0,a*c<0这种情况 其次就是一般情况: 首先exgcd求出ax+by=GCD(a,b)的一组任意解 然后两边 ...
- 导出excel表
<?phppublic function export(){ #提现状态 $status = isset($_REQUEST['status'])? ...
- PHP的注释规范
<?php //注释规范 /** *函数的功能 *@param 参数类型 参数名1 参数解析 *@param 参数类型 参数名2 参数解析 *@return 返回值类型 返回值解析 *@auth ...
- (转)Cookie存中文乱码的问题
有个奇怪的问题:登录页面中使用Cookie存值,Cookie中要存中文汉字.代码在本地调试,一切OK,汉字也能顺利存到Cookie和从Cookie中读出,但是放到服务器上不管用了,好好的汉字成了乱码, ...
- 【机器学习PAI实战】—— 玩转人工智能之商品价格预测
摘要: 我们经常思考机器学习,深度学习,以至于人工智能给我们带来什么?在数据相对充足,足够真实的情况下,好的学习模型可以发现事件本身的内在规则,内在联系.我们去除冗余的信息,可以通过最少的特征构建最简 ...
- Spring_通过Bean的Factory配置Bean
package com.tanlei.bean.FactoryBean; import org.springframework.beans.factory.FactoryBean; public cl ...
- 一.数据结构&算法的引言+时间复杂度
目录(contents): 1.什么是计算机科学?什么是算法? 2.如何形象化的理解算法? 3.什么是算法分析? 4.时间复杂度 5.数据结构 6.总结算法和数据结构之间的关联 一.什么是计算机科学? ...
- FastAdmin 自学教程 - 目录(持续更新)(2019-10-11)
FastAdmin 自学教程 - 目录 本自学教程将不定期更新. 了解 FastAdmin FastAdmin 开发第 1 天:了解 FastAdmin 框架 FastAdmin 开发第 2 天:安装 ...