二分法刷题

P1182 数列分段 Section II

  • 答案范围:左边界 l=max(Ai)(每段至少包含一个数,最大值不可能小于数组最大元素),右边界 r=∑Ai(不分段时的总和)。
  • 核心逻辑 :对于一个候选的最大值 mid,验证是否能将数组分成不超过 M 段 ,且每段和 ≤mid。
    • 若能分:说明 mid 可行,尝试更小的值(r=mid)
    • 若不能分:说明 mid 太小,尝试更大的值(l=mid+1)
cpp 复制代码
#include <bits/stdc++.h>
using namespace std;

typedef long long ll;
const int N = 100010;
ll a[N];
ll n, m;

bool check(ll x){
	ll cnt = 1;//刚开始 1 段 
	ll num = 0;
	for(int i =0; i<n; i++){
		if(num + a[i] > x){
			cnt ++;
			num = a[i];
			if(cnt > m) return false;
		}
		else {
			num +=a[i];
		}
		
	} 
	
	return true;
}

int main(){
	ios::sync_with_stdio(false); 
	cin.tie(nullptr); cout.tie(nullptr);
	cin >> n >> m;
	ll l = 0, r = 0;
	for(int i = 0; i<n; i++){
		cin >> a[i];
		l = max(l , a[i]);
		r += a[i];
	}
	
	while(l < r){
		ll mid = l+r >> 1;
		if(check(mid)) r = mid;//如果这个最大值能分成的段数<=mid;可以更小
		else l = mid + 1;		 
	}
	
	cout << r;
	return 0;
}

一、DP 状态定义

dp[i][j]将前 i 个数字,分成 j 段时,每段和的最大值的最小值


二、状态转移方程

要计算 dp[i][j],我们枚举最后一段的起点 kk < i):

  • 最后一段是 [k+1, i],和为 sum[i] - sum[k]sum 是前缀和数组)
  • k 个数字分成 j-1 段的最优解是 dp[k][j-1]
  • 那么以 k 为分界点时,当前方案的最大值是 max(dp[k][j-1], sum[i] - sum[k])
  • 我们要在所有 k 中取最小值,因此转移方程:

dp[i][j]=min0≤k<i{max(dp[k][j−1], sum[i]−sum[k])}


三、初始化

  • dp[i][1] = sum[i]:前 i 个数分成 1 段,最大值就是总和
  • 其余状态初始化为极大值(如 0x3f3f3f3f
cpp 复制代码
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;

const int N = 1010; // 注意:DP版时间复杂度O(n2m),仅适合n≤1000的小规模
const ll INF = 0x3f3f3f3f3f3f3f3f;

ll a[N], sum[N];
ll dp[N][N]; // dp[i][j]:前i个数分j段的最小最大值

int main() {
    ios::sync_with_stdio(false);
    cin.tie(0);

    int n, m;
    cin >> n >> m;
    for (int i = 1; i <= n; i++) {
        cin >> a[i];
        sum[i] = sum[i-1] + a[i]; // 前缀和
    }

    // 初始化
    memset(dp, 0x3f, sizeof(dp));
    for (int i = 1; i <= n; i++) {
        dp[i][1] = sum[i]; // 分1段,最大值就是总和
    }

    // 状态转移
    for (int j = 2; j <= m; j++) { // 枚举段数
        for (int i = j; i <= n; i++) { // 枚举前i个数(至少j个数才能分j段)
            for (int k = j-1; k < i; k++) { // 枚举最后一段的起点k,前j-1段最少要j-1个数 
                ll last = sum[i] - sum[k]; // 最后一段的和
                dp[i][j] = min(dp[i][j], max(dp[k][j-1], last));
            }
        }
    }

    cout << dp[n][m] << endl;
    return 0;
}
相关推荐
超级码力6662 小时前
【Latex文件架构】Latex文件架构模板
算法·数学建模·信息可视化
穿条秋裤到处跑2 小时前
每日一道leetcode(2026.04.29):二维网格图中探测环
算法·leetcode·职场和发展
Merlos_wind3 小时前
HashMap详解
算法·哈希算法·散列表
汉克老师3 小时前
GESP2025年3月认证C++五级( 第三部分编程题(1、平均分配))
c++·算法·贪心算法·排序·gesp5级·gesp五级
Yzzz-F5 小时前
Problem - 2205D - Codeforces
算法
智者知已应修善业6 小时前
【51单片机2个按键控制流水灯运行与暂停】2023-9-6
c++·经验分享·笔记·算法·51单片机
Halo_tjn6 小时前
Java Set集合相关知识点
java·开发语言·算法
生成论实验室7 小时前
《事件关系阴阳博弈动力学:识势应势之道》第四篇:降U动力学——认知确定度的自驱演化
人工智能·科技·神经网络·算法·架构
AI科技星7 小时前
全域数学·72分册:场计算机卷【乖乖数学】
算法·机器学习·数学建模·数据挖掘·量子计算
科研前沿8 小时前
镜像孪生VS视频孪生核心技术产品核心优势
大数据·人工智能·算法·重构·空间计算