首页 > 其他分享 >P7219 [JOISC2020] 星座 3【笛卡尔树+整体dp】

P7219 [JOISC2020] 星座 3【笛卡尔树+整体dp】

时间:2024-04-02 20:24:31浏览次数:29  
标签:p2 p1 rs int JOISC2020 P7219 ls dp

P7219 [JOISC2020] 星座 3

笛卡尔树+整体 dp(线段树合并维护 dp)

考虑转化题意,不存在矩形为星座,即对于每个极大矩形中都只有一颗星星。而观察题目的方格,对于两个位置是否能够成为一个矩形,只和两个位置的区间最大值(小白船的位置)有关 ①。图中的红蓝矩形即为两个极大矩形。

将删除星星的最小代价改为最多能够取多少点 ②。明显是 dp 的问题,所以看到 ①②,这里有个 trick,建出笛卡尔树,在树上求解。

我们思考在笛卡尔树上求解有什么好处。首先在大根笛卡尔树下,对于每个节点,都对应着一个极大矩形,即以自身高度为底,向左右上延伸的极大矩形。这样在遍历完整棵树后,我们相当于考虑了所有极大矩形。对于每个星星,影响的矩形是树上自叶子结点到某节点的一条链。

考虑树形 dp。观察到转移时只会被最高的星星所影响,设 \(f_{i,j}\) 表示当前在 \(i\) 节点,当前所有选中星星的高度最大值为 \(j\) 的最大权值和。有转移:

当 \(j>a_i\) 时,

​ 星星放在第 \(i\) 列,\(f_{i,j}=f_{ls,k}+f_{rs,k}+v_i(k\le a_i)\)

​ 放在左右两边,\(f_{i,j}=\max(f_{ls,j}+f_{rs,k})(k\le a_i)\) 或 \(f_{i,j}=\max(f_{ls,k}+f_{rs,j})(k\le a_i)\)

当 \(j\le a_i\) 时,

​ \(f_{i,j}=\max(f_{ls,k}+f_{rs,p})(k,p\le j)\)。

我们发现转移位置都是一段区间,并且直接开是开不下 \(f\) 数组的,可以用动态开点线段树维护。

考虑整体 dp,即线段树合并维护 dp。对于 \(j>a_i\),需要区间询问、单点修改 \(\max\),区间加;对于 \(j\le a_i\),此时每个位置的转移依赖自身,如果考虑每个转移,复杂。可以发现转移一直是前缀最大值,于是只需要用左右子树的最大值修改其中一个位置,保证后面能够转移到即可。

注意线段树合并时不一定要同时转移左右子树,可以一个一个和父亲合并转移,处理好 \(i\) 与左右子树的关系。

复杂度 \(O(n\log n)\)。

#include <bits/stdc++.h>
#define pii std::pair<int, i64>
#define fi first
#define se second
#define pb push_back

typedef long long i64;
const int N = 2e5 + 10;
int n, m, tot;
int a[N], st[N], top;
i64 sum;
std::vector<pii> point[N];
int tr[N][2];
struct seg {
	int ls, rs;
	i64 mx, lzg;
} t[N * 40];
void pushup(int u, int ls, int rs) {
	t[u].mx = std::max(t[ls].mx, t[rs].mx);
}
void gtag(int u, i64 v) {
	if(u) t[u].mx += v, t[u].lzg += v;
}
void pd(int u) {
	if(!t[u].lzg) return;
	gtag(t[u].ls, t[u].lzg), gtag(t[u].rs, t[u].lzg);
	t[u].lzg = 0;
}
void ins(int &u, int l, int r, int x, i64 y) { 
	if(!u) u = ++tot;
	if(l == r) {
		t[u].mx = std::max(t[u].mx, y);
		return;
	}
	int mid = (l + r) >> 1;
	pd(u);
	if(x <= mid) ins(t[u].ls, l, mid, x, y);
	else ins(t[u].rs, mid + 1, r, x, y);
	pushup(u, t[u].ls, t[u].rs);
}
void mdf(int u, int l, int r, int L, int R, i64 x) {
	if(L <= l && r <= R) {
		gtag(u, x);
		return;
	}
	int mid = (l + r) >> 1;
	pd(u);
	if(L <= mid) mdf(t[u].ls, l, mid, L, R, x);
	if(R > mid) mdf(t[u].rs, mid + 1, r, L, R, x);
	pushup(u, t[u].ls, t[u].rs);
}
void mg(int p1, int p2, int l, int r) {
	if(l == r) {
		t[p1].mx = std::max(t[p1].mx, t[p2].mx);
		return;
	}
	int mid = (l + r) >> 1;
	pd(p1), pd(p2);
	if(t[p1].ls && t[p2].ls) mg(t[p1].ls, t[p2].ls, l, mid);
	else if(t[p2].ls) t[p1].ls = t[p2].ls;
	if(t[p1].rs && t[p2].rs) mg(t[p1].rs, t[p2].rs, mid + 1, r);
	else if(t[p2].rs) t[p1].rs = t[p2].rs;
	pushup(p1, t[p1].ls, t[p1].rs);
}
i64 query(int u, int l, int r, int L, int R) {
	if(L <= l && r <= R) {
		return t[u].mx;
	}
	int mid = (l + r) >> 1;
	i64 ret = 0;
	pd(u);
	if(L <= mid) ret = std::max(ret, query(t[u].ls, l, mid, L, R));
	if(R > mid) ret = std::max(ret, query(t[u].rs, mid + 1, r, L, R));
	return ret;
}
void dp(int u, int v, int h) {
	i64 lv = query(u, 1, n, 1, h), rv = query(v, 1, n, 1, h);
	mdf(u, 1, n, h + 1, n, rv), mdf(v, 1, n, h + 1, n, lv);
	mg(u, v, 1, n);
	ins(u, 1, n, h, lv + rv);
}
void dfs(int u) {
	for(auto x : point[u]) {
		ins(u, 1, n, x.fi, x.se);
	}
	if(tr[u][0]) dfs(tr[u][0]), dp(u, tr[u][0], a[u]);
	if(tr[u][1]) dfs(tr[u][1]), dp(u, tr[u][1], a[u]);
} 
void Solve() {
	std::cin >> n;

	tot = n;
	for(int i = 1; i <= n; i++) {
		std::cin >> a[i];
		int now = top;
		while(now && a[st[now]] < a[i]) now--;
		if(now) tr[st[now]][1] = i;
		if(now < top) tr[i][0] = st[now + 1];
		st[++now] = i;
		top = now; 
	}
	std::cin >> m;
	for(int i = 1; i <= m; i++) {
		int x, y;
		i64 c;
		std::cin >> x >> y >> c;
		point[x].pb({y, c});
		sum += c;
	}
	dfs(st[1]);
	std::cout << sum - t[st[1]].mx << "\n";
}
int main() {
    std::ios::sync_with_stdio(false);
    std::cin.tie(nullptr);
    
	Solve();

	return 0;
}

标签:p2,p1,rs,int,JOISC2020,P7219,ls,dp
From: https://www.cnblogs.com/FireRaku/p/18111415

相关文章

  • 三道dp的题解报告
    三道dp的题解报告圆题目来源牛客练习赛122D题练习赛链接https://ac.nowcoder.com/acm/contest/75768题面原题面为中文就直接放原题面截图了。解法事实上,把\(n\)与\(1\)断掉,断环为链,把原来的线段看成区间,线段相交就是区间之间部分覆盖(区间有重复部分但是没有相互包含关......
  • 数位 dp / lianggj 让你学小专题应该怎么办
    P4999#include<bits/stdc++.h>#defineintlonglong#defineup(i,l,r)for(inti=l;i<=r;++i)#definedn(i,r,l)for(inti=r;i>=l;--i)usingnamespacestd;constintN=20,M=9*18+5,P=1e9+7;intt,l,r,f[N][M],a[N],len,ans;/*Q:查询......
  • 状压dp板子(cf div4 #937)
    #include<bits/stdc++.h>usingnamespacestd;intn;vector<int>v[20];stringa[20],b[20];booldp[500010][20];voiddfs(ints,intnow){dp[s][now]=true;for(autonxt:v[now]){if(s&(1<<nxt))continue;......
  • RudderStack 开源CDP 平台
    RudderStack是基于golang开发的开源CDP平台包含的特性eventstreaming 支持16+sdkprofiles 快速基于dw或者datalake构建客户画像reverseetl 支持反向etl数据治理 支持增强追踪,方便对于一些隐私数据的管理event转换 支持基于js以及python进行数据修复200+......
  • 树形DP+树上路径贡献
    题目一棵树有n个节点,每个节点都同有一个符号+或者-,表示该节点是正极节点还是负极节点。如果从正极走到负极,或者从负极走到正极,都会受到1点伤害。现在需要知道走过所有路径后会受到的总伤害是多少?树上任意2点存在唯一的路径。需要计算所有任意2点的路径的伤害和。注意:从u到,和从v......
  • Offer必备算法17_子数组子串dp_八道力扣题详解(由易到难)
    目录①力扣53.最大子数组和解析代码②力扣918.环形子数组的最大和解析代码③力扣152.乘积最大子数组解析代码④力扣1567.乘积为正数的最长子数组长度解析代码⑤力扣413.等差数列划分解析代码⑥力扣978.最长湍流子数组解析代码⑦力扣139.单词拆分解析代码......
  • Offer必备算法19_子序列dp_八道力扣题详解(由易到难)
    目录①力扣300.最长递增子序列解析代码②力扣376.摆动序列解析代码③力扣673.最长递增子序列的个数解析代码④力扣646.最长数对链解析代码⑤力扣1218.最长定差子序列解析代码⑥力扣873.最长的斐波那契子序列的长度解析代码⑦力扣1027.最长等差数列解析代......
  • wordpress负载均衡
    部署的顺序先有后端web7,8,再有前端的lb-5。web7#装nginxgroupaddwww-g666useraddwww-s/sbin/nologin-M-u666-g666#你要确保,你装的所有机器,软件版本都一致,否则可能出奇怪bug#web7,web8用同一套软件,你最好自己去自建yum源cat>/etc/yum.repos.d/61.repo......
  • Matlab构建上位机:UDP测试
    参考:UDP理解及UDP的MATLAB实现MatlabUDP-CSDN博客文中代码:建立连接fclose(instrfindall);%先关闭之前可能存在的UDP%127.0.0.1即为本地u1=udp('127.0.0.1','RemotePort',8847,'LocalPort',8848);%u1的本机端口为8848,即监听所有发到8848端口的消息;%u1的远程端口为8847,......
  • CF1557D (dx)(dp技巧)
    比较有意思的一道题。看到将一个区间涂黑可以想到线段树。然后看到最少删除,想到最多保留。然后我一开始想的是贪心,对于每条线段找到前面最近的,然后对于每个高度取min即可。然后测了一下样例,寄了。会被这个hack掉对于这个,我们在做2时会把中间删了,然后做1的时候就寄了。这就说明......