首页 > 其他分享 >题解:P7482 不条理狂诗曲

题解:P7482 不条理狂诗曲

时间:2024-07-22 09:41:43浏览次数:16  
标签:P7482 int 题解 mid 狂诗 Maxn max date mod

题解:P7482 不条理狂诗曲

本题解借鉴 blossom_j 大佬思路,但这位大佬的题解似乎没放正确代码

题意

对于每一个 \(a\) 的子区间 \(a_{l\dots r}\),求选择若干个不连续的数的和的最大值,对答案取模 \(10^{9}+7\)。

思路

主要算法:分治。

计算跨过中点 \(mid\) 的区间的 \(f\) 之和。

首先我们可以写一个 DP。\(f_{i,0/1,0/1}\) 表示 DP 已到达 \(i\) 的位置。第一个 \(0/1\) 表示 \(mid\) 或 \(mid+1\) 是否被选择。第二个 \(0/1\) 表示是否选择了第 \(i\) 个元素。

枚举 \(l\) 对所有的 \(r\) 求和:设 \(g_{i,j}=\max\{f_{i,j,0},f_{i,j,1}\}\)。区间 $\left [ l,r \right ] $ 的答案就是 \(\max\{g_{l,0}+\max\{g_{r,0},g_{r,1}\},g_{l,1}+g_{r,0}\}\)。

若 \(g_{l,0}-g_{l,1}\le g_{r,0}-\max\{g_{r,0},g_{r,1}\}\),则必需选择后者。

将 \(g_{r,0}-\max\{g_{r,0},g_{l,1}\}\) 排序,二分寻找分界点。

代码

#include<bits/stdc++.h>
#define int long long
using namespace std;
const int mod = 1000000007, Maxn = 2 * 1e5 + 10;
int n, a[Maxn], ans = 0, f[Maxn][2][2], g[Maxn][2],sa[Maxn], sb[Maxn];
struct Node {
	int date, ID;
} d[Maxn];

bool  cmp(Node x, Node y) {
	return x.date < y.date;
}

int ask(int x, int l, int r) {
	while (l < r) {
		int mid = (l + r + 1) >> 1;
		if (d[mid].date <= x)l = mid;
		else r = mid - 1;
	}
	return l;
}

void solve(int l, int r) {
	if (l == r) {
		ans = (ans + a[l]) % mod;
		return;
	}

	int mid = (l + r) >> 1;
	f[mid + 1][0][0] = f[mid + 1][0][1] = f[mid + 1][1][0] = 0, f[mid + 1][1][1] = a[mid + 1];

	g[mid + 1][0] = f[mid + 1][0][0];
	g[mid + 1][1] = f[mid + 1][1][1];

	d[mid + 1].ID = mid + 1;
	d[mid + 1].date = g[mid + 1][0] - max(g[mid + 1][0], g[mid + 1][1]);


	for (int i = mid + 2; i <= r; i++) {
		f[i][0][0] = max(f[i - 1][0][0], f[i - 1][0][1]);
		f[i][1][0] = max(f[i - 1][1][0], f[i - 1][1][1]);
		f[i][0][1] = f[i - 1][0][0] + a[i];
		f[i][1][1] = f[i - 1][1][0] + a[i];

		g[i][0] = max(f[i][0][0], f[i][0][1]);
		g[i][1] = max(f[i][1][0], f[i][1][1]);

		d[i].ID = i;
		d[i].date = g[i][0] - max(g[i][0], g[i][1]);
	}


	sort(d + mid + 1, d + r + 1, cmp);

	sa[mid] = sb[mid] = sa[r + 1] = sb[r + 1] = 0;
	for (int i = mid + 1; i <= r + 1; i++) {
		sa[i] = sa[i - 1] + g[d[i].ID][0];
		sb[i] = sb[i - 1] + max(g[d[i].ID][0], g[d[i].ID][1]);
	}

	f[mid][0][0] = f[mid][0][1] = f[mid][1][0] = 0, f[mid][1][1] = a[mid];

	g[mid][0] = f[mid][0][0];
	g[mid][1] = f[mid][1][1];

	int k = ask(g[mid][0] - g[mid][1], mid, r);
	ans = (ans + g[mid][0] * (k - mid) % mod + sb[k] + sa[r] - sa[k] + g[mid][1] * (r - k) % mod + mod) % mod;

	for (int i = mid - 1; i >= l; i--) {
		f[i][0][0] = max(f[i + 1][0][0], f[i + 1][0][1]);
		f[i][1][0] = max(f[i + 1][1][0], f[i + 1][1][1]);
		f[i][0][1] = f[i + 1][0][0] + a[i];
		f[i][1][1] = f[i + 1][1][0] + a[i];

		g[i][0] = max(f[i][0][0], f[i][0][1]);
		g[i][1] = max(f[i][1][0], f[i][1][1]);

		int k = ask(g[i][0] - g[i][1], mid, r);
		ans = (ans + g[i][0] * (k - mid) % mod + sb[k] + sa[r] - sa[k] + g[i][1] * (r - k) % mod + mod) % mod;
	}

	solve(l, mid), solve(mid + 1, r);
}
signed main() {
	cin >> n;
	for (int i = 1; i <= n; i++)cin >> a[i];
	solve(1, n);
	cout << ans << '\n';
	return 0;
}

标签:P7482,int,题解,mid,狂诗,Maxn,max,date,mod
From: https://www.cnblogs.com/AUBSwords/p/18315433

相关文章

  • P3041 [USACO12JAN] Video Game G 题解 AC自动机
    本题是一道AC自动机上的dp。首先不难想到状态定义f(i,j)表示仅考虑前i 个位置,第i 个字符是j 的分数,但无法转移,所以考虑将j这一维转化为表示AC自动机上的点。再定义val(i)表示以i 结尾的所有技能种数,则转移方程为f(i,j)=max(f(i,j),f(i-1,father(j)+val(j......
  • Loj P10064 黑暗城堡 题解 最短路径树记数
    这道题是一道最短路径树问题。首先,什么是最短路径树呢?定义一个图的最短路径树表示一个图上的生成树,且根节点到图上任一点的距离等于原图上两者之间的距离。而不难发现,题目其实是在求图上最短路径树记数。首先,建出最短路径树。枚举每条边,判断边两个端点到根的距离之差是否为......
  • 【题解】P4648 [IOI2007] pairs 动物对数
    Problem给定模板\(B(1\leB\le3)\),代表\(B\)维空间。其中有\(n\)个点,给出坐标与坐标上限\(M\),求\(n\)个点中曼哈顿距离\(\leD\)的对数。Solve\(B=1\)考虑将问题化简成:求\(\sum\limits_{i=1}^n\sum\limits_{j=1}^{i-1}[dis(i,j)\leqD]\)。其中\(dis(i,j)\)......
  • 圆方树学习笔记 & 最短路 题解
    前言圆方树学习笔记,从一道例题讲起。题目链接:Hydro&bzoj。题意简述仙人掌上求两点距离。题目分析为了把仙人掌的性质发挥出来,考虑将其变成一棵树。圆方树就是这样转换的工具。先讲讲圆方树的概念:原图上的点为圆点,每个点双对应一个方点,树边都是方点连向点双内的圆点。具......
  • 跑步爱天天 题解
    题意简述一棵以\(1\)为根的树,儿子间有先后顺序。初始每个结点上有一个警卫,警卫按照深度优先遍历其子树,儿子间的先后顺序体现在这里,回到起始点后开始新一轮的遍历。yzh想要从\(S\)走到\(1\),请问她会在路上遇到多少警卫(\(S\)点的也算)。题目分析法\(1\)先来讲一讲我考场......
  • 题解:Codeforces Round 960 (Div. 2) D
    D.GridPuzzletimelimitpertest:2secondsmemorylimitpertest:256megabytesinput:standardinputoutput:standardoutputYouaregivenanarray\(a\)ofsize\(n\).Thereisan\(n\timesn\)grid.Inthe\(i\)-throw,thefirst\(a_i\)......
  • 【2024最新华为OD-C/D卷试题汇总】[支持在线评测] LYA的生日派对座位安排(200分) - 三
    ......
  • 题解:Codeforces Round 960 (Div. 2) C
    C.MadMADSumtimelimitpertest:2secondsmemorylimitpertest:256megabytesinput:standardinputoutput:standardoutputWedefinethe\(\operatorname{MAD}\)(MaximumAppearingDuplicate)inanarrayasthelargestnumberthatappearsatleast......
  • NOI2024 集合 题解
    给个链接:集合。很神秘的题目。基本上看到之后就可以想到哈希。首先想到一个比较神秘的暴力。就是对于每个询问,扫一遍所有\(a\)中的数出现的位置,把它弄成一个哈希值(具体怎么弄随意)存到set里,然后看看是不是和\(b\)中的数出现的位置这样操作后的集合完全相等。事实上就是判断......
  • B3956 [GESP202403 三级] 字母求和 题解
    B3956[GESP202403三级]字母求和题解当时在考试,3分钟A了,结果第二题T了。#include<bits/stdc++.h>usingnamespacestd;constintN=1e5+2;constintN1=1e3+2;typedeflonglongll;typedefunsignedlonglongull;#definefo(i,n,m)for(inti=n;i<=m;i++)......