详解数组分段和最大值最小问题(最小m段和问题)
数组分段和最大值最小问题(最小m段和问题)
问题描述
给定n个整数组成的序列,现在要求将序列分割为m段,每段子序列中的数在原序列中连续排列。如何分割才能使这m段子序列的和的最大值达到最小?
清洁工:假设有n个房间,清洁每个房间耗时用一个数组表示,10、20、30、40、50、60、70、80、90,安排m个清洁工,将连续的房间分成m份,每部分耗时求和,其最大值为此种分法的总耗时。求最快的耗时是多少。例如3个清洁工的话,10 20 30 40 50 | 60 70 | 80 90,此时是最快的,耗时为170。
装桶:把数据按顺序装入桶中,m即是给定的桶数,问桶的容量至少应该为多少才能恰好把这些数装入m个桶中
思路
先考虑最简单的情况,假设有 \(1\) 个数,只有一个桶,\(m=1\):至少需要容量为该数的值;
- 如果 \(n\) 个数,只有一个桶,\(m=1\):至少需要容量为 \(n\) 个数之和;
 - 如果 \(2\) 个数,两个桶,\(m=2\):至少需要容量为两数之间最大值
 - 如果 \(t\) 个数,两个桶,\(m=2\) 呢?
 
我们将 \(t\) 个数划分为两份,随机选一个划分位置 \(k\)
10, 20, 30, 40, | 50, 60, 70, 80, 90
就变成两部分: \(k\) 个数分到 1 个桶;\(t-k\) 个数分到一个桶
最少所需容量为 \(f(t,2) = min\{max[f(k,1),f(t-k,1)]\},1\leq k < t\),其中 \(f(t-k,1)\) 表示后 \(t-k\) 个数之和,表示为\(f(t-k,1) = f(t,1)-f(k,1)\)
- 现在考虑 \(n\) 个数,\(m\) 个桶:
 
我们同样将 \(n\) 个数划分为两份,即前 \(m-1\) 个桶和最后 \(1\) 个桶,随机选一个划分位置 \(k\):
\]
可以使用递归求解了,但是太耗时。我们使用动态规划填表就可以搞定了:
\]
可以用行表示对应的数字,列表示桶的数目,对1,2,3,4,5,两个桶:
| 桶1 | 桶2 | |
|---|---|---|
| 1 | 1 | 1 | 
| 2 | 3 | 2 | 
| 3 | 6 | 3 | 
| 4 | 10 | 7 | 
| 5 | 15 | 9 | 
coding:
#---python
#动态规划
# dp[i][j] = min(max(dp[k][j-1],dp[i][1]-dp[k][1]))
def dp_minMsum(nums,m):
    n = len(nums)
    dp = [[float('inf')]*(m+1) for _ in range(n+1)]
    #初始化
    for i in range(1,n+1):#只有一个桶
        dp[i][1] = sum(nums[:i])
    for j in range(1,m+1):#只有一个数
        dp[1][j] = nums[0]
    for i in range(2,n+1):
        for j in range(2,m+1):
            for k in range(1,i+1):
                dp[i][j] = min(dp[i][j],max(dp[k][j-1],dp[i][1]-dp[k][1]))
    return dp[-1][-1]
测试下:
>>> nums = [5,3,2,4,1]
>>> m = 2
>>> dp_minMsum(nums,m)
8
>>> nums = [10, 20, 30, 40, 50, 60, 70, 80, 90]
>>> m=3
>>> dp_minMsum(nums,m)
170
复杂度
时间复杂度:\(O(MN^2)\)
空间复杂度:\(O(NM)\)
二分查找优化
给定当前的桶容量,可以计算出需要的最少桶数:
如果需要的桶数量大于给定的桶数量k,说明桶容量太小了,增大容量;
如果计算需要的桶数量小于等于k,说明桶容量可能大了(也可能正好是要找的最小桶容量)
- 在给定容量时,求需要桶数
 - 二分搜索桶容量
 
coding:
#---python
#根据桶容量进行二分查找对应的桶数
def bs_minMsum(nums,m):
    low = max(nums)#容量下界 最大元素
    high = sum(nums)#容量上界 元素和
    while low < high:
        mid = low+(high-low)//2
        buckets = get_buckets(nums,mid)
        if buckets <= m:#该容量下桶数 小于 给定桶数。减小容量
            high = mid
        else:           #桶数 大于 给定桶数。增大容量
            low = mid +1
    return low
def get_buckets(nums,volume):
    buckets = 1
    ans = 0
    for num in nums:
        ans += num
        if ans > volume:
            buckets += 1
            ans = num
    return buckets
复杂度
时间复杂度:\(O(Nlog(\sum n))\),\(\sum n\) 为给定数组的元素和
空间复杂度:\(O(1)\)
详解数组分段和最大值最小问题(最小m段和问题)的更多相关文章
- linux进程的堆栈空间_代码段(指令,只读)、数据段(静态变量,全局变量)、堆栈段(局部变量)、栈【转】
		
转自:http://blog.csdn.net/gongweijiao/article/details/8207333 原文参见:http://blog.163.com/xychenbaihu@yea ...
 - 3532: [Sdoi2014]Lis 最小字典序最小割
		
3532: [Sdoi2014]Lis Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 865 Solved: 311[Submit][Status] ...
 - 黄聪:中国大陆的所有IP段,中国电信所有IP段、中国铁通所有IP段、中国网通所有IP段。
		
中国大陆的所有IP段,中国电信所有IP段.中国铁通所有IP段.中国网通所有IP段. 中国大陆的所有IP段: 47.153.128.0 47.154.255.25558.14.0.0 58.25.255 ...
 - 紫书 例题 11-2 UVa 1395(最大边减最小边最小的生成树)
		
思路:枚举所有可能的情况. 枚举最小边, 然后不断加边, 直到联通后, 这个时候有一个生成树.这个时候,在目前这个最小边的情况可以不往后枚举了, 可以直接更新答案后break. 因为题目求最大边减最小 ...
 - 【BZOJ-2229】最小割         最小割树(最大流+分治)
		
2229: [Zjoi2011]最小割 Time Limit: 10 Sec Memory Limit: 259 MBSubmit: 1565 Solved: 560[Submit][Status ...
 - BZOJ2229[Zjoi2011]最小割——最小割树
		
题目描述 小白在图论课上学到了一个新的概念——最小割,下课后小白在笔记本上写下了如下这段话: “对于一个图,某个对图中结点的划分将图中所有结点分成两个部分,如果结点s,t不在同一个部分中,则称这个划分 ...
 - POJ 3522 ——Slim Span——————【最小生成树、最大边与最小边最小】
		
Slim Span Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 7102 Accepted: 3761 Descrip ...
 - 【BZOJ2229】[Zjoi2011]最小割 最小割树
		
[BZOJ2229][Zjoi2011]最小割 Description 小白在图论课上学到了一个新的概念——最小割,下课后小白在笔记本上写下了如下这段话: “对于一个图,某个对图中结点的划分将图中所有 ...
 - 2229: [Zjoi2011]最小割(最小割树)
		
Description 小白在图论课上学到了一个新的概念——最小割,下课后小白在笔记本上写下了如下这段话: “对于一个图,某个对图中结点的划分将图中所有结点分成两个部分,如果结点s,t不在同一个部分中 ...
 
随机推荐
- MATLAB——时间,日期及显示格式
			
一.日期和时间 1.生成指定格式日期和时间 标准日期格式 2.获取当前时间的数值 >> datestr(now,) ans = -- :: >> datestr(now,'yy ...
 - 利用Python批量重命名文件夹下文件
			
#!/usr/bin/python # -*- coding: UTF-8 -*- # -*- coding:utf8 -*- import os from string import digits ...
 - Django-FileField与ImageField
			
1: 在views中用户表添加:class userinfo: avatar = models.FileField(upload_to='avatars/', default="/avata ...
 - Python学习-第五节:面向对象
			
概念: 核心是“过程”二字,“过程”指的是解决问题的步骤,即先干什么再干什么......,基于面向过程设计程序就好比在设计一条流水线,是一种机械式的思维方式.若程序一开始是要着手解决一个大的问题,面向 ...
 - Spring Boot整合Thymeleaf视图层
			
目录 Spring Boot整合Thymeleaf Spring Boot整合Thymeleaf 的项目步骤 Thymeleaf 语法详解 Spring Boot整合Thymeleaf Spring ...
 - 直播回顾 | IOT、AI、云计算等融合技术推进制造业产业转型(二)
			
3月31日,BoCloud博云.京东智联云.海尔集团联手,以“制造”到“智造”为主题,进行了IT赋能企业数字化转型实践分享. 博云售前解决方案架构师尹贺杰,京东云与AI企业云业务部高级业务技术经理吴世 ...
 - vue 动态添加body背景图片
			
<script> export default { data () { return { bodyBgImage: 'url(' + require('../asse ...
 - HDU1158:Employment Planning(暴力DP)
			
Employment Planning Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Othe ...
 - 浅谈动态规划(Dynamic Programming)
			
利用Leetcode#198打劫家舍 浅谈动态规划 Origin:https://leetcode-cn.com/problems/house-robber/ 题目本身不难,就是一个动态规划的问题.在 ...
 - 使用 python 创建&更改 word 文档
			
使用 python 修改 word 文档 说明:这个需求是老师想要一个自动识别 word 文档中指定位置的分数,并填入相应表格. 使用库 python-docx 的官方文档地址是:python-doc ...