线性规划库PuLP使用教程

Python求解线性规划——PuLP使用教程
简洁是智慧的灵魂,冗长是肤浅的藻饰。——莎士比亚《哈姆雷特》

文章目录

  • 一、说明
  • 二、安装 PuLP 库
  • 三、线性规划简介
    • 3.1 线性规划
      • 3.1.1 高考题目描述
      • 3.1.2 基本概念
    • 3.2 整数规划
      • 3.2.1 题目描述[3]
      • 3.2.2 解题思路
  • 四、求解过程
    • 4.1 定义模型
    • 4.2 定义决策变量
    • 4.3 添加约束条件
    • 4.4 添加目标函数
    • 4.5 模型求解
  • 五、 示例代码
    • 5.1 高考题代码
    • 5.2 汽车厂代码
  • 六、结论

一、说明

PulP库是求解线性规划问题的python库,本篇告诉大家,如何使用这种库,用案例给出说明。

二、安装 PuLP 库

如果您使用的是 Anaconda[1] 的话(事实上我也更推荐这样做),需要先激活你想要安装的虚拟环境,之后在 Prompt 输入

pip install pulp

不出意外的话等一会就安装完毕。

三、线性规划简介

想必大家能点开这篇文章一定都知道线性规划是什么意思吧……那么我用两个例子再简单说一下。

3.1 线性规划

3.1.1 高考题目描述

若变量 x,y, 满足约束条件:
{ 2 x + 3 y − 6 ≥ 0 x + y − 3 ≤ 0 y − 2 ≤ 0 \left\{ \begin{aligned} & 2x + 3y - 6\geq 0\\ & x + y - 3 \leq 0\\ & y - 2 \leq 0 \end{aligned} \right. 2x+3y60x+y30y20

z = 3 x + y z=3x+y z=3x+y 的最大值。

3.1.2 基本概念

首先,我们要认清在这道题中,x 和y 是可以变的,所以把它们叫做决策变量。三个不等式叫做约束条件,即 x 和 y 必须同时满足这三个不等式。我们若画出图来:
在这里插入图片描述
其中不满足约束条件的区域被我标上了颜色,所以 x,y 可以取得值只能在纯白区域内,这一片区域称作可行域。

再看最后的我们的目标:求 z=x+3y 的最大值。
于是 z=x+3y 就被称作目标函数,我们的工作就是求这个目标函数的最大值。

整个问题描述为:
在这里插入图片描述

然后怎么算?别急我们再看一个例子。

3.2 整数规划

3.2.1 题目描述[3]

汽车厂生产小、中、大三种类型的汽车,已知各类型每辆车对钢材、劳动时间的需求以及利润如下表所示。要求每月的钢材消耗不超过 600 t,总劳动时间不超过 60 000 h。试指定生产计划使得工厂每月的利润最大。

-小型车中型车大型车
钢材 / t1.535
劳动时间 / h280250400
利润 / 万元234

3.2.2 解题思路

首先,设三个决策变量,用x1,x2,x3 分别表示生产小型车、中型车、大型车的数量,但是注意要满足:

  • 车的数量只能是整数;
  • 车的数量大于等于 0。
    其他约束条件看题直接列:
    在这里插入图片描述

最后写出目标函数:
z = 2 x 1 + 3 x 2 + 4 x 3 z = 2x_1 + 3x_2 + 4x_3 z=2x1+3x2+4x3
综合起来整个问题描述为:
在这里插入图片描述

另外可以看出这个题由于涉及到三个决策变量,可行域是相当抽象的,这里就不画了 hhh~

四、求解过程

首先在最前面引入所需的pulp工具库:

import pulp as pl

这句话是引入 pulp 库并简写为 pl,一个 python 库只有在开始 import 了之后才能在后面使用。这样后面凡是用到 pulp 的功能都要写成 pl.xxx。

接下来是以下几个步骤:

  • 定义模型
  • 定义决策变量
  • 添加约束条件
  • 添加目标函数
  • 模型求解
  • 打印结果

4.1 定义模型

# Define the model
model = pl.LpProblem(name="My-Model", sense=pl.LpMaximize)

这个操作是使用 pl.LpProblem 创建了一个模型并赋值给变量 model,接收两个参数:

  • name:模型的名字,随便起一个;
  • sense:模型的类型,pl.LpMinimize是求目标函数的最小值,pl.LpMaximize 是求最大值

4.2 定义决策变量

# Define the decision variables
x = pl.LpVariable(name='x')
y = pl.LpVariable(name='y')

如果你的变量比较少的话可以简单这么写。这个意思是定义了两个浮点数变量,取值范围是整个实数域。注意等号左边的变量才是你在之后的计算式中使用的符号,而参数 name 只有在最后打印结果的时候才会被打印出来。另外如果你对变量有其他要求的话可以添加以下参数:

lowBound:变量的最小取值(不写的话默认负无穷); upBound:变量的最大取值(默认正无穷); cat:变量的类型,有
pl.Binary 逻辑变量、pl.Integer 整数、pl.Continuous 实数(默认值);

如果你的变量比较多而不得不用 1, 2, 3…… 来编号,可以采用类似这样的写法:

# Define the decision variables
x = {i: pl.LpVariable(name=f"x{i}", lowBound=0, cat=pl.LpInteger) for i in range(1, 9)}

这是一次定义 8 个变量并保存在一个类似数组的结构中,变量都是正整数,分别用 x[1],x[2], …, x[8] 表示,依次命名为 x1, x2,…, x8。

注意 range(left, right) 表示的区间是左闭右开。

4.3 添加约束条件

# Add constraints
model += (2 * x + 3 * y - 6 >= 0, "constrain_1")
model += (x + 3 * y - 3 == 0, "constrain_2")

没错!如你所见就是这么简单,括号里第一个变量就是你的约束不等式或等式,第二个变量是你的自定义的约束名(可以起一个有意义的名字,当然也可以省略)。

由于一些比较数学的原因,约束条件里是不能使用大于号“>”或小于号“<”的。

如果你像前面一样把变量定义在了数组中,那么可以直接用方括号调用:

model += (2 * x[1] + 3 * x[2] - 6 >= 0)

4.4 添加目标函数

# Set the objective
model += x + 3 * y

与前面添加约束条件不同,添加目标函数这一步不用加最外层的括号。

4.5 模型求解

# Solve the optimization problem
status = model.solve()

就写这一句话,调用 model 的 solve() 方法,并把结果保存在 status 中。

 ## 4.4 打印结果# Get the results
print(f"status: {model.status}, {pl.LpStatus[model.status]}")
print(f"objective: {model.objective.value()}")for var in model.variables():print(f"{var.name}: {var.value()}")for name, constraint in model.constraints.items():print(f"{name}: {constraint.value()}")

然后你就能看到模型求解的结果了。

五、 示例代码

5.1 高考题代码

首先解决一下 3.1 的高考题:

import pulp as pl# 定义一个模型,命名为 "Model_3.1",求最大值
model = pl.LpProblem(name="Model_3.1", sense=pl.LpMaximize)# 定义两个决策变量,取值为整个实数域
x = pl.LpVariable(name='x')
y = pl.LpVariable(name='y')# 添加三个约束条件
model += (2 * x + 3 * y - 6 >= 0)
model += (x + y - 3 <= 0)
model += (y - 2 <= 0)# 目标函数
model += x + 3 * y# 求解
status = model.solve()# 打印结果
print(f"status: {model.status}, {pl.LpStatus[model.status]}")
print(f"objective: {model.objective.value()}")for var in model.variables():print(f"{var.name}: {var.value()}")for name, constraint in model.constraints.items():print(f"{name}: {constraint.value()}")

查看结果的最后几行:

status: 1, Optimal objective: 7.0 x: 1.0 y: 2.0
_C1: 2.0
_C2: 0.0
_C3: 0.0

最大值是 7.0,在 x=1.0,y=2.0 时取到。

5.2 汽车厂代码

import pulp as pl# 定义一个模型,命名为 "Model_3.2",求最大值
model = pl.LpProblem(name="Model_3.2", sense=pl.LpMaximize)# 定义三个决策变量,取值正整数
x = {i: pl.LpVariable(name=f"x{i}", lowBound=0, cat=pl.LpInteger) for i in range(1, 4)}# 添加约束条件
model += (1.5 * x[1] + 3 * x[2] + 5 * x[3] <= 600)
model += (280 * x[1] + 250 * x[2] + 400 * x[3] <= 60000)# 目标函数
model += 2 * x[1] + 3 * x[2] + 4 * x[3]# 求解
status = model.solve()# 打印结果
print(f"status: {model.status}, {pl.LpStatus[model.status]}")
print(f"objective: {model.objective.value()}")for var in model.variables():print(f"{var.name}: {var.value()}")for name, constraint in model.constraints.items():print(f"{name}: {constraint.value()}")

查看结果的最后几行:

status: 1, Optimal objective: 632.0 x1: 64.0 x2: 168.0 x3: 0.0
_C1: 0.0
_C2: -80.0

三种车的产量分别取 64、168、0,最大收益 632 万元。

六、结论

众所周知 Python 在各个领域如此受欢迎很大程度上是因为其有众多强大的第三方库,但是用的多了就会发现如果安装太多库就有点乱。而 Anaconda 就是一种很方便的管理 Python 环境的工具,不仅可以将不同的库分门别类管理好,更有用的是可以在电脑上安装不同版本的 Python 而不用担心会互相冲突。 ↩︎

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

五分钟”手撕“图书管理系统

前言&#xff1a; 图书馆管理系统需要结合JavaSE的绝大部分知识&#xff0c;是一个很好的训练项目。 为了让大家更加方便的查阅与学习&#xff0c;我把代码放开头&#xff0c;供大家查询。 还有对代码的分析&#xff0c;我将以类为单位分开讲解。 目录 全部代码 Main类 Us…

智慧校园(安校易)管理系统 FileUpProductupdate.aspx 任意文件上传漏洞复现

0x01 产品简介 “安校易”是银达云创公司基于多年教育市场信息化建设经验沉淀,经过充分的客户需求调研,并依据国家“十三五”教育信息化建设规范而推出的综合互联网+教育信息化解决方案。“安校易”以物联网技术为基础,以学生在校“学食住行”管理为中心,将消费管理、门禁…

头歌01 -部分背包

蜜雪冰城店要举行一场调制饮品比赛。具体说明如下&#xff1a; 1&#xff09;参赛者拥有容量为800ml的杯子&#xff0c;可任选图2中的饮品进行混合&#xff1b;说明&#xff1a;图2中的饮品价格和体积已确定。 图2 2&#xff09;调制饮品价格为各所使用饮料的价格之和&#xff…

spring boot集成Knife4j

文章目录 一、Knife4j是什么&#xff1f;二、使用步骤1.引入依赖2.新增相关的配置类3.添加配置信息4.新建测试类5. 启动项目 三、其他版本集成时常见异常1. Failed to start bean ‘documentationPluginsBootstrapper2.访问地址后报404 一、Knife4j是什么&#xff1f; 前言&…

链游:区块链技术的游戏新纪元

随着区块链技术的快速发展&#xff0c;越来越多的行业开始探索与其结合的可能性&#xff0c;其中&#xff0c;游戏行业与区块链的结合尤为引人注目。链游&#xff0c;即基于区块链技术的游戏&#xff0c;正以其独特的优势&#xff0c;为玩家带来全新的游戏体验。本文将对链游进…

算法打卡 Day13(栈与队列)-滑动窗口最大值 + 前 K 个高频元素 + 总结

文章目录 Leetcode 239-滑动窗口最大值题目描述解题思路 Leetcode 347-前 K 个高频元素题目描述解题思路 栈与队列总结 Leetcode 239-滑动窗口最大值 题目描述 https://leetcode.cn/problems/sliding-window-maximum/description/ 解题思路 在本题中我们使用自定义的单调队列…

监控上网的软件有哪些?含泪推荐的电脑监控软件

监控上网的软件有很多&#xff0c;企业选择的时候应该遵循什么样的原则呢&#xff1f;鄙人愚见&#xff0c;认为以下四项原则是选择监控软件时首要考虑的。 1、功能需求&#xff1a; 监控软件不应该只是起到控制上网的作用&#xff0c;因为一些泄密行为可能是通过USB接口、打印…

如何在 Jupyter Notebook 中切换/使用 conda 虚拟环境?

参考文章&#xff1a; 【最全指南】如何在 Jupyter Notebook 中切换/使用 conda 虚拟环境&#xff1f;_多个conda环境 notebook用的哪个-CSDN博客 感谢这篇文章博主的解答&#xff0c;成功解决了我的难题。以下做一些具体的操作方法以及心得体会&#xff1a; 这里我使用的这篇…

MySQL主从复制(四):主备切换

一主多从结果&#xff1a; 图中&#xff0c; 虚线箭头表示的是主备关系&#xff0c; 也就是A和A’互为主备&#xff0c; 从库B、 C、 D指向的是主库A。 一主多从的设置&#xff0c; 一般用于读写分离&#xff0c; 主库负责所有的写入和一部分读&#xff0c; 其他的读请求则由从…

网络传输层

叠甲&#xff1a;以下文章主要是依靠我的实际编码学习中总结出来的经验之谈&#xff0c;求逻辑自洽&#xff0c;不能百分百保证正确&#xff0c;有错误、未定义、不合适的内容请尽情指出&#xff01; 文章目录 1.端口号的基础2.传输层两协议2.1.UDP 协议2.1.1.协议结构2.1.2.封…

【Vue2.x】props技术详解

1.什么是prop&#xff1f; 定义&#xff1a;组件标签上注册的一些自定义属性作用&#xff1a;向子组件传递数据特点 可以传递任意数量的prop可以传递任意类型的prop 2.prop校验 为了避免乱传数据&#xff0c;需要进行校验 完整写法 将之前props数组的写法&#xff0c;改为对象…

【软件测试】Selenium + Chrome UI自动化环境搭建

文章目录 自动化测试 Selenium Chrome 环境搭建1、下载Chrome 浏览器2、取消Chrome浏览器自动更新3、下载ChromeDriver4、测试环境是否搭建成功 自动化测试 Selenium Chrome 环境搭建 1、下载Chrome 浏览器 https://www.slimjet.com/chrome/google-chrome-old-version.php …

YOLOv8_seg的训练、验证、预测及导出[实例分割实践篇]

实例分割数据集链接,还是和目标检测篇一样,从coco2017val数据集中挑出来person和surfboard两类:链接:百度网盘 请输入提取码 提取码:3xmm 1.实例分割数据划分及配置 1.1实例分割数据划分 从上面得到的数据还不能够直接训练,需要按照一定的比例划分训练集和验证集,并按…

Linux信号:信号的保存

目录 一、信号在内核中的表示 二、sigset_t 2.1sigset_t的概念和意义 2.2信号集操作数 三、信号集操作数的使用 3.1sigprocmask 3.2sigpending 3.3sigemptyset 四、代码演示 一、信号在内核中的表示 实际执行信号的处理动作称为信号 递达(Delivery) 。 信号从产生到递达…

SpringCloud微服务03-微服务保护-分布式事务-MQ基础-MQ高级

一、微服务保护 1.雪崩问题 如何做好后备方案就是后续&#xff1a; 2.雪崩解决方案 某一个服务的线程是固定的&#xff0c;出现故障线程占满后&#xff0c;就不会让取调用这个服务&#xff0c;对其他服务就没有影响。 3.Sentinel ①初识Sentinel 配置过程&#xff1a;day05-服…

小短片创作-组装场景(一)

1、项目基础设置 通过第三人称模板&#xff0c;创建1个项目 1.自动曝光&#xff1a;关闭&#xff0c;因为要做专业的小短片&#xff0c;曝光需要手动控制。 2.扩展自动曝光中的默认亮度范围&#xff1a;启用 3.全局光照系统&#xff1a;选择屏幕空间光照&#xff08;SSGI&am…

P2P服务端模型配合 Tool.net P2pServerAsync 类使用

Tool.Net 支持的 P2P 服务器模型实例 说明服务器部分相关代码相关调用实例Tcp版本Udp版本 最后附一张思维图 说明 当前文章&#xff0c;仅是Tool.Net 开源库的一个缩影。本次更新V5.0版本以上提供支持。可以提供简单实现P2P功能用于业务开发。 服务器部分相关代码 完整代码&…

远程PLC、工控设备异地调试,贝锐蒲公英异地组网方案简单高效

北京宇东宁科技有限公司专门提供非标机电设备&#xff0c;能够用于金属制品的加工制造。设备主要采用西门子的PLC作为控制系统&#xff0c;同时能够连接上位机用于产量、温度、压力、电机运行数据的监控&#xff0c;以及工厂的大屏呈现需求。目前&#xff0c;客户主要是市场上的…

画图工具之PlantUML插件使用

文章目录 1 PlantUML插件1.1 引言1.2 什么是PlantUML1.3 PlantUML插件1.3.1 IntelliJ IDEA中插件1.3.2 VS Code中插件1.3.3 使用例子 1.4 PlantUML时序图语法1.4.1 声明参与者1.4.2 消息传递1.4.2.1 同步消息1.4.2.2 异步消息1.4.2.3 返回消息1.4.2.4 自调用 1.4.3 生命线&…

allegro 无法删除Xnet

allegro 无法删除Xnet Orcad中打开Constraint Manager之后&#xff0c;再生成网表&#xff0c;导入PCB后就会出现一堆Xnet网络。无法去除Xnet。 解决办法 在原理图ORCAD中&#xff0c; 1、打开Edit Object properties 2、选择Filter by:Capture 3、点击New Property 4、设置…