CF EDU 105 C - 1D Sokoban

C - 1D Sokoban

二分 + 找性质

可分正负的箱子分别讨论

本题的关键是发现一个重要的性质:因为推箱子这个过程会让被推到的箱子成为连续的一段,若想让在特殊位置的箱子最多,则这一段的终点一定要在特殊位置上(起点也可以, 这里的一定不是说不在特殊位置就取不到最优,而是在特殊位置上的某些情况一定可以取到最优,所以只考虑在特殊位置上的最大值就是全局的最大值)

证明思路:若结尾不是在特殊位置上,那么那把它移到最近的特殊位置上一定不会更劣

所以可以枚举每一个特殊位置,让连续一段箱子的末尾被推到这个点上,看这时的答案是多少

答案 = 这一段箱子覆盖的特殊位置 + 这一段箱子之后的点中,有多少箱子原来就在特殊位置上

第一项可二分求出,第二项可用前缀和求出

#include <iostream>
#include <cstring>
#include <algorithm>
#include <vector>
#include <cmath>
#include <map>
using namespace std;
typedef long long ll;

const int N = 2e5 + 10;
int n, m;
int a[N], b[N], c[N], d[N], pre[N];
map<int, bool> mp;

int solve(int a[], int b[], int n, int m)
{
	if (n <= 0 || m <= 0)
		return 0;
	sort(a + 1, a + n + 1);
	sort(b + 1, b + m + 1);
	
	mp.clear();
	for (int i = 1; i <= n; i++)
		mp[a[i]] = true;
	for (int i = 1; i <= m; i++)
	{
		pre[i] = pre[i-1];
		if (mp.count(b[i]))
			pre[i]++;
	}
	int ans = 0;
	for (int i = 1; i <= m; i++)
	{
		int len = upper_bound(a + 1, a + n + 1, b[i]) - a - 1;
		int r = b[i], l = r - len + 1;
		int cnt = upper_bound(b + 1, b + m + 1, r) - lower_bound(b + 1, b + m + 1, l);
		cnt += pre[m] - pre[i];
		ans = max(ans, cnt);
	}
	return ans;
}

int main()
{
	ios::sync_with_stdio(false), cin.tie(0), cout.tie(0);
	int T;
	cin >> T;
	while(T--)
	{
		cin >> n >> m;
		for (int i = 1; i <= n; i++)
			cin >> a[i];
		for (int i = 1; i <= m; i++)
			cin >> b[i];
		int ans = 0;
		int cnta = 0, cntb = 0;
		//正数
		for (int i = 1; i <= n; i++)
			if (a[i] > 0) c[++cnta] = a[i];
		for (int i = 1; i <= m; i++)
			if (b[i] > 0) d[++cntb] = b[i];
		ans = solve(c, d, cnta, cntb);
		//负数
		cnta = 0, cntb = 0;
		for (int i = 1; i <= n; i++)
			if (a[i] < 0) c[++cnta] = -a[i];
		for (int i = 1; i <= m; i++)
			if (b[i] < 0) d[++cntb] = -b[i];
		ans += solve(c, d, cnta, cntb);
		cout << ans << endl;
	}
	return 0;
}
posted @ 2022-05-23 12:25  hzy0227  阅读(26)  评论(0编辑  收藏  举报