最大M子段和

题目模型

  • N个整数组成的序列 \(a_1,a_2,a_3,…,a_n\) ,将这N个数划分为互不相交的M个子段,并且这M个子段的和是最大的。

问题分析

  • 方法一

    • 看到序列,我们首先要尝试用线性dp去处理,线性dp经典状态定义:f[i][j]i一般表示序列的前i个元素,j表示限制,这里表示划分了j个不相交的子段,我们还需要对i进行进一步的定义,即是否包含第i项,因为对当前元素a[i]来说,要么单独成一个子段,要么和最后一个子段合并,所以必须包含第i个元素。

    • 动态转移方程:dp[i][j]=max(dp[i-1][j],dp[k][j-1])+a[i] (j-1<=k<i)

    • Code

      #include <bits/stdc++.h>
      const int maxn = 1e3+3,Inf=0x3f3f3f3f;
      typedef long long LL;
      int a[maxn],dp[maxn][maxn];
      void Solve(){
      int n,m;scanf("%d%d",&n,&m);
      for(int i=1;i<=n;++i)
      scanf("%d",&a[i]);
      for(int i=1;i<=n;++i){//前i个元素
      for(int j=1;j<=std::min(i,m);++j){//划分出j个子段
      if(i==j)dp[i][j]=dp[i-1][j-1]+a[i];//显然
      else{
      int temp=dp[i-1][j];//把a[i]直接并到最后一子段
      for(int k=j-1;k<i;++k)//枚举上一个状态的最后一个子段的右端点,a[i]单独作为一个子段
      temp=std::max(temp,dp[k][j-1]);
      dp[i][j]=temp+a[i];
      }
      }
      }
      int ans=-Inf;
      for(int i=m;i<=n;++i)
      ans=std::max(ans,dp[i][m]);
      printf("%d\n",ans);
      }
      int main(){
      Solve();
      return 0;
      }
    • 时间效率为:\(O(n^3)\) ,空间效率为:\(O(m*n)\)。

  • 方法二

    • 我们尝试对方法一的dp阶段和状态进行修改, 即把子段限制数M作为阶段,即状态dp[i][j]表示把序列前j分成i个子段且包含a[j]的最大子段和。

    • 动态转移方程有:dp[i][j]=max(dp[i][j-1],dp[i-1][k])+a[j] (i-1<=k<j)

      • dp[i][j-1]+a[i]:表示合并到最后一个子段里

      • dp[i-1][k]+a[i]:表示前k元素挑出k个子段,所以k>=j-1,然后a[i]单独的子段。

      • 此动态转移方程同样满足无后效性和最优子结构。

      • 我们把问题的所有状态记录下来形成一个二维矩阵,显然当前状态只跟它上一行和左边的状态有关,我们可以把空间效率压掉以为变成 \(O(n)\) 。

      • 同时上一行的状态只有在当前状态前面的最大值对转移有用,我们可以在遍历当前行时维护一下上一行的最大值,这样时间效率就压掉了一个n,变成\(O(n*m)\)。

      • Code

        #include <bits/stdc++.h>
        typedef long long LL;
        const int maxn = 1e4+5;
        const LL Inf=0x3f3f3f3f3f3f3f3f;
        LL a[maxn],dp[2][maxn];
        void Solve(){
        int n,m;scanf("%d%d",&n,&m);
        for(int i=1;i<=n;++i)
        scanf("%lld",&a[i]);
        int k=1;//滚动数组指针,k表示当前行,!k表示上一行
        for(int i=1;i<=m;++i,k=!k){//枚举区间个数
        LL Max=-Inf;
        for(int j=i;j<=n;j++){
        Max=std::max(Max,dp[!k][j-1]);//记录前j-1,分成i-1个区间时最大值
        if(i==j)
        dp[k][j]=dp[!k][j-1]+a[j];
        else//要么是a[j]单独成一个区间,此时为Max+a[j],或者直接合并为dp[k][j-1]+a[j]
        dp[k][j]=std::max(Max,dp[k][j-1])+a[j];
        }
        } LL ans=-Inf;
        for(int i=m;i<=n;++i)//!k行才记录的是第m行的状态
        ans=std::max(ans,dp[!k][i]);
        printf("%lld\n",ans);
        }
        int main(){
        Solve();
        return 0;
        }
  • 方法三

    • 方法二把空间优化到线性,时间优化到\(O(n^2)\) ,但如果 \(n\) 和 \(m\) 高达 \(10^5\)显然dp是无法解决了。比如:51nod 1115 最大M子段和 V3

    • 对这个问题我们先对原数组进行处理,然后利用可撤销贪心解决。

      • 首先把原数组连续的正数加起来变成一个数,连续的负数加起来变成一个数,0加到哪里都一样,这样我们就得到了一个正负交替的环形序列。

      • 如果新的环形序列里正数的个数为cnt,所有正数之和为ans则存在两种情况:

        1. \(cnt<=M\),因为可以选空,所以答案就是所有正数之和ans
        2. \(cnt>M\) ,此时ans包含了cnt个正数区间,所以我们需要通过操作减少cnt-M个区间。
      • 对于情况2,我们可以通过两种操作减少区间:

        1. 删除一个正数(ans包含所有正数之和,删除一个正数相当于减少了一个区间).
        2. 将一个负数与它两边的正数合并,相当于把两个正数区间合并成了一个区间,也减少了一个区间。
      • 很容易能想到一个贪心思想,即每次选择最小的正数删除一个区间,或选择最大一个负数与它两边的正数合并减少一个区间。

      • 但这个贪心是不正确的,因为删除一个较小的正数是在减少一个区间的最优,但有可能不删除这个正数,而是通过若干次合并在减少多个区间。

        • 例如:序列10 -4 3 -4 8 -100其中M=1,根据刚才的贪心策略,我们先删除3,减少了一个区间,然后最大的负数为-4,此时因为3已经被删除-4就无法合并了。为了解决类似的问题,我们要用到可撤销的贪心思想。
      • 实现步骤:

        1. 把序列中正数均变成负数,并把数都压入大根堆。
        2. 套用可撤销贪心思想,做cnt-M次即可,具体见代码。
      • Code

        #include <bits/stdc++.h>
        typedef long long LL;
        const int maxn=2e5+5;
        struct Node{
        int id;
        LL w;
        Node(){};
        Node(int x,LL y){id=x;w=y;}
        bool operator <(const Node &a)const{
        return w<a.w;
        }
        };
        int n,m,N=0,L[maxn],R[maxn];
        LL ans=0,a[maxn],b[maxn];
        std::priority_queue<Node> q;
        bool flag[maxn];
        void Init(){
        scanf("%d%d",&n,&m);
        for(int i=1;i<=n;++i){
        scanf("%lld",&a[i]);//原始数组
        if(!N || (a[i]>=0)!=(b[N]>=0))
        b[++N]=a[i];//新数组,上个数和当前数符号不一样节点++
        else
        b[N]+=a[i];//符号一样,累加
        }
        if((b[1]>=0)==(b[N]>=0))//因为是环,b[1]和b[N]同号就合并
        b[1]+=b[N--];//合并后新数组个数减一,记得N--
        }
        void Solve(){
        Init();
        int tot=0;//记录正数个数
        for(int i=1;i<=N;++i){//遍历新数组,压入大根堆
        L[i]=i-1; R[i]=i+1;//初始化i的左右邻居
        if(b[i]>0){//正数累加到答案,然后变负
        ans+=b[i]; b[i]=-b[i]; tot++;
        }
        q.push(Node(i,b[i]));
        }
        R[N]=1; L[1]=N;//注意是环形
        if(m>=tot){//正数个数小于m,则全部选
        printf("%lld\n",ans);return;
        }
        m=tot-m;//正数大于m则合并或删除tot-m个区间
        while(m--){
        Node t=q.top(); q.pop();
        int i=t.id;
        if(flag[i]){++m; continue;}
        else{//可撤销贪心
        ans+=b[i];
        flag[L[i]]=1;
        flag[R[i]]=1;
        b[i]=b[L[i]]+b[R[i]]-b[i];
        R[L[L[i]]]=i;
        L[R[R[i]]]=i;
        q.push(Node(i,b[i]));//压入新点
        L[i]=L[L[i]];
        R[i]=R[R[i]];
        }
        }
        printf("%lld\n",std::max(0LL,ans));
        }
        int main(){
        Solve();
        return 0;
        }

最大子段和之M子段和的更多相关文章

  1. (最大连续和/最大子段和) P1115 最大子段和 洛谷

    题目描述 给出一段序列,选出其中连续且非空的一段使得这段和最大. 输入输出格式 输入格式: 第一行是一个正整数NN,表示了序列的长度. 第二行包含NN个绝对值不大于1000010000的整数A_iAi ...

  2. XCOJ 1103 (LCA+树链最大子段和)

    题目链接: http://xcacm.hfut.edu.cn/problem.php?id=1103 题目大意:链更新.链查询,求树链的最大子段和.(子段可以为空) 解题思路: 将所有Query离线存 ...

  3. hdu1003 dp(最大子段和)

    题意:给出一列数,求其中的最大子段和以及该子段的开头和结尾位置. 因为刚学过DP没几天,所以还会这题,我开了一个 dp[100002][2],其中 dp[i][0] 记录以 i 为结尾的最大子段的和, ...

  4. HDOJ-1003 Max Sum(最大连续子段 动态规划)

    http://acm.hdu.edu.cn/showproblem.php?pid=1003 给出一个包含n个数字的序列{a1,a2,..,ai,..,an},-1000<=ai<=100 ...

  5. 最大子段和(c++)

    // 最大子段和.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #include<iostream> using namesp ...

  6. 51Node 1065----最小正子段和

    51Node  1065----最小正子段和 N个整数组成的序列a[1],a[2],a[3],…,a[n],从中选出一个子序列(a[i],a[i+1],…a[j]),使这个子序列的和>0,并且这 ...

  7. 最大M子段和 V2

    51nod1053 这题还是我们熟悉的M子段和,只不过N,M<=50000. 这题似乎是一个堆+链表的题目啊 开始考虑把所有正数负数锁在一起. 比如: 1 2 3 -1 –2 -3 666 缩成 ...

  8. 51nod 循环数组最大子段和

    http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1050 对于普通的数组,只要求一次最大子段和即可.但是这题是可以循环的,所 ...

  9. [日常训练]最大M子段和

    Description 在长度为的序列中选出段互不相交的子段,求最大字段和. Input 第一行两个整数. 第二行个整数. Output 一行一个整数表示最大值. Sample Input 5 2 1 ...

随机推荐

  1. 预处器的对比——Sass、LESS和Stylus

    http://www.w3cplus.com/css/sass-vs-less-vs-stylus-a-preprocessor-shootout.html

  2. 【新阁教育】基于EtherNet/IP实现欧姆龙NX系列PLC通信

    1.引言 工业以太网协议 (Ethernet/IP) 是由ODVA所开发并得到了罗克韦尔自动化的强大支持.它使用已用于ControlNet和DeviceNet的控制和信息协议 (CIP) 为应用层协议 ...

  3. python文件的读写权限以及相关应用read、write和文件指针

    f=open('ceshi.txt','a',encoding='utf-8')r=open('ceshi.txt','r',encoding='utf-8')上面的2种写法可以用with来写:wit ...

  4. Fiddler无法抓取web项目中的http请求解决方案

    问题:webform项目中对接API使用Fiddler无法获取该API的请求,该webform比较老的一个项目, 同一个API写在控制台可以抓取到请求,用web项目放在本地IIS却不行,使用IIS E ...

  5. 《Java从入门到失业》第四章:类和对象(4.5):包

    4.5包 前面我们已经听过包(package)这个概念了,比如String类在java.lang包下,Arrays类在java.util包下.那么为什么要引入包的概念呢?我们思考一个问题:java类库 ...

  6. Redis学习(二)redis的特点

    一.Redis的特性 Redis是基于内存,常用作于缓存的技术 Redis实现的是分布式缓存,如果有多台实例(机器)的话,每个实例都共享一份缓存,缓存具有一致性. 常见的性能问题一般都是由于数据库(磁 ...

  7. 并发编程(二)Java中的多线程

    一.创建多线程 创建多线程有以下几种方法: 继承Thread,重写run方法 实现Runnable接口,重写run方法[无返回值] 实现Callable接口,重写call方法[有返回值] 继承Thre ...

  8. Redis利用,攻击内网(ssrf)

    Redis语法 REmote DIctionary Server(Redis) 是一个由Salvatore Sanfilippo写的key-value存储系统. Redis是一个开源的使用ANSI C ...

  9. mac电脑,charles,安卓手机如何配置代理,以及配置代理之后无法上网。已解决

    设备: 电脑:mac book pro 手机:小米10 charles:4.5.6 方法一: 步骤: 首先确保电脑,手机在同一局域网, 1. charles设置代理:proxy -> proxy ...

  10. vue项目中视频播放结束返回首页出现1秒左右的白屏问题

    vue项目的性能优化问题,一直以来都是大家比较关注的. 近日负责的项目中,使用了SignalR实时通讯,客户端中点击发起播放视频的请求到服务器,服务器接到请求后再调用前端的播放视频方法,以此来达到播放 ...