IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> CCF202109-2 非零段划分(100分)【序列处理】 -> 正文阅读

[数据结构与算法]CCF202109-2 非零段划分(100分)【序列处理】

试题编号: 202109-2
试题名称: 非零段划分
时间限制: 1.0s
内存限制: 512.0MB
问题描述:

题目描述
A1,A2,…,An是一个由 n 个自然数(非负整数)组成的数组。我们称其中 Ai,…,Aj 是一个非零段,当且仅当以下条件同时满足:
·1≤i≤j≤n;
·对于任意的整数 k,若 i≤k≤j,则 Ak>0;
·i=1 或 Ai-1=0;
·j=n 或 Aj+1=0。
下面展示了几个简单的例子:
·A = [3, 1, 2, 0, 0, 2, 0, 4, 5, 0, 2] 中的 4 个非零段依次为 [3, 1, 2]、[2]、[4, 5] 和 [2];
·A = [2, 3, 1, 4, 5] 仅有 1 个非零段;
·A = [0, 0, 0] 则不含非零段(即非零段个数为 0)。
现在我们可以对数组 A 进行如下操作:任选一个正整数 p,然后将 A 中所有小于 p 的数都变为 0。试选取一个合适的 p,使得数组 A 中的非零段个数达到最大。若输入的 A 所含非零段数已达最大值,可取 p = 1,即不对 A 做任何修改。

输入格式
从标准输入读入数据。

输入的第一行包含一个正整数 n。

输入的第二行包含 n 个用空格分隔的自然数 A1, A2, … , An。

输出格式
输出到标准输出。

仅输出一个整数,表示对数组 A 进行操作后,其非零段个数能达到的最大值。

样例1输入
11
3 1 2 0 0 2 0 4 5 0 2
样例1输出
5
样例1解释
p = 2 时,A = [3, 0, 2, 0, 0, 2, 0, 4, 5, 0, 2],5 个非零段依次为 [3]、[2]、[2]、[4, 5] 和 [2];此时非零段个数达到最大。

样例2输入
14
5 1 20 10 10 10 10 15 10 20 1 5 10 15
样例2输出
4
样例2解释
p = 12 时,A = [0, 0, 20, 0, 0, 0, 0, 15, 0, 20, 0, 0, 0, 15],4 个非零段依次为 [20]、[15]、[20] 和 [15];此时非零段个数达到最大。

样例3输入
3
1 0 0
样例3输出
1
样例3解释
p = 1 时,A = [1, 0, 0],此时仅有 1 个非零段 [1],非零段个数达到最大。

样例4输入
3
0 0 0
样例4输出
0
样例4解释
无论 p 取何值,A 都不含有非零段,故非零段个数至多为 0。

子任务
70% 的测试数据满足 n ≤ 1000;

全部的测试数据满足 n≤5×105,且数组 A 中的每一个数均不超过 104

问题链接CCF202109-2 非零段划分
问题简述:(略)
问题分析
  给出的题解程序,当n≤1000时,采用枚举法,结果是没有问题;当n>1000时,采用3分法求极值的方法,不知道逻辑是否严密?
  比较一下100分程序和70程序,就可以知道,似乎100分程序的逻辑是不严密的。当n≤1000时也采用了3分法,只能得70分,似乎说明逻辑不够严密。
程序说明:解题程序根据太理陈同学的100分解题程序改写而来。
参考链接:(略)
题记:(略)

100分的C++语言程序(集合+三分法+枚举)如下:

/* CCF202109-2 非零段划分 */

#include <iostream>
#include <cstring>

using namespace std;

const int LR = 50;
const int N = 500000 + 1;
const int M = 10000 + 1;
int n, a[N], s[M];

int mycount(int p)
{
    int cnt = 0, flag = 1;
    for (int i = 1; i <= n; i++)
        if (flag) {
            if (a[i] >= p) cnt++, flag = 0;
        } else {
            if (a[i] < p) flag = 1;
        }
    return cnt;
}

int main()
{
    int maxa = 0, ans = 0;
    memset(s, 0, sizeof s);
    cin >> n;
    for (int i = 1; i <= n; i++) {
        cin >> a[i];
        maxa = max(maxa, a[i]);
        s[a[i]] = 1;
    }

    int scnt = 0;
    for (int i = 1; i <= M; i++)
        if (s[i]) s[scnt++] = i;

    int l = 0, r = scnt - 1;
    ans = max(mycount(s[l++]), mycount(s[r--]));
    while (r - l > LR) {
        int ll = l + (r - l) / 3;
        int rr = r - (r - l) / 3;
        if (mycount(s[ll]) < mycount(s[rr])) l = ll;
        else r = rr;
    }
    for(int i = l; i <= r; i++)
        ans = max(ans, mycount(s[i]));

    cout << ans << endl;

    return 0;
}

100分的C++语言程序(三分法+枚举)如下:

/* CCF202109-2 非零段划分 */

#include <iostream>

using namespace std;


const int N = 500000 + 1;
int n, a[N];

int mycount(int p)
{
    int cnt = 0, flag = 1;
    for (int i = 1; i <= n; i++)
        if (flag) {
            if (a[i] >= p) cnt++, flag = 0;
        } else {
            if (a[i] < p) flag = 1;
        }
    return cnt;
}

int main()
{
    int maxa = 0, ans = 0;
    cin >> n;
    for (int i = 1; i <= n; i++) {
        cin >> a[i];
        maxa = max(maxa, a[i]);
    }
    if (n <= 1000)
        for (int i = 1; i <= maxa; i++) ans = max(ans, mycount(i));
    else {
        int l = 1, r = maxa;
        while (r - l > 4) {
            int ll = l + (r - l + 1) / 3;
            int rr = r - (r - l + 1) / 3;
            if (mycount(ll) < mycount(rr)) l = ll;
            else r = rr;
        }
        for(int i = l; i <= r; i++)
            ans = max(ans, mycount(i));
    }

    cout << ans << endl;

    return 0;
}

70分的C++语言程序(三分法)如下:

/* CCF202109-2 非零段划分 */

#include <iostream>
#include <set>

using namespace std;

const int N = 500000 + 1;
int n, a[N];

int mycount(int p)
{
    int cnt = 0, flag = 1;
    for (int i = 1; i <= n; i++)
        if (flag) {
            if (a[i] >= p) cnt++, flag = 0;
        } else {
            if (a[i] < p) flag = 1;
        }
    return cnt;
}

int main()
{
    int maxa = 0, ans = 0;
    cin >> n;
    for (int i = 1; i <= n; i++) {
        cin >> a[i];
        maxa = max(maxa, a[i]);
    }
    int l = 1, r = maxa;
    while (r - l > 4) {
        int ll = l + (r - l + 1) / 3;
        int rr = r - (r - l + 1) / 3;
        if (mycount(ll) < mycount(rr)) l = ll;
        else r = rr;
    }
    for(int i = l; i <= r; i++)
        ans = max(ans, mycount(i));

    cout << ans << endl;

    return 0;
}
  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2021-10-04 13:05:02  更:2021-10-04 13:05:28 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/26 4:45:14-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码