Problem

Description

小 \(Q\) 有 \(n\) 本书,每本书有一个独一无二的编号,现在它们正零乱地在地上排成了一排。

小 \(Q\) 希望把这一排书分成恰好 \(k\) 段,使得每段至少有一本书,然后把每段按照现在的顺序依次放到 \(k\) 层书架的每一层上去。

将所有书都放到书架上后,小 \(Q\) 这才突然意识到它们是乱序的,他只好把每一层的书分别按照编号从小到大排序。排序每次可以在 \(1\) 单位时间内交换同一层上两本相邻的书。

请写一个程序,帮助小 \(Q\) 计算如何划分这 \(k\) 段,且如何交换这些书,使得总交换次数最少。

Input Format

第一行包含两个正整数 \(n,k\)。

第二行包含 \(n\) 个互不相同的正整数 \(a_1, a_2, ..., a_n\),分别表示地面上每本书的编号。

Output Format

输出一行一个整数,即最少的总交换次数。

Sample

Input

6 3
4 3 6 2 5 1

Output

1

Explanation

Explanation for Input

按 \([4, 3, 6][2, 5][1]\) 划分,需要排序 \(1 + 0 + 0 = 1\) 次。

Range

\(1 \le n \le 40000, 1 \le k \le min(10, n), \forall 1 \le a_i \le n\)

Algorithm

\(DP\) ,决策单调性

Mentality

其实决策单调性也没啥可怕的地方,主要重点在于你的思考!

我们先列出最朴素的 \(dp\) 方程:\(dp[i][j]\) 表示将前 \(i\) 本书划分到 \(j\) 层书架的最小代价,那么我们设 \(w(i,j)\) 代表区间 \([i,j]\) 内的逆序对数目,我们就有如下方程:

\[dp[i][j]=Min_{p<i}(dp[p][j-1]+w(p+1,i))
\]

答案即为 \(dp[n][k]\) 。

这样 \(dp\) 的复杂度为 \(n^2k\) , 时间显然过不去。

那么由于式子非常决策单调性,那么我们考虑打表证明,果然有 \(w\) 函数满足四边形不等式。

随后当我们枚举一个 \(j\) 时,我们设 \(g[i]\) 为 \(dp[i][j]\) 的最优决策点,即 \(dp[i][j]=dp[g[i]][j-1]+w(g[i]+1,i)\),则有 \(g[i-1]\le g[i]\) 。

这个证明很简单,对于两个决策点 \(p<q\) ,如果 \(dp[q]+w\le dp[p]+w\) ,由于 \(w\) 函数越来越大,所以如果 \(q\) 比 \(p\) 更优,那么 \(q\) 永远比 \(p\) 更优。所以我们的最优决策点必定单调右移,即 \(g[i-1]\le g[i]\) 。

那么根据这个单调性搞事情,我们考虑分治 \(dp\) ,对于区间 \([l,r]\) 的 \(dp\) ,我们考虑确定它的最优决策点所在的区间 \([L,R]\) 。那么我们找出 \(mid\) 处的最优决策点 \(p\),根据决策单调性,则区间 \([mid+1,r]\) 的最优决策区间必定为 \([p,R]\) ,而区间 \([l,mid-1]\) 的最优决策区间则必定为 \([L,p]\) 。

根据这样递归分治,每次枚举最优决策区间更新 \(mid\) ,并递归处理,枚举决策区间便构成了一颗类似线段树的情况,那么对于同一递归层数,决策区间的和正好就是 \(O(n)\) ,那么决策区间枚举的总复杂度就是 \(nlog\) ,均摊下来,则我们枚举决策点的复杂度变为 \(log\) ,枚举状态复杂度为 \(O(nk)\) 。

样例的递归分治,当 \(k=2\) 时如下图:

由此可见,每层的决策点枚举只会是 \(O(n)\) ,那么总复杂度为 层数× \(n\) ,即 \(O(nlog)\) 。

接下来还剩一个状态转移,那么我们只需要管 \(w\) 函数如何快速地求出来就好了。这里我们先看我们的处理顺序为先递归左区间再递归右区间,那么大多数时候 \(w\) 的右端点只是根据 \(dp\) 需求不断增加 \(1\) ,递归下去的时候也只是减少一些。

则我们可以考虑维护一个类似莫队的东西,维护一个全局的 \(L\) 与 \(R\) ,每次需要获取答案的时候,我们就将 \(L\) 和 \(R\) 一步一步移动到指定位置,用树状数组动态维护逆序对即可。这样做的话大概总复杂度是 \(nlog\) 带个玄学常数?那么均摊下来就是 \(log\) 的复杂度。

则最后的时间复杂度=枚举状态×枚举决策点×状态转移=\(O(nk)×O(log)×O(log)\)=\(O(nklog^2)\) 。能够通过题目。

Code

#include <cstdio>
#include <cstring>
#include <iostream>
using namespace std;
int n, K, sum, L, R, now, a[40001], c[40001], f[40001][11];
void add(int k, int x) {
for (int i = k; i <= n; i += i & -i) c[i] += x;
}
int query(int x) {
int ans = 0;
for (int i = x; i > 0; i -= i & -i) ans += c[i];
return ans;
}
void Move(int l, int r) //莫队式移动
{
while (L < l) sum -= query(a[L] - 1), add(a[L++], -1);
while (L > l) sum += query(a[L - 1] - 1), add(a[--L], 1);
while (R < r) sum += R - L + 1 - query(a[R + 1]), add(a[++R], 1);
while (R > r) sum -= R - L + 1 - query(a[R]), add(a[R--], -1);
}
void solve(int l, int r, int L, int R) {
if (l > r) return;
int mid = (l + r) >> 1, p = L;
for (int i = L; i <= min(mid - 1, R); i++) {
Move(i + 1, mid);
int Sum = f[i][now - 1] + sum;
if (Sum < f[mid][now]) f[mid][now] = Sum, p = i; //确定最优决策点
}
solve(l, mid - 1, L, p);
solve(mid + 1, r, p, R); //递归
}
int main() {
cin >> n >> K;
L = 1; //初始化莫队指针
for (int i = 1; i <= n; i++) scanf("%d", &a[i]);
memset(f, 10, sizeof(f));
for (int i = 1; i <= n; i++) Move(1, i), f[i][1] = sum; //先计算 k=1
for (now = 2; now <= K; now++) solve(1, n, 1, n); //开始递归
cout << f[n][K];
}

【BZOJ 5125】小Q的书架的更多相关文章

  1. BZOJ5125: [Lydsy1712月赛]小Q的书架【决策单调性优化DP】【BIT】【莫队】【分治】

    小Q有n本书,每本书有一个独一无二的编号,现在它们正零乱地在地上排成了一排. 小Q希望把这一排书分成恰好k段,使得每段至少有一本书,然后把每段按照现在的顺序依次放到k层书架的每一层上去.将所有书都放到 ...

  2. bzoj 4017: 小Q的无敌异或

    4017: 小Q的无敌异或 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 593  Solved: 197[Submit][Status][Discu ...

  3. bzoj 5125: [Lydsy1712月赛]小Q的书架

    新学了一波 决策单调性 dp 套路.... 这种dp一般是长这样的 => f[i][j] = max/min  { f[i-1][k] + cost(k+1,j)} ,其中cost函数满足四边形 ...

  4. BZOJ5125 小Q的书架(决策单调性+动态规划+分治+树状数组)

    设f[i][j]为前i个划成j段的最小代价,枚举上个划分点转移.容易想到这个dp有决策单调性,感性证明一下比较显然.如果用单调栈维护决策就不太能快速的求出逆序对个数了,改为使用分治,移动端点时树状数组 ...

  5. [BZOJ5125]小Q的书架(决策单调性+分治DP+树状数组)

    显然有决策单调性,但由于逆序对不容易计算,考虑分治DP. solve(k,x,y,l,r)表示当前需要选k段,待更新的位置为[l,r],这些位置的可能决策点区间为[x,y].暴力计算出(l+r)/2的 ...

  6. BZOJ5125: [Lydsy1712月赛]小Q的书架(DP决策单调性)

    题意:N个数,按顺序划分为K组,使得逆序对之和最小. 思路:之前能用四边形不等式写的,一般网上都还有DP单调性分治的做法,今天也尝试用后者写(抄)了一遍.即: 分成K组,我们进行K-1次分治,get( ...

  7. bzoj 4815 小Q的表格 —— 反演+分块

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4815 思路就和这里一样:https://blog.csdn.net/leolyun/arti ...

  8. BZOJ 4017 小 Q 的无敌异或 ( 树状数组、区间异或和、区间异或和之和、按位计贡献思想 )

    题目链接 题意 : 中文题 分析 : 首先引入两篇写的很好的题解 题解一.题解二 听说这种和异或相关区间求和的问题都尽量按位考虑 首先第一问.按二进制位计贡献的话.那么对于第 k 位而言 其贡献 = ...

  9. BZOJ [Cqoi2017] 小Q的棋盘

    题解:枚举最后在哪里停止,然后剩下的步数/2 也就是找最大深度 枚举终止位置算是一种思路吧 #include<iostream> #include<cstdio> #inclu ...

随机推荐

  1. MySQL超时配置

    connect_timeout:连接响应超时时间.服务器端在这个时间内如未连接成功,则会返回连接失败. wait_timeout:连接空闲超时时间.与服务器端无交互状态的连接,直到被服务器端强制关闭而 ...

  2. Hive静态分区和动态分区

    一.静态分区 1.创建分区表 hive (default)> create table order_mulit_partition( > order_number string, > ...

  3. 特定条件下批量解压文件改变编码,顺便修改.so.0找不到等一些小问题

    直接结论: 1.linux解压文件乱码: unzip -O GBK *.zip 2.linux改变文件内容编码: 安装enca,下载地址:https://github.com/nijel/enca/i ...

  4. 【前端安全】JavaScript防流量劫持

    劫持产生的原因和方式 在网页开发的访问过程中,http是我们主要的访问协议.我们知道http是一种无状态的连接.即没有验证通讯双方的身份,也没有验证信息的完整性,所以很容易受到篡改.运营商就是利用了这 ...

  5. HTTPS实战之单向验证和双向验证

    转载自:https://mp.weixin.qq.com/s/UiGEzXoCn3F66NRz_T9crA 原创: 涛哥 coding涛 6月9日 作者对https 解释的入目三分啊 (全文太长,太懒 ...

  6. ESXi 嵌套KVM虚拟化 配置

    VMware ESXi5.x默认不支持嵌套虚拟化,需要修改相关配置才能支持   1.ESXi5.1主机开通ssh,修改VMware ESXi配置文件使之嵌套虚拟化. 在配置文件后面加入如下配置:vhv ...

  7. 了解Redis过期策略及实现原理

    我们在使用redis时,一般会设置一个过期时间,当然也有不设置过期时间的,也就是永久不过期. 当我们设置了过期时间,redis是如何判断是否过期,以及根据什么策略来进行删除的. redis设置过期时间 ...

  8. tomcat 启动时遇到org.apache.jasper.servlet.TldScanner.scanJars At least one JAR was scanned for TLDs yet contained no TLDs

    当发生这样的错误的时候 org.apache.jasper.servlet.TldScanner.scanJars At least one JAR was scanned for TLDs yet ...

  9. ES6知识整理(4)--数组的扩展

    最近工作比较忙,基本每天都会加班到很晚.处理一些客户端兼容问题以及提升用户体验的优化.也将近一周没更文了,现在继续es6的学习总结. 上篇回顾 ES6知识整理(三)--函数的扩展 扩展运算符 形式是3 ...

  10. AspectJ框架基于注解的AOP实现

    AspectJ的AOP实现:有两种方式,一种是基于XML配置文件,一种是基于注解的,由于注解更为常用,这里 这里只针对注解来学习. ---------------------------------- ...