Luogu P10843 Turtle and Cycles 题解 [ 蓝 ] [ 差分 ] [ 前缀和 ] [ 贪心 ] [ 数学 ]
Turtle and Cycles:修改转化为交换差分数组的 trick 运用。这个 trick 实际上在 NOIp2021 里出过一次了。
转化
首先,\(a_{(i - 1) \bmod n} + a_{(i + 1) \bmod n} - a_i\) 是个很典的形式。设 \(a_{(i - 1) \bmod n}=x,a_i=y,a_{(i + 1) \bmod n}=z\),那么 \(a_i\) 处的原来的差分值为 \(y-x\),\(a_{(i + 1) \bmod n}\) 处的原来的差分值为 \(z-y\)。
修改后,\(a_i\) 处的差分值变为 \(x+z-y-x=z-y\),\(a_{(i + 1) \bmod n}\) 处的差分值变为 \(z-(x+z-y)=z-x-z+y=y-x\)。所以对 \(a_i\) 进行修改后相当于把 \(a_i\) 与 \(a_{(i + 1) \bmod n}\) 处的差分值交换了。
设 \(b_i\) 表示 \(a_i\) 处的差分值。
那么我们要让这个环满足条件,首先就得破环为链。拆成链之后,显然只能有一个地方满足 \(b_i>0,b_{(i + 1) \bmod n}<0\)。于是设 \(b_i>0\) 时为 \(1\),否则为 \(0\),就可以先转化为 \(01\) 串问题来解决。这个 \(01\) 串合法的条件就是满足 \(111\dots000\dots111\) 的形式。
贪心
接下来考虑如何最小化操作次数,不难发现,贪心移动时,对于在中间点左边的 \(1\) 都必须移动到左边来,对于在中间点右边的 \(1\) 都必须移动到右边来。那么如何快速计算这个值呢?
假设我们当前计算的区间为 \([l,r]\)。
先考虑左半边的情况,设 \(g_i\) 表示前 \(i\) 个数里的 \(1\) 的下标之和为多少,\(f_i\) 表示前 \(i\) 个数中 \(1\) 的个数。如果先让这些 \(1\) 全都移动到 \(l\) 处,那么答案就是 \(g_{mid}-g_{l-1}-l\times (f_{mid}-f_{l-1})\)。
但是实际有些 \(1\) 是不需要完全移动到 \(l\) 处的,它们有些可以移动到 \(l+1,l+2,l+3\) 等位置,这就需要减掉它们的贡献。而这些贡献显然构成一个等差数列,那么只需要求和一下就好了,多算的贡献即为 \(\frac{(f_{mid}-f_{l-1}-1)\times (f_{mid}-f_{l-1})}{2}\)。
左半边的答案就是 \(g_{mid}-g_{l-1}-l\times (f_{mid}-f_{l-1})-\frac{(f_{mid}-f_{l-1}-1)\times (f_{mid}-f_{l-1})}{2}\)。
右半边同理,就不写了。
时间复杂度 \(O(n)\)。
代码
#include <bits/stdc++.h>
#define fi first
#define se second
#define lc (p<<1)
#define rc ((p<<1)|1)
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int,int> pi;
int t,n,a[400005];
ll g[400005],f[400005],sg[400005];
bitset<400005>b;
ll cal()
{
ll ans=0x3f3f3f3f3f3f3f3f;
for(ll i=1;i<=n;i++)
{
int mid=i+n/2-1;
ll lk=f[mid]-f[i-1];
ll lres=g[mid]-g[i-1]-lk*i-(lk-1)*lk/2;
ll rk=f[i+n-1]-f[mid];
ll rres=sg[mid+1]-sg[i+n]-rk*(2*n-(i+n-1)+1)-(rk-1)*rk/2;
ans=min(ans,lres+rres);
}
return ans;
}
void solve()
{
cin>>n;
for(int i=1;i<=n;i++)
{
cin>>a[i];
a[i+n]=a[i];
}
for(int i=2;i<=n+1;i++)
{
b[i-1]=(a[i]>a[i-1]);
b[i-1+n]=b[i-1];
}
for(int i=1;i<=2*n;i++)
{
f[i]=f[i-1]+b[i];
g[i]=g[i-1]+i*b[i];
}
sg[2*n+1]=0;
for(int j=1,i=2*n;i>=1;i--,j++)sg[i]=sg[i+1]+j*b[i];
cout<<cal()<<'\n';
}
int main()
{
ios::sync_with_stdio(0);
cin.tie(0);
cout.tie(0);
cin>>t;
while(t--)solve();
return 0;
}
Luogu P10843 Turtle and Cycles 题解 [ 蓝 ] [ 差分 ] [ 前缀和 ] [ 贪心 ] [ 数学 ]的更多相关文章
- Luogu 1083 借教室(二分,差分)
Luogu 1083 借教室(二分,差分) Description 在大学期间,经常需要租借教室.大到院系举办活动,小到学习小组自习讨论,都需要 向学校申请借教室.教室的大小功能不同,借教室人的身份不 ...
- luogu P1126 机器人搬重物 题解
luogu P1126 机器人搬重物 题解 题目描述 机器人移动学会(\(RMI\))现在正尝试用机器人搬运物品.机器人的形状是一个直径\(1.6\)米的球.在试验阶段,机器人被用于在一个储藏室中搬运 ...
- Educational Codeforces Round 61 C 枚举 + 差分前缀和
https://codeforces.com/contest/1132/problem/C 枚举 + 差分前缀和 题意 有一段[1,n]的线段,有q个区间,选择其中q-2个区间,使得覆盖线段上的点最多 ...
- HRBUST 1909——理工门外的树——————【离线处理,差分前缀和】
理工门外的树 Time Limit: 1000 MS Memory Limit: 32768 KB 64-bit integer IO format: %lld , %llu Java class n ...
- HDU 5419——Victor and Toys——————【线段树|差分前缀和】
Victor and Toys Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 262144/131072 K (Java/Others ...
- HDU 5452——Minimum Cut——————【树链剖分+差分前缀和】ACdream 1429——Diversion——————【树链剖分】
Minimum Cut Time Limit: 3000/2000 MS (Java/Others) Memory Limit: 65535/102400 K (Java/Others)Tota ...
- 【题解】Cut the Sequence(贪心区间覆盖)
[题解]Cut the Sequence(贪心区间覆盖) POJ - 3017 题意: 给定一大堆线段,问用这些线段覆盖一个连续区间1-x的最小使用线段的数量. 题解 考虑一个这样的贪心: 先按照左端 ...
- 洛谷 P3258 [JLOI2014]松鼠的新家 树链剖分+差分前缀和优化
目录 题面 题目链接 题目描述 输入输出格式 输入格式 输出格式 输入输出样例 输入样例: 输出样例: 说明 说明 思路 AC代码 优化 优化后AC代码 总结 题面 题目链接 P3258 [JLOI2 ...
- 【题解】[HNOI2015]菜肴制作(贪心+topo序)
[题解][HNOI2015]菜肴制作(贪心+topo序) 题意:请你构造一个排列\(p[i]\)使得对于数组\(arc[i]=p[i]\)的字典序最小,并且对于给定的有序数对\((u,v)\)保证你给 ...
- 【题解】P1712 [NOI2016]区间(贪心+线段树)
[题解]P1712 [NOI2016]区间(贪心+线段树) 一个observe是,对于一个合法的方案,将其线段长度按照从大到小排序后,他极差的来源是第一个和最后一个.或者说,读入的线段按照长度分类后, ...
随机推荐
- mongo迁移工具之mongo-shake
最近需要进行MongoDB中数据迁移,之前使用过阿里系的redisShake感觉不错, 这次打算使用mongoShake来进行同步 github: https://github.com/alibaba ...
- 使用PicGo存储markdown图片(阿里云或者github)
PicGo代替极简图床 之前使用极简床图,但是后来好像挂了,真是一件悲伤的事,最近才发现了一个神器,开源的PicGo,已经有各个平台的版本了.链接如下:https://github.com/Molun ...
- c++动态库详解
dmjcb个人博客 原文地址 概念 动态库, 又称动态链接库(\(Dynamic\) \(Link\) \(Library\), \(DLL\)), 是包含程序代码和数据的可执行文件, 在运行时被程序 ...
- 编译器-FOLLOW集合
语法分析器的两个重要函数 FIRST和FOLLOW 一.FOLLOW的定义 在句型中紧跟在A右边的终结符号的集合 如果A是某些句型的最右符号,那么$在FOLLOW(A)中 A:非终结符 二.计算方法 ...
- LSTM学习三维轨迹的Python实现
一.引言 长短期记忆网络(LSTM)是一种强大的递归神经网络(RNN),广泛应用于时间序列预测.自然语言处理等任务.在处理具有时间序列特征的数据时,LSTM通过引入记忆单元和门控机制,能够更有效地捕捉 ...
- 【报错解决】【人工智能】【深度学习】验证cuda和tensorflow之间的版本对应关系时遇到的问题
验证环境B 验证成功,没有问题 验证环境A 得到结果false 检查是否与CUDA关联成功 tf.test.is_built_with_cuda() 发现没有关联成功 根据查询可知道,失败的原因是1. ...
- jenkins build
clean test org.jacoco:jacoco-maven-plugin:0.8.5:prepare-agent org.owasp:dependency-check-maven:5.3.0 ...
- Qt/C++音视频开发78-获取本地摄像头支持的分辨率/帧率/格式等信息/mjpeg/yuyv/h264
一.前言 上一篇文章讲到用ffmpeg命令方式执行打印到日志输出,可以拿到本地摄像头设备信息,顺藤摸瓜,发现可以通过执行 ffmpeg -f dshow -list_options true -i v ...
- Qt编写物联网管理平台37-逻辑设计
一.前言 本系统的逻辑设计是个人认为做过的系统中最好的,一个系统支持多个通信端口,每个通信端口都可选不同的通信协议,一个通信端口可以接255个控制器,相当于主设备,一个控制器可以接255个探测器,相当 ...
- Qt编写安防视频监控系统31-onvif设备搜索
一.前言 做视频监控系统,绕不过onvif这玩意,这玩意主要就是为了统一一个大概的标准,能够对各个厂家的监控设备进行常用的一些操作,比如搜索.获取信息.云台控制.事件订阅.抓拍图片等,如果没有这个规范 ...