1013 机器分配(多重与分组背包问题扩展)

1. 问题描述:

总公司拥有M台相同的高效设备,准备分给下属的N个分公司。各分公司若获得这些设备,可以为国家提供一定的盈利。盈利与分配的设备数量有关。问:如何分配这M台设备才能使国家得到的盈利最大?求出最大盈利值。分配原则:每个公司有权获得任意数目的设备,但总台数不超过设备数M。

输入格式

第一行有两个数,第一个数是分公司数N,第二个数是设备台数M;接下来是一个N * M的矩阵,矩阵中的第 i 行第 j 列的整数表示第 i 个公司分配 j 台机器时的盈利。

输出格式

第一行输出最大盈利值;接下N行,每行有2个数,即分公司编号和该分公司获得设备台数。
答案不唯一,输出任意合法方案即可。

数据范围

1 ≤ N ≤ 10,
1 ≤ M ≤ 15

输入样例:

3 3
30 40 50
20 30 50
20 25 30

输出样例:

70
1 1
2 1
3 1
来源:https://www.acwing.com/problem/content/description/1015/

2. 思路分析:

对于这种有相应情景的题目,我们一般是根据自己的理解将其抽象为具体的模型,由题目可知这道题目属于选择性的组合问题,对于这种没有选择顺序的组合问题我们想到背包问题的模型,可以发现这道题目类似于多重背包问题和分组背包的组合(每个公司可以看成是一个组,组内的物品可以选择一定的数量),我们可以借助于多重背包与分组背包的思路来解决这个问题,可以将机器看成是一个物品,机器的总数量看成是背包的容量,最大的盈利可以看成在背包容量最多能够装满的前提下能够获得的最大价值,因为最终需要求解出具体的方案所以我们可以定义一个二维数组或者二维列表dp(在逆序递推具体的方案的时候需要使用到上一个状态的值所以我们需要使用二维数组来记录下对应状态的值,,如果求解的是方案数目可以声明一个一维数组逆序遍历体积),其中dp[i][j]表示前i家公司中已经分配j台机器能够获得的最大价值,类似于多重背包的枚举思路,我们可以使用三层循环枚举,第一层循环枚举所有的公司,第二层循环枚举机器数量(相当于是体积),第三层循环枚举第i家公司分配k台机器能够获得的最大价值,最终dp[n][m]为当前n家公司分配机器数量不超过m能够获得的最大盈利;因为需要求解任何一个合法的方案,所以我们可以从dp[n][m]倒着递推找到当前公司需要分配多少台机器才能从上一个状态到达当前的状态,可以使用一维数组或者列表记录下答案,最终递推到第一家公司就结束了。(顺序递推也是可以的)

3. 代码如下:

逆序递推:

if __name__ == '__main__':
    n, m = map(int, input().split())
    dp = [[0] * (m + 1) for i in range(n + 1)]
    # w为每家公司分配的机器数量对应的盈利, 二维列表前面加上一行0并且每一行的最前面加上一个0这样下标可以从1开始
    w = [[0] * (m + 1)]
    for i in range(1, n + 1):
        w.append([0] + list(map(int, input().split())))
    for i in range(1, n + 1):
        for j in range(m + 1):
            # 枚举第i家公司分配k台机器能够获得的最大价值
            for k in range(j + 1):
                dp[i][j] = max(dp[i][j], dp[i - 1][j - k] + w[i][k])
    print(dp[n][m])
    j = m
    res = [0] * (n + 1)
    # 逆序求解方案这样才可以从dp[n][m]逆推寻找上一个转移过来的合法方案
    for i in range(n, 0, -1):
        for k in range(0, j + 1):
            if dp[i][j] == dp[i - 1][j - k] + w[i][k]:
                res[i] = k
                j -= k
                break
    for i in range(1, n + 1): print(i, res[i])

顺序递推:

from typing import List

class Solution:
    def process(self):
        n, m = map(int, input().split())
        dp = [[0] * (m + 10) for i in range(n + 10)]
        w = [[0] * (m + 1)]
        for i in range(n):
            w.append([0] + list(map(int, input().split())))
        for i in range(n, 0, -1):
            for j in range(m + 1):
                for k in range(j + 1):
                    dp[i][j] = max(dp[i][j], dp[i + 1][j - k] + w[i][k])
        print(dp[1][m])
        # 因为是从逆序递推过来的, 所以这里顺序获得每一家公司分配的机器数量
        k = m
        for i in range(1, n + 1):
            for j in range(k + 1):
                # 第i家公司分配j台, 后面的公司分配k - j台
                if dp[i][k] == dp[i + 1][k - j] + w[i][j]:
                    # 输出对应的公司编号以及分配的机器数量
                    print(i, j, sep=" ")
                    k -= j
                    break

if __name__ == "__main__":
    Solution().process()
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值