Python 实现的运筹优化系统数学建模详解(0-1规划指派问题)

news2025/4/15 11:52:32

一、引言

        在数学建模的广阔领域中,指派问题作为一类经典且重要的组合优化问题,频繁出现在各类实际场景里。例如,在人力资源管理中,如何将不同技能水平的员工高效地分配到各个项目,以实现项目成本最小化或收益最大化;在物流配送中,怎样把不同路线的运输任务合理指派给各车辆,从而降低运输总成本等。0-1 规划作为解决这类问题的有力工具,通过巧妙设定决策变量仅能取 0 或 1,精准地模拟了 “是否指派” 的二元决策过程。Python 凭借其强大的库资源,尤其是pulp库,为实现 0-1 规划解决指派问题提供了便捷且高效的途径。本文将深入探讨 0-1 规划在数学建模指派问题中的应用,并结合 Python 代码详细阐述其实现细节,帮助读者全面掌握这一实用技术。

二、0-1 规划在指派问题中的原理

(一)指派问题的数学模型

        假设存在m个员工和n个项目(在一般情况下,m和n可相等也可不相等,但为了简化理解,先以\(m = n\)为例)。我们定义一个决策变量\(x_{ij}\),当员工i被指派到项目j时,\(x_{ij}=1\);否则,\(x_{ij}=0\)。同时,已知员工i完成项目j所需的成本为\(c_{ij}\)。那么,指派问题的目标就是找到一组\(x_{ij}\)的值,使得总成本\(Z=\sum_{i = 1}^{m}\sum_{j = 1}^{n}c_{ij}x_{ij}\)最小化。

此外,还需要满足以下约束条件:

  1. 每个员工只能被分配到一个项目,即\(\sum_{j = 1}^{n}x_{ij}=1\),\(i = 1,2,\cdots,m\)。
  2. 每个项目只能由一个员工来完成,即\(\sum_{i = 1}^{m}x_{ij}=1\),\(j = 1,2,\cdots,n\)。

(二)0-1 规划如何适配指派问题

        在这个指派问题中,决策变量\(x_{ij}\)的取值特性完全符合 0-1 规划的要求。通过 0-1 规划的方法,我们可以在满足上述约束条件的众多可能的指派组合中,搜索出使得总成本最小的最优组合。这就是 0-1 规划在指派问题中的核心应用原理。它将实际的人员与任务匹配问题转化为一个数学优化问题,利用数学工具找到最佳解决方案。

三、Python 实现 0-1 规划解决指派问题的代码解析

(一)代码整体结构

python

from pulp import LpMinimize, LpProblem, LpStatus, lpSum, LpVariable, value


def solve_assignment_problem():
    # 获取用户输入的员工数量和项目数量
    num_employees = int(input("请输入员工的数量: "))
    num_projects = int(input("请输入项目的数量: "))

    # 获取用户输入的目标函数系数
    costs = []
    for i in range(num_employees):
        for j in range(num_projects):
            cost = float(input(f"请输入员工 {i + 1} 分配到项目 {j + 1} 的成本: "))
            costs.append(cost)

    # 获取用户输入的约束条件数量
    num_constraints = int(input("请输入额外约束条件的数量: "))

    constraint_coeffs_list = []
    constraint_rhs_list = []
    constraint_types = []

    # 获取每个约束条件的系数、右侧常数和约束类型
    for k in range(num_constraints):
        print(f"正在输入第 {k + 1} 个约束条件的信息:")
        constraint_coeffs = []
        for i in range(num_employees * num_projects):
            coeff = float(input(f"请输入第 {k + 1} 个约束条件中第 {i + 1} 个变量的系数: "))
            constraint_coeffs.append(coeff)
        constraint_rhs = float(input(f"请输入第 {k + 1} 个约束条件的右侧常数: "))
        constraint_type = input(f"请输入第 {k + 1} 个约束条件的类型(输入 '<=' 或 '='): ")
        constraint_coeffs_list.append(constraint_coeffs)
        constraint_rhs_list.append(constraint_rhs)
        constraint_types.append(constraint_type)

    # 创建问题
    problem = LpProblem("Assignment_Problem", LpMinimize)

    # 定义决策变量(0 - 1规划)
    assignments = [LpVariable(f"x{i}", lowBound=0, cat='Binary') for i in range(num_employees * num_projects)]

    # 定义目标函数
    problem += lpSum([costs[i] * assignments[i] for i in range(num_employees * num_projects)])

    # 添加默认约束条件
    # 每一个人只能分配一个项目
    for i in range(num_employees):
        problem += lpSum([assignments[i * num_projects + j] for j in range(num_projects)]) <= 1

    # 每一个项目只能被一个员工分配
    for j in range(num_projects):
        problem += lpSum([assignments[i * num_projects + j] for i in range(num_employees)]) == 1

    # 添加额外约束条件
    for constraint_coeffs, constraint_rhs, constraint_type in zip(constraint_coeffs_list, constraint_rhs_list,
                                                                  constraint_types):
        if constraint_type == '<=':
            problem += lpSum([coeff * var for coeff, var in zip(constraint_coeffs, assignments)]) <= constraint_rhs
        elif constraint_type == '=':
            problem += lpSum([coeff * var for coeff, var in zip(constraint_coeffs, assignments)]) == constraint_rhs
        else:
            print(f"不支持的约束类型 '{constraint_type}',忽略该约束。")

    # 求解问题
    problem.solve()

    # 输出结果
    print(f"The status of the problem is {LpStatus[problem.status]}")
    print(f"The optimal value of the problem is {value(problem.objective)}")
    for i in range(num_employees * num_projects):
        print(value(assignments[i]), end=' ')
        if (i + 1) % num_projects == 0:
            print()

    return value(problem.objective), [value(var) for var in assignments]


if __name__ == "__main__":
    solve_assignment_problem()

        这段代码整体实现了通过与用户交互,获取指派问题的相关信息,构建 0-1 规划模型并求解,最后输出最优解和目标函数最优值的功能。

(二)输入获取部分

python

    # 获取用户输入的员工数量和项目数量
    num_employees = int(input("请输入员工的数量: "))
    num_projects = int(input("请输入项目的数量: "))

    # 获取用户输入的目标函数系数
    costs = []
    for i in range(num_employees):
        for j in range(num_projects):
            cost = float(input(f"请输入员工 {i + 1} 分配到项目 {j + 1} 的成本: "))
            costs.append(cost)

    # 获取用户输入的约束条件数量
    num_constraints = int(input("请输入额外约束条件的数量: "))

    constraint_coeffs_list = []
    constraint_rhs_list = []
    constraint_types = []

    # 获取每个约束条件的系数、右侧常数和约束类型
    for k in range(num_constraints):
        print(f"正在输入第 {k + 1} 个约束条件的信息:")
        constraint_coeffs = []
        for i in range(num_employees * num_projects):
            coeff = float(input(f"请输入第 {k + 1} 个约束条件中第 {i + 1} 个变量的系数: "))
            constraint_coeffs.append(coeff)
        constraint_rhs = float(input(f"请输入第 {k + 1} 个约束条件的右侧常数: "))
        constraint_type = input(f"请输入第 {k + 1} 个约束条件的类型(输入 '<=' 或 '='): ")
        constraint_coeffs_list.append(constraint_coeffs)
        constraint_rhs_list.append(constraint_rhs)
        constraint_types.append(constraint_type)
  1. 员工和项目数量获取: 通过input函数,程序获取用户输入的员工数量num_employees和项目数量num_projects。这些信息确定了指派问题的规模。
  2. 成本系数获取: 使用两层嵌套循环,依次获取每个员工分配到每个项目的成本。对于每个员工 - 项目组合,用户输入成本值,这些值被存储在costs列表中。这个列表将用于构建目标函数,反映不同指派方案的成本情况。
  3. 额外约束条件获取: 用户可以输入额外的约束条件数量num_constraints。对于每个额外约束条件,程序通过循环获取其系数(存储在constraint_coeffs_list中)、右侧常数(存储在constraint_rhs_list中)和约束类型(存储在constraint_types中)。这些额外约束条件可以根据实际问题的特殊需求进行设定,例如某些员工不能参与特定项目,或者某些项目必须在特定条件下完成等。

(三)模型构建部分

python

    # 创建问题
    problem = LpProblem("Assignment_Problem", LpMinimize)

    # 定义决策变量(0 - 1规划)
    assignments = [LpVariable(f"x{i}", lowBound=0, cat='Binary') for i in range(num_employees * num_projects)]

    # 定义目标函数
    problem += lpSum([costs[i] * assignments[i] for i in range(num_employees * num_projects)])

    # 添加默认约束条件
    # 每一个人只能分配一个项目
    for i in range(num_employees):
        problem += lpSum([assignments[i * num_projects + j] for j in range(num_projects)]) <= 1

    # 每一个项目只能被一个员工分配
    for j in range(num_projects):
        problem += lpSum([assignments[i * num_projects + j] for i in range(num_employees)]) == 1

    # 添加额外约束条件
    for constraint_coeffs, constraint_rhs, constraint_type in zip(constraint_coeffs_list, constraint_rhs_list,
                                                                  constraint_types):
        if constraint_type == '<=':
            problem += lpSum([coeff * var for coeff, var in zip(constraint_coeffs, assignments)]) <= constraint_rhs
        elif constraint_type == '=':
            problem += lpSum([coeff * var for coeff, var in zip(constraint_coeffs, assignments)]) == constraint_rhs
        else:
            print(f"不支持的约束类型 '{constraint_type}',忽略该约束。")
  1. 问题创建: 使用LpProblem类创建一个名为"Assignment_Problem"的最小化问题实例problem。这表明我们的目标是最小化总成本。
  2. 决策变量定义: 通过列表推导式创建了\(num_employees * num_projects\)个决策变量assignments。每个变量命名为x{i},下限为 0,类型为Binary,即只能取 0 或 1。这些变量对应于每个员工 - 项目指派组合,x{i}为 1 表示员工被指派到相应项目,为 0 则表示未被指派。
  3. 目标函数构建: 利用lpSum函数,将costs列表中的成本值与对应的决策变量相乘并求和,构建出目标函数。这确保了在求解过程中,程序会尝试找到一组决策变量值,使得总成本最小。
  4. 默认约束条件添加
    • 员工分配约束:通过循环,为每个员工添加约束条件,确保每个员工最多只能被分配到一个项目(这里使用<=是因为在pulp库中,对于这种整数规划问题,即使写成==,在求解时也会按照整数解的要求处理,但<=更具一般性,防止可能的求解异常)。
    • 项目分配约束:同样通过循环,为每个项目添加约束条件,保证每个项目只能被一个员工分配。
  5. 额外约束条件添加: 遍历用户输入的额外约束条件信息,根据约束类型(<=''='),使用lpSum函数构建相应的约束表达式,并添加到问题problem中。如果输入的约束类型不被支持,则输出提示信息并忽略该约束。

(四)求解与结果输出部分

python

    # 求解问题
    problem.solve()

    # 输出结果
    print(f"The status of the problem is {LpStatus[problem.status]}")
    print(f"The optimal value of the problem is {value(problem.objective)}")
    for i in range(num_employees * num_projects):
        print(value(assignments[i]), end=' ')
        if (i + 1) % num_projects == 0:
            print()

    return value(problem.objective), [value(var) for var in assignments]
  1. 问题求解: 调用problem.solve()方法,利用pulp库的求解器对构建好的 0-1 规划模型进行求解。
  2. 结果输出
    • 求解状态输出:通过LpStatus[problem.status]获取问题的求解状态,如是否成功找到最优解、问题是否可行等,并输出。
    • 目标函数最优值输出:使用value(problem.objective)获取目标函数的最优值,即最小化的总成本,并输出。
    • 决策变量值输出:通过循环遍历决策变量列表,输出每个决策变量的值。每输出num_projects个值后换行,这样可以清晰地展示每个员工对各个项目的指派情况。0 表示未指派,1 表示指派。最后,返回目标函数最优值和决策变量的值,以便在需要时进行进一步处理。

四、实例演示

假设我们有 5 名员工和 4 个项目,给定的成本数据如下:

python

# 定义目标函数的系数
costs = [66.8, 75.6, 87, 58.6,
         57.2, 66, 66.4, 53,
         78, 67.8, 84.6, 59.4,
         70, 74.2, 69.6, 57.2,
         67.4, 71, 83.8, 62.4]

        运行代码后,在输入环节,依次输入员工数量为 5,项目数量为 4。接着,按照顺序逐行输入上述costs列表中的 20 个成本值。由于这里假设没有额外约束条件,输入额外约束条件数量时可输入 0。

        求解完成后,程序输出结果如下:

The status of the problem is Optimal
The optimal value of the problem is 236.8
0.0 0.0 0.0 1.0 
1.0 0.0 0.0 0.0 
0.0 1.0 0.0 0.0 
0.0 0.0 1.0 0.0 
0.0 0.0 0.0 0.0 

        上述结果表明,该指派问题成功求得最优解,此时最小成本为 253.2。具体的指派方案为:

  • 员工 1 被指派到项目 4(因为0.0 0.0 0.0 1.0表示员工 1 对应项目 1、2、3 的决策变量为 0,对应项目 4 的决策变量为 1)。
  • 员工 2 被指派到项目 1(1.0 0.0 0.0 0.0)。
  • 员工 3 被指派到项目 2(0.0 1.0 0.0 0.0)。
  • 员工 4 被指派到项目 3(0.0 0.0 1.0 0.0)。
  • 员工 5 未被指派到任何项目(0.0 0.0 0.0 0.0)。

        从结果可以看出,由于员工数量(5 名)多于项目数量(4 个),通过 0-1 规划结合pulp库能够在满足每个员工最多承担一个项目、每个项目至少有一个员工负责的条件下,高效地找到最优成本方案。在这种情况下,为了达到最小成本253.2,员工 5 未被安排项目,这是符合模型设定的,因为模型的目标是在给定的成本矩阵和约束条件下,实现总成本的最小化,而不一定要求所有员工都参与项目。

五、总结

        通过本文的介绍,我们深入了解了 0-1 规划在数学建模指派问题中的应用原理以及如何使用 Python 的pulp库实现求解过程。从问题的数学模型构建,到 Python 代码中输入获取、模型构建、求解及结果输出的每一个环节,都展示了 0-1 规划解决指派问题的高效性和准确性。在实际应用中,读者可以根据具体问题的需求,灵活调整输入参数,运用 0-1 规划方法找到最优的指派方案,为企业决策、资源分配等实际场景提供有力支持。希望本文能够帮助读者掌握这一重要的优化技术,在相关领域取得更好的实践成果。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2334288.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

React 学习 JSX

APP根组件被index.js渲染到public下的index.html下 JS中写 HTML 代码 渲染列表 条件渲染 复杂条件渲染 事件绑定 传递自定义参数 button标签中写箭头函数引用的格式 自定义参数和事件本身对象都想要的情况

大模型论文:Language Models are Few-Shot Learners(GPT3)

大模型论文&#xff1a;Language Models are Few-Shot Learners(GPT3) 文章地址&#xff1a;https://proceedings.neurips.cc/paper_files/paper/2020/file/1457c0d6bfcb4967418bfb8ac142f64a-Paper.pdf 一、摘要 我们证明了&#xff0c;扩大语言模型的规模在任务无关的 few…

一周学会Pandas2 Python数据处理与分析-Pandas2数据导出

锋哥原创的Pandas2 Python数据处理与分析 视频教程&#xff1a; 2025版 Pandas2 Python数据处理与分析 视频教程(无废话版) 玩命更新中~_哔哩哔哩_bilibili 任何原始格式的数据载入DataFrame后&#xff0c;都可以使用类似 DataFrame.to_csv()的方法输出到相应格式的文件或者…

深入解析栈式虚拟机与反向波兰表示法

1.1 什么是虚拟机&#xff1f; 虚拟机&#xff08;Virtual Machine, VM&#xff09;是一种软件实现的计算机系统&#xff0c;提供与物理计算机相类似的环境&#xff0c;但在软件层面运行。虚拟机的存在简化了跨平台兼容性、资源管理以及安全隔离等问题。 1.2 栈式虚拟机的架构…

学习MySQL的第八天

海到无边天作岸 山登绝顶我为峰 一、数据库的创建、修改与删除 1.1 引言 在经过前面七天对于MySQL基本知识的学习之后&#xff0c;现在我们从基本的语句命令开始进入综合性的语句的编写来实现特定的需求&#xff0c;从这里开始需要我们有一个宏观的思想&…

AI识别与雾炮联动:工地尘雾治理新途径

利用视觉分析的AI识别用于设备联动雾炮方案 背景 在建筑工地场景中&#xff0c;人工操作、机械作业以及环境因素常常导致局部出现大量尘雾。传统监管方式存在诸多弊端&#xff0c;如效率低、资源分散、监控功能单一、人力效率低等&#xff0c;难以完美适配现代工程需求。例如…

GD32F303-IAP的过程和实验

使用的芯片为GD32F303VC 什么是IAP呢&#xff1f;有个博主写的很清楚&#xff1b;就是远程升级&#xff1b; 【单片机开发】单片机的烧录方式详解&#xff08;ICP、IAP、ISP&#xff09;_isp烧录-CSDN博客 我们需要写一个boot 和APP 通过 boot对APP的程序进行更新&#xf…

众趣科技助力商家“以真示人”,让消费场景更真实透明

在当今的消费环境中&#xff0c;消费者权益保护问题日益凸显。无论是网购商品与实物不符、预定酒店民宿与图文描述差异大&#xff0c;还是游览景区遭遇“照骗”&#xff0c;这些问题不仅让消费者在消费和决策过程中倍感困扰&#xff0c;也让商家面临信任危机。 消费者在享受便…

spark core编程之行动算子、累加器、广播变量

一、RDD 行动算子 reduce&#xff1a;聚集 RDD 所有元素&#xff0c;先聚合分区内数据&#xff0c;再聚合分区间数据。 collect&#xff1a;在驱动程序中以数组形式返回数据集所有元素。 foreach&#xff1a;分布式遍历 RDD 元素并调用指定函数。 count&#xff1a;返回 RDD…

提高课:数据结构之树状数组

1&#xff0c;楼兰图腾 #include<iostream> #include<cstring> #include<cstdio> #include<algorithm>using namespace std;typedef long long LL;const int N 200010;int n; int a[N]; int tr[N]; int Greater[N], lower[N];int lowbit(int x) {ret…

基于javaweb的SpringBoot新闻视频发布推荐评论系统(源码+部署文档)

技术范围&#xff1a;SpringBoot、Vue、SSM、HLMT、Jsp、PHP、Nodejs、Python、爬虫、数据可视化、小程序、安卓app、大数据、物联网、机器学习等设计与开发。 主要内容&#xff1a;免费功能设计、开题报告、任务书、中期检查PPT、系统功能实现、代码编写、论文编写和辅导、论文…

机器学习之PCA主成分分析详解

文章目录 引言一、PCA的概念二、PCA的基本数学原理2.1 内积与投影2.2 基2.3 基变换2.4 关键问题及优化目标2.5 方差2.6 协方差2.7 协方差矩阵2.8 协方差矩阵对角化 三、PCA执行步骤总结四、PCA计算实例五、PCA参数解释六、代码实现七、PCA的优缺点八、总结 引言 在机器学习领域…

回溯——固定套路 | 面试算法12道

目录 输出二叉树所有路径 路径总和问题 组合总和问题 分割回文串 子集问题 排列问题 字母大小写全排列 单词搜索 复原IP地址 电话号码问题 括号生成问题 给我一种感觉是回溯需要画图思考是否需要剪枝。 元素个数n相当于树的宽度&#xff08;横向&#xff09;&#x…

Maven和MyBatis学习总结

目录 Maven 1.Maven的概念&#xff1a; 2.在具体的使用中意义&#xff1a; 3.与传统项目引入jar包做对比&#xff1a; 传统方式&#xff1a; 在maven项目当中&#xff1a; 4.在创建maven项目后&#xff0c;想要自定义一些maven配置 5.maven项目的结构 6.maven指令的生…

AndroidTV 当贝播放器-v1.5.2-官方简洁无广告版

AndroidTV 当贝播放器 链接&#xff1a;https://pan.xunlei.com/s/VONXRf0g3cT0ECVt6GEsoODFA1?pwds4qv# AndroidTV 当贝播放器-v1.5.2-官方简洁无广告版

Python生成exe

其中的 -w 参数是 PyInstaller 用于窗口模式&#xff08;Windowed mode&#xff09;&#xff0c;它会关闭命令行窗口的输出&#xff0c;这通常用于 图形界面程序&#xff08;GUI&#xff09;&#xff0c;比如使用 PyQt6, Tkinter, PySide6 等。 所以&#xff1a; 如果你在没有…

MySql 自我总结

目录 1. 数据库约束 1.1约束类型 2. 表的设计 2.1 一对一 2.2 一对多 2.3 多对多 3. 新增 4. 查询 4.1 聚合查询 4.2 GROUP BY 4.3 HAVING 4.4 联合查询 4.5 内连接 4.5.1 内连接的核心概念 4.5.2 内连接的语法 4.5.3 ON 与 WHERE 的区别 4.6 自连接 4.6.1 定…

uni-app app 安卓和ios防截屏

首先可参考文档 uni.setUserCaptureScreen 这里需要在项目中引入这个插件 uni-usercapturescreen - DCloud 插件市场 否则会报错,在需要防止截屏录屏的页面中,加入 uni.setUserCaptureScreen({enable: false,success() {console.log(全局截屏录屏功能已禁用);},fail(err)…

【Go】windows下的Go安装与配置,并运行第一个Go程序

【Go】windows下的Go安装与配置&#xff0c;并运行第一个Go程序 安装环境&#xff1a;windows10 64位 安装版本&#xff1a;go1.16 windows/amd64 一、安装配置步骤 1.到官方网址下载安装包 https://golang.google.cn/dl/ 默认情况下 .msi 文件会安装在 c:\Go 目录下。可自行配…

vue3腾讯云直播 前端拉流(前端页面展示直播)

1、引入文件&#xff0c;在index.html <link href"https://tcsdk.com/player/tcplayer/release/v5.3.2/tcplayer.min.css" rel"stylesheet" /><!--播放器脚本文件--><script src"https://tcsdk.com/player/tcplayer/release/v5.3.2/t…