CF1849
氵
在吃了五次罚时后,我终于放弃了卡常优先队列,并发现:把余 \(0\) 看作余 \(k\),答案就是余数从大到小排列的,每种余数内部又按照下标排序。
我为什么没想到哈希?自我检讨:见到关于字符串判定相等/不同个数时,一定要尝试用哈希!!!
记前缀 \([0,i)\) 的哈希值为 \(f_i\),全为 \(1\) 的前缀 \([0,i)\) 的哈希值为 \(g_i\),\([0,i)\) 中 \(1\) 的个数为 \(h_i\)。
设排序的区间为 \([l,r)\),则新哈希值为 \(f_n-f_r+f_l+g_r-g_{[r-(h_r-h_l)]}\)。
用 std::set 维护即可。
非哈希做法:
对每个位置求出它左边(包括它)的第一个 0 和右边(包括它)的第一个 1。存在两个数组 \(lf,rg\) 里面。
对于一个修改区间 \([l,r]\),它真实改动的部分其实是 \([rg_l,lf_r]\)。如果这两个元素设定为不存在,那么就没有修改。
把每对 \((rg_l,lf_r)\) 用 set 维护。
贪心法:先有一个错误但是有启发性的贪心:先把所有 \(2\) 涂黑,然后标记所有 \(2\) 边上的,再把剩下的涂黑。
这是错的,比如 0 2 1 2 0,把第一个 \(2\) 涂黑之后可以传递过去。
于是想到把一段连续非 0 的数合并为一个块。 这一段只需要一个硬币。
如果一个块内有 \(2\),这个块左右两边的两个都能被涂黑;如果只有 \(1\),那只有一个能被涂黑。
模拟即可。
dp 法:
\(dp_i\) 表示涂黑前 \(i\) 个的最小硬币数。
基本的,\(dp_{i+1}=\min(dp_{i+1},dp_i+1).\)
若 \(a[i+1]=1\),\(dp_{i+2}=\min(dp_{i+2},dp_{i}+1)\)。
若 \(a[i+2]=1\),\(dp_{i+2}=\min(dp_{i+2},dp_{i}+1)\)。
若 \(a[i+1]=1\),\(dp_{i+2}=\min(dp_{i+1},dp_{i}+1)\)。
若 \(a[i+1]=2\),\(dp_{i+2}=\min(dp_{i+2},dp_{i}+1)\)。
若 \(a[i+2]=2\),\(dp_{i+2}=\min(dp_{i+3},dp_{i}+1)\)。
答案为 \(\min(dp_n,dp_{n+1}).\)
朴实无华的题面:有多少个区间最大值在最小值右边。
先单调栈求出每个数左右两边第一个比他大/小的位置。
记 \(a_i\) 左边第一个比它大的位置是 \(l_i\),右边第一个比它大的位置是 \(r_i\),左边第一个比他小的位置 \(s_i\),右边第一个比它小的位置 \(t_i\)。
按套路,枚举每个 \(a_i\) 作为区间的最大值。
有两种方法计算这种区间的个数:
枚举 \(j:l_i\le j<i\),求出 \(j\sim i-1\) 的最小值 \(p\),要求在这个区间内 \(i\) 的右边不允许有比 \(p\) 小的。(否则最大值 \(i\) 就在最小值左边)即右端点 \(\ge t_p\) 就不合法,总个数减去不合法个数。
枚举 \(j:i<j\le r_i\),求出 \(i+1\sim j\) 的最小值 \(p\)。同理,只要左端点 \(\le s_p\) 就合法。
注意以上两种方法都需单独考虑 \(i=j\) 的情况。
我们枚举 \(a_i\) 时,根据 \(i-l_i,r_i-i\) 的大小选择复杂度较低的方法。便可通过此题。
但是感觉时间复杂度不太对啊?
如果按最大值从大到小枚举,每次在循环 \(x\) 时如果是重复循环,区间长度至多是上一次循环 \(x\) 时的一半。
为什么呢?假设上一次循环 \(x\) 时是枚举到 \(a_i\),这次是 \(a_j\)。(不妨 \(x\) 是左端点)
显然应该 \(j<i\),不然因为我们从大到小枚举,有 \(a_i>a_j\),那枚举 \(a_j\) 的左端点不应该枚举到 \(x\),因为此时 \(x\) 在 \(i\) 的左边,违反了上面的方法。
那么就有 \(x<j<i\),注意因为从大到小,所以 \(r_j\le i\)。
我们选择了用枚举左端点的方法,说明 \(j-l_j\le r_j-j\le i-j\).
那也就说明 \(j\) 这个位置应该在 \([l_j,i]\) 的左半部分,而 \(x\in [l_j,j]\),上次 \(x\) 被枚举到是和 \(i\) 一起,显然应该有 \(j-x\le \dfrac{i-x}{2}\).
那每个数被枚举到,区间长度至少除以 \(2\),因此复杂度 \(O(n\log n)\).
随机推荐
- python之十进制、二进制、八进制、十六进制转换
数字处理的时候偶尔会遇到一些进制的转换,以下提供一些进制转换的方法 一.十进制转化成二进制 使用bin()函数 1 x=10 2 print(bin(x)) 二.十进制转化为八进制 使用oct()函数 ...
- HTTP协议六种请求:GET,HEAD,PUT,DELETE,POST
Http定义了与服务器交互的不同方法,标准Http协议支持六种请求方法,即: 1.GET 2.POST 3.PUT 4.Delete 5.HEAD 6.Options 最基本的方法有4种,分别是GET ...
- KVM 核心功能:磁盘虚拟化
1 磁盘虚拟化简介 QEMU-KVM 提供磁盘虚拟化,从虚拟机角度看其自身拥有的磁盘即是实际的物理磁盘.实际上,虚拟机读写的磁盘数据保存在 host 上的物理磁盘. QEMU-KVM 主要有如下几 ...
- 每天学五分钟 Liunx 0101 | 服务篇:创建进程
创建子进程 上一节说过创建子进程的三种方式: 1. fork 复制进程:fork 会复制当前进程的副本,产生一个新的子进程,父子进程是完全独立的两个进程,他们掌握的资源(环境变量和普通变量)是一样的. ...
- Feign 进行rpc 调用时使用ribbon负载均衡源码解析
转载请注明出处: Feign客户端接口的动态代理生成是基于JDK的动态代理来实现的,那么在所有的方法调用的时候最终都会走InvocationHandler接口的实现,默认就是ReflectiveFei ...
- pycharm等编辑器快捷键大赏
1.序论 平时打代码的时候用鼠标在写代码的时候还行,代码一长就GG,快捷键便是我们的好选择 2.应用场景 1.问题一--选中一段指定区域 我想选中一段区域(我们一般用鼠标拖动),或者某一个标识符(光标 ...
- 【Mysql系列】(一)MySQL语句执行流程
首发博客地址 首发博客地址 系列文章地址 参考文章 MySQL 逻辑架构 连接器 连接命令一般是这么写的 mysql -h$ip -P$port -u$user -p 那么 什么是连接器? MySQL ...
- [转帖]AL32UTF8/UTF8(Unicode)数据库字符集含义 (文档 ID 1946289.1)
AL32UTF8/UTF8(Unicode)数据库字符集含义 (文档 ID 1946289.1) 适用于: Oracle Database Cloud Schema Service - 版本 N/A ...
- Redis异常问题分析黄金一分钟
Redis异常问题分析黄金一分钟 背景 同事发现一个环境redis比较卡顿,导致业务比较难以开展. 问题是下午出现的. 六点左右找到我这边. 想着帮忙看看, 问题其实没有定位完全, 仅是发现了一个可能 ...
- Prometheus+alertmanager实现告警的简单验证
Prometheus+alertmanager实现告警的简单验证 背景 学习源自: http://www.mydlq.club/article/126/ 上午没搞定, 中午睡不着,继续学习处理. 发现 ...