首页 > 其他分享 >【SD集训】20230425 T2 差(difference) 题解 CF1500F 【Cupboards Jumps】

【SD集训】20230425 T2 差(difference) 题解 CF1500F 【Cupboards Jumps】

时间:2023-04-25 16:11:28浏览次数:49  
标签:20230425 max int 题解 Jumps long leq ans front

大家可以猜猜看为什么有两个标题,因为这个原因本文就不设密码了,被 He_ren 的原题创到了。

吐槽一下,He_ren 甚至出原题还用脚造数据,虽然数据确实比较难造。不过那两个 \(O(n^2)\) 老哥好像都没最后将所有数调整成非负,遗憾 20。

有人场切 * 3500 却没过签到题,我不说是谁。

题目描述

给定正整数 \(n,C\) 和长度为 \(n-2\) 的序列 \(w(0\leq w_i\leq C)\)。

你需要构造一个长度为 \(n\) 的序列 \(h\),满足:

  • 对于任意整数 \(i\in[1,n-2]\) 有 \(\max\{h_i,h_{i+1},h_{i+2} \} -\min\{h_i,h_{i+1},h_{i+2} \}=w_i\)。

  • 序列 \(h\) 中的任意元素 \(h_i\) 满足 \(0\leq h_i\leq 10^{18}\)。

有解输出 YES 之后输出任意一个满足要求的序列 \(h\),无解输出 NO

数据范围:\(3\leq n\leq10^6\),\(0\leq C\leq10^{12}\)。

题解

这道题在中山集训时讲过,感谢 Kostlin 大师。

不过当时笔者鸽了啊啊啊啊啊啊,于是模拟赛赛时补题.jpg

首先考虑差分,令 \(f_i=h_{i+1}-h_i\)。于是限制就转化成了 \(\max\{0, h_i, h_{i+1}\} - \min\{0, h_i, h_{i+1}\} = w_i\),显然等价于 \(\max\limits_{i,j\in \{0,h_i,h_{i+1}\}}\{i-j\} = w_i\)。考虑展开这 \(9\) 项,不难发现 \(\max\limits_{i,j\in \{0,h_i,h_{i+1}\}}\{i-j\} = \max\{|h_i|,|h_{i+1}|,|h_i+h_{i+1}|\}\)。

之后考虑一个 DP,\(f_{i,j}\) 表示第 \(i\) 个位置 \(|h_i|=j(j\ge 0)\),值为 \(0/1\) 表示是否有解。这么设状态的一个重要原因是,我们观察到一个性质,在确定 \(\max\{|h_i|,|h_{i+1}|,|h_i+h_{i+1}|\}\),我们只关心相邻两个数是否同号,以及它们的绝对值,并不关心它们具体谁正谁负,或者到底是同正还是同负,只需要 DP 完构造方案时判断一下。

我们发现只需要维护值为 \(1\) 的 DP 状态集合即可。我们具体分析一下我们的 DP 转移:

  1. \(h_i\) 和 \(h_{i+1}\) 同号,则显然 \(\max\{|h_i|,|h_{i+1}|,|h_i+h_{i+1}|\}=|h_i+h_{i+1}|\)。相当于是 \(f_{i,j}\to f_{i+1,w_i-j}\);

  2. \(h_i\) 和 \(h_{i+1}\) 异号,则显然 \(\max\{|h_i|,|h_{i+1}|,|h_i+h_{i+1}|\}=\max\{|h_i|,|h_{i+1}|\}\)。分类讨论一下:

    1. \(j=w_i\),则 \(f_{i+1,k}(0\le k\le w_i)\);
    2. \(k=w_i\),则若存在 \(j\),满足 \(f_{i,j}\),则 \(f_{i+1,w_i}=true\)。

我们发现新状态中,首先,情况 1 是很好维护的,我们只需要维护一个一次函数 \(ax+b\) 即可,并且这里 \(a\in \{-1,1\}\)。其次,情况 2.1 也是很好维护的,相当于当前状态变成全集。最后,我们考虑情况 2.2。这样的状态我们额外维护一下即可。因为,我们注意到这样的状态每次要么是当前最大,要么是当前最小(取决于 \(a\) 的正负),删除状态的时候需要弹出头或尾,所以我们考虑使用 deque 维护,@帅气yuyue(雾。

时间复杂度 \(O(n)\)。

代码

#include <bits/stdc++.h>
#define int long long
#define SZ(x) (int) x.size() - 1
#define all(x) x.begin(), x.end()
#define ms(x, y) memset(x, y, sizeof x)
#define F(i, x, y) for (int i = (x); i <= (y); i++)
#define DF(i, x, y) for (int i = (x); i >= (y); i--)
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
template <typename T> void chkmax(T& x, T y) { x = max(x, y); }
template <typename T> void chkmin(T& x, T y) { x = min(x, y); }
template <typename T> void read(T &x) {
	x = 0; int f = 1; char c = getchar();
	for (; !isdigit(c); c = getchar()) if (c == '-') f = -f;
	for (; isdigit(c); c = getchar()) x = x * 10 + c - '0';
	x *= f;
}
const int N = 1e6 + 10;
int n, C, a = 1, b, l = -1e18, r = 1e18, t[N], ans[N], s[N];
deque <int> q;
int calc1(int x) {
	return (x - b) / a;
}
int calc2(int x) {
	return a * x + b;
}
signed main() {
	read(n); read(C);
	F(i, 1, n - 2) {
		int x; read(x); t[i] = x;
		int cx = calc1(x);
		int L = calc1(0), R = cx; if (L > R) swap(L, R);
		chkmax(l, L); chkmin(r, R);
		while (q.size() && q.front() < L) q.pop_front();
		while (q.size() && q.back() > R) q.pop_back();
		if (q.empty() && l > r) return puts("NO"), 0;
		if ((l <= r && (l == cx || r == cx)) || (q.size() && (q.front() == cx || q.back() == cx))) {
			ans[i] = x;
			a = 1, b = 0;
			l = 0, r = x;
			q.clear();
		} else {
			ans[i] = calc2(l > r ? q.front() : l);
			a *= -1, b = x - b;
			if (a == 1) q.push_back(calc1(x));
			else q.push_front(calc1(x));
		}
	} puts("YES");
	ans[n - 1] = calc2(l > r ? q.front() : l);
	DF(i, n - 2, 1) {
		assert(abs(ans[i + 1]) <= t[i]);
		if (ans[i] == t[i] || abs(ans[i + 1]) == t[i]) {
			if (ans[i + 1] > 0) ans[i] *= -1;
		} else {
			ans[i] = t[i] - abs(ans[i + 1]);
			if (ans[i + 1] < 0) ans[i] *= -1;
		}
	}
	int mn = 0;
	F(i, 2, n) s[i] = s[i - 1] + ans[i - 1], chkmin(mn, s[i]);
	F(i, 1, n) cout << s[i] - mn << " ";
	return 0;
}

标签:20230425,max,int,题解,Jumps,long,leq,ans,front
From: https://www.cnblogs.com/zhaohaikun/p/17352935.html

相关文章

  • Net6+axios 返回401 axios不能获取 状态码问题解决
    错误使用app.UseAuthentication();//认证 这里要加,位置不能反app.UseAuthorization();//授权 app.UseCors();//启用Cors解决方法app.UseCors();//启用Corsapp.UseAuthentication();//认证 这里要加,位置不能反app.UseAuthorization();//授权  更换前更换后  ......
  • 20230425001 - DataGridView绑定了数据之后, 再添加CheckBox列的解决方案
                 DataGridViewCheckBoxColumncheckBoxColumn=newDataGridViewCheckBoxColumn();           checkBoxColumn.Name="select";           checkBoxColumn.HeaderText="选择";           dgv_M.Columns.Inse......
  • 题解:【CTS2022】 独立集问题
    题目链接来自2023SDPT-Round1-Day4课上Qingyu的讲解。考虑对于一个点多次操作会发生什么?第一次操作会将周围的点的权值吸过来,自己对答案的贡献乘\(-1\),周围的点的贡献乘\(+1\),得到新的权值\(a_x'=\pma_x\mp\sum_{y\inson_x}a_y\);再一次操作,我们可以将这个新的贡......
  • Mount cifs存储时提示not supported问题解决
    Mountcifs存储时提示notsupported问题解决:报错: mounterror(95):Operationnotsupported排查:1、当时刚好是mount2个存储,结果1个可以1个不行,就反馈给负责存储同事查看2个存储的区别2、存储同事查看后得出不行的是比较老的Netapp存储,mounterror(95)错误应该是不支持的smb协议......
  • leetcode-217-存在重复元素 题解
    题目描述给你一个整数数组nums。如果任一值在数组中出现至少两次,返回true;如果数组中每个元素互不相同,返回false。示例1:输入:nums=[1,2,3,1]输出:true示例2:输入:nums=[1,2,3,4]输出:false示例3:输入:nums=[1,1,1,3,3,4,3,2,4,2]输出:true提......
  • P9228 原神 题解
    题目传送门题目大意有一个魔法师,她可以用火元素攻击魔法把对附着冰元素的怪物的伤害\(\times2\),用冰元素攻击魔法把对附着火元素的怪物的伤害\(+5\)。每个怪物初始时没有附着任何元素,给出冰、火元素对每个怪物的初始伤害,魔法师可以任意安排攻击顺序,求最大总伤害。解题思路......
  • CF题解
    E.RearrangeBrackets2100括号树gq!https://codeforces.com/contest/1821/problem/E题解:若我们把序列看作是一个由匹配括号组成的森林,外层括号是内层括号的父亲,则整个正则括号序列的cost可以看作是森林中所有点的深度之和,根节点深度为0,故每次操作可以看作是将某棵树的父节点......
  • 跨域问题解决、其他权限校验方法
    跨域问题解决浏览器出于安全的考虑,使用XMLHttpRequest对象发起HTTP请求时必须遵守同源策略,否则就是跨域的HTTP请求,默认情况下是被禁止的。同源策略要求源相同才能正常进行通信,即协议、域名、端口号都完全一致。前后端分离项目前端项目和后端项目一般都不是同源的,所以肯定会存在......
  • 2023年团体程序设计天梯赛 题解
    仅更新L1,L2随后写**更好的阅读体验:2023年团体程序设计天梯赛题解**L1-1最好的文档有一位软件工程师说过一句很有道理的话:“Goodcodeisitsownbestdocumentation.”(好代码本身就是最好的文档)。本题就请你直接在屏幕上输出这句话。输入格式:本题没有输入。输出格式:在一行中输出......
  • 2023年团体程序设计天梯赛 题解
    仅更新L1,L2随后写**更好的阅读体验:2023年团体程序设计天梯赛题解**L1-1最好的文档有一位软件工程师说过一句很有道理的话:“Goodcodeisitsownbestdocumentation.”(好代码本身就是最好的文档)。本题就请你直接在屏幕上输出这句话。输入格式:本题没有输入。输出格式:在一行中输出......