很有(\(bu\))质(\(hui\))量(\(xie\))的一个题目。

第一问:求最少改变几个数能把一个随机序列变成单调上升序列。

\(Solution:\)似乎是一个结论?如果两个数\(A_i\)和\(A_j\)可以保留(\(i > j, A_i > A_j\)),即中间其他数都可以通过修改成为\([A_i, A_j]\)区间内的一个数,那么一定有\(i - j <= A_i - A_j\),即\(A_i - i >= A_j - j\)。这个东西我们可以设为数列\(B\),求一个最长不下降子序列就可以了。

(其实我中间智障了写成了最长上升子序列居然还有\(90ptshhhhh\))

第二问:结论看这里。有了第一问的铺垫其实并不难想,但是问题在于可以有很多种最长不下降子序列,该怎么办?

我们考虑对答案\(DP\),设\(f_x\)为前\(x\)个数有序化的最小代价(其中\(x\)一定是一个子序列内的点),然后来一发轻松愉快的\(DP\)。由于数据完全随机,所以可以剪枝过去。(虽然就是不随机恐怕也很难卡满就是了\(QwQ\))

#include <bits/stdc++.h>
using namespace std; #define int long long
const int N = 35000 + 5;
const int INF = 0x3f3f3f3f; int n, A[N], B[N], Longest[N]; vector <int> q, G[N];
vector <int> :: iterator it; int solve1 () {
for (int i = 1; i <= n + 1; ++i) {
if (q.empty () || B[i] >= q[q.size () - 1]) {
q.push_back (B[i]); Longest[i] = q.size ();
} else {
it = upper_bound (q.begin (), q.end (), B[i]);
*it = B[i]; Longest[i] = it - q.begin () + 1;
}
G[Longest[i]].push_back (i);
}
// Longest[i] : 以 i 结尾的最长不下降子序列
return q.size () - 1; // 可以保留的数的个数
} int dp[N], hl[N], hr[N]; // hl[i] -> 从 l 到 i 都选择变成左端点值的代价, hr 同理 int solve2 () {
memset (dp, 0x3f, sizeof (dp));
dp[0] = 0; G[0].push_back (0);
for (int v = 1; v <= n + 1; ++v) {
for (int i = 0; i < G[Longest[v] - 1].size (); ++i) {
int u = G[Longest[v] - 1][i];
// cout << "u = " << u << " v = " << v << endl;
if (v < u || B[v] < B[u]) continue;
hl[u] = hr[v] = 0;
for (int k = u + 1; k < v; ++k) {
hl[k] = hl[k - 1] + abs (B[k] - B[u]);
}
for (int k = v - 1; k > u; --k) {
hr[k] = hr[k + 1] + abs (B[k] - B[v]);
}
for (int k = u; k < v; ++k) {
dp[v] = min (dp[v], dp[u] + hl[k] + hr[k + 1]);
}
}
}
return dp[n + 1];
} signed main () {
// freopen ("data.in", "r", stdin);
// freopen ("data.out", "w", stdout);
cin >> n;
B[0] = -INF, B[n + 1] = INF;
for (int i = 1; i <= n; ++i) {
cin >> A[i]; B[i] = A[i] - i;
}
cout << n - solve1 () << endl; // 对数列 B 做最长不下降子序列
cout << solve2 () << endl; // ans
}

最新文章

  1. iOS开发系列--Swift语言
  2. JS this指向
  3. Java 之 异常处理
  4. 关于firstChild,firstElementChild和children
  5. Mybatis批量添加对象List
  6. HDU 1054 Strategic Game(最小点覆盖+树形dp)
  7. 移动端富文本编辑器artEditor
  8. iOS开发- 蓝牙后台接收数据(BLE4.0)
  9. 最简便的MySql数据库备份方法
  10. Cocos2d-x 3.0 场景切换
  11. 【6】python核心编程 第九章-文件和输入输出
  12. S - stl 的mapⅠ
  13. 聊天界面使用IQKeyboardManager导航栏及整个页面上移的解决方法
  14. BZOJ2726: [SDOI2012]任务安排
  15. 【Android 应用开发】AndroidUI设计之 布局管理器 - 详细解析布局实现
  16. C++中几种输入输出cin、cin.getline()、getline()、sscanf()、sprintf()、gets()等
  17. 1 Introduction
  18. element 树形菜单加title
  19. Photoshop影像匀色技术
  20. CF815C Karen and Supermarket

热门文章

  1. firefox PAC代理
  2. &quot;alert(1) to win&quot; writeup
  3. OI模板のpoke流[大型考试复习必备/kl]
  4. 平衡树(Splay、fhq Treap)
  5. 我的python学习之旅——安装python
  6. Python验证码登录(Tesseract安装配置)
  7. DockerFile 编译语法详解
  8. QThread::wait(),一直以来我以为它阻塞的是QThread对象,可是我现在明白,原来阻塞的是这个对象所在的线程(通常是主线程)——所有事情源于 QThread 的事件循环——如果使用继承QThread这一方法,QThread::quit()没有效果,因为这个线程根本就不需要事件循环
  9. Collection接口的子接口——List接口
  10. 简单搭建http服务器-HttpListener使用