AcWing 3549. 最长非递减子序列

AcWing 3549. 最长非递减子序列

一、题目描述

给定一个长度为 n 的数字序列 a1,a2,,an,序列中只包含数字 12

现在,你要选取一个区间 [l,r](1lrn),将 al,al+1,,ar 进行翻转,并且使得到的新数字序列 a 的最长非递减子序列的长度尽可能长。

请问,这个最大可能长度是多少?

一个非递减子序列是指一个索引为 p1,p2,,pk的序列,满足 p1<p2<<pk 并且 ap1ap2apk,其长度为 k

输入格式

第一行一个整数 n

第二行 n个空格隔开的数字 12,表示 a1,,an

输出格式

输出一个整数,表示得到的新数字序列 a 的最长非递减子序列的最大可能长度。

数据范围

对于 30% 的数据,1n100
对于 100% 的数据,1n106

本题读入数据规模较大,需注意优化读入。
C++ 尽量使用 scanf 读入,Java 尽量使用 BufferedReader 读入。

输入样例1

4
1 2 1 2

输出样例1

4

输入样例2

10
1 1 2 2 2 1 1 2 2 1

输出样例2

9

二、从简化题目出发(求只含12的序列长度)

首先我是考虑的求出该序列的最长非递减子序列

其实只有 2 种状态

  • 1111111
    只可能由 111111… 这种状态转移而来

  • 111111112222222222
    可能由 11111111… 转移而来
    也可能由 111112222… 转移而来

Code

#include <bits/stdc++.h>
using namespace std;

int main() {
    int n;
    cin >> n;
    int s1 = 0, s2 = 0;
    while (n--) {
        int x;
        cin >> x;
        if (x == 1) {
            s1++; // 如果当前数字是 1,则状态1的长度加1
        } else {
            s2 = max(s1 + 1, s2 + 1); // 如果当前数字为2,可能由两种状态转移而来
        }
    }
    cout << max(s1, s2) << endl;
    return 0;
}

三、回归题目(包括反转)

对于本题目而言,比上述的简单模型多加了一个条件:可以进行反转操作。

这就意味着,我们可以求一个形如:111111222222111111222222的子序列,然后进行反转操作让其变成 1111111112222222…的子序列

现在我们开始枚举状态:

  • 1111111....
    只能通过 111111…转移

  • (111111)2
    可以通过 1111111… 转移(仅限于转移到 111111111111..2 这种状态)
    也可通过 (11111)2222222…转移

  • (111111122222)11111
    可以通过 11111111222222… 转移(仅限于转移到 11111112222222..1 状态)
    也可以通过 (111111122222)11111…转移

  • (111111122222221111111)22222

    • 可以通过 (111112222)111… 转移(仅限于转移到 1111222222111111..2 状态)
    • 也可以通过 (111111122222221111111)22222… 转移

Code

#include <bits/stdc++.h>
using namespace std;

int main() {
    // 加快读入
    ios::sync_with_stdio(false), cin.tie(0);
    int s1 = 0, s2 = 0, s3 = 0, s4 = 0;
    int n;
    cin >> n;
    while (n--) {
        int x;
        cin >> x;
        if (x == 1) {
            s1++;
            s3 = max(s2 + 1, s3 + 1);
        } else {
            s2 = max(s1 + 1, s2 + 1);
            s4 = max(s3 + 1, s4 + 1);
        }
    }
    cout << max({s1, s2, s3, s4}) << endl;
    return 0;
}
posted @   糖豆爸爸  阅读(30)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 【自荐】一款简洁、开源的在线白板工具 Drawnix
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· Docker 太简单,K8s 太复杂?w7panel 让容器管理更轻松!
历史上的今天:
2022-05-06 POJ 3067 Japan
2022-05-06 POJ 2299 Ultra-QuickSort
2022-05-06 POJ 2155 Matrix
2020-05-06 在线pdf预览的办法
2018-05-06 如何利用外网服务器和本地内网机搭建反向隧道?
2013-05-06 在线JS/CSS/HTML压缩
2013-05-06 Kettle连接Mysql数据库异常
Live2D
点击右上角即可分享
微信分享提示