【运筹优化】(1) TSP 旅行商问题,Python + Gurobi 代码

简介: TSP(旅行商问题)涉及寻找有向完全图中起点到所有其他点的最短回路。目标是最小化路径权重总和,保证每个节点仅访问一次。模型通过0-1决策变量表示边的存在,约束确保每个节点恰好一次作为起点和终点。为消除子圈,引入MTZ方法,添加辅助变量破坏环路。实验中,随机生成30个点,计算距离并应用MTZ模型求解,通过Gurobi库实现并展示结果。

TSP

1. 介绍

问题描述:

给定一个有向完全图 G = (V, E)

有向完全图:一个节点可以直接到达其他节点

V:图中节点集合 |V| = N

E:图中边集合

当边 eij 与边 eji 对称时,称为对称 TSP 问题

目标:

找到一条从起点(任意点)出发依次不重复地经过所有其他节点,最终返回起点的最小权和(最短)的路径

输入: 一系列坐标点的集合

输出: 一个访问序列。每个节点被且只被访问一次;每个节点被且只被离开一次。

Pasted image 20240405203628.png


2. 建模

变量

符号 释义
$i, j$ 节点 i 和节点 j
$x_{ij}$ 0-1 决策变量,判断最优解中是否有边 (i, j)
$c_{ij}$ 边 $(i,j)$ 的权重 (距离、成本)
V 所有节点的集合

模型框架

公式 编号
$$min \sum_{i\in V} \sum_{j \in V} c_{ij}x_{ij}$$ (1)
$$\sum_{i \in V} x_{ij} =1 \quad\quad \forall j \in V, i\ne j$$ (2)
$$\sum_{j\in V}x_{ij} = 1 \quad\quad \forall i \in V, i\ne j$$ (3)
$$x_{ij} \in {0, 1} \quad\quad \forall i,j \in V, i\ne j$$ (4)

公式解释

  • (1) 最小化解的总权重,在最优解中所有边的权重相加
  • (2) 访问约束。被且只被访问一次。对于任意 j 节点,最优解中流入该节点的边总数等于 1
  • (3) 离开约束。被且只被离开一次。对于任意 i 节点,最优接种流出该节点的边的总数等于 1
  • (4) 决策变量的 0-1

不足

求解时不会出现一个环路,而是出现若干个子圈,子圈的权重之和与大圈相比更小

Pasted image 20240405210513.png

改进:破子圈约束

方法:MTZ 消除环路约束

加入辅助变量。破坏所有环路的存在。对每个节点引入一个决策变量 $\mu_i$ ,$\forall i \in V, \mu_i \ge 0$。
$$\mu_i - \mu_j + N\times x_{ij} \le N-1, \quad \forall i,j\in V, i,j\ne 0,i\ne j $$
其中,$\mu_i$ 是节点 i 被访问的次序,正常情况下,先访问 i 后访问 j,$\mu_i-\mu_j=-1$,若 j 被访问过了,j 的序号比 i 小,那么 $\mu_i-\mu_j>0$

最终模型

目标

  • $$min \sum_{i\in V} \sum_{j \in V} c_{ij}x_{ij}$$

约束

  • $$\sum_{i \in V} x_{ij} =1, \quad\quad \forall j \in V, i\ne j$$
  • $$\sum_{j\in V}x_{ij} = 1, \quad\quad \forall i \in V, i\ne j$$
  • $$\mu_i - \mu_j + N\times x_{ij} \le N-1, \quad\quad \forall i,j\in V, i,j\ne 0,i\ne j$$
  • $$x_{ij} \in \{0, 1\}, \quad\quad \forall i,j \in V, i\ne j$$
  • $$\mu_i \ge 0, \quad \mu_i \in R, \quad \forall i\in V$$

实验

随机生成30个点,计算两两节点 i 和 j 之间的距离,若 i=j 则将距离设为无限大。使用MTZ方法破子圈。

Pasted image 20240406212305.png

代码展示


import gurobipy as gp
import numpy as np
import matplotlib.pyplot as plt

# ------------------------------------------ #
# 数据集
# ------------------------------------------ #

coords = []  # 保存所有的节点坐标
n = 0  # 记录产生节点的个数
while n<30:
    coord = [np.random.randint(1,10), np.random.randint(1,10)]
    if coord not in coords:
        coords.append(coord)
        n += 1
grids = gp.tuplelist(coords)  # 转换为gurobi列表类型
N = len(grids)  # 节点个数
# 创建距离矩阵
dis_matrix = np.zeros(shape=[N,N])
for i in range(N):
    for j in range(N):
        if i==j:  # 如果节点自身到自身
            dis_matrix[i][j] = 1000000  # 无限远
        else:
            dis_matrix[i][j] = np.sqrt((grids[i][0]-grids[j][0])**2 \
                                + (grids[i][1]-grids[j][1])**2)

# ------------------------------------------ #
# 模型构建
# ------------------------------------------ #
m = gp.Model()

# 变量
x = m.addMVar((N,N), lb=0, ub=1, vtype=gp.GRB.BINARY)  # 代表i和j节点之间是否可达
u = m.addVars((N), vtype=gp.GRB.CONTINUOUS)  # 给每个节点一个编号 len=N
m.update()

# 目标
# m.setObjective(gp.quicksum(dis_matrix[i][j]*x[i][j] for j in range(N) for i in range(N)), gp.GRB.MINIMIZE)
obj = 0
for i in range(N):
    for j in range(N):
        obj += x[i][j] * dis_matrix[i][j]
m.setObjective(obj, gp.GRB.MINIMIZE)

# 约束
# m.addConstr(gp.quicksum(x[:,j] for j in range(N))==1, name='con_in')  # 流入约束
# m.addConstr(gp.quicksum(x[i,:] for i in range(N))==1, name='con_out')  # 流出约束
# 流入约束
for j in range(N):
    m.addConstr(gp.quicksum(x[:,j])==1)
# 流出约束
for i in range(N):
    m.addConstr(gp.quicksum(x[i,:])==1)


# 破子圈约束
for i in range(1,N):
    for j in range(1,N):
        if i != j:
            m.addConstr(u[i]-u[j]+N*x[i][j]<=N-1)

# 求解
m.optimize()

# 结果展示
var_lst = m.getVars()  # 变量值
for i in range(0, len(var_lst)):
    if var_lst[i].X != 0:
        print(var_lst[i].VarName, var_lst[i].X)
print("目标函数值:", m.ObjVal)
best=(x.X).astype(int)
print(f"最优分配为:\n {best}")

# 记录按顺序遍历的节点编号
ans = np.where(best==1)  # 获取被选节点的索引
ans_i = ans[0]
ans_j = ans[1]
n0 = ans_i[0]  # 起始节点
anslist = [n0]  # 记录编号
coordlist = [coords[n0]]  # 记录坐标
for _ in range(N):
    n1 = ans_j[n0]  # 下一个节点
    anslist.append(n1) 
    coordlist.append(coords[n1])
    n0 = n1  # 更新当前节点,当前走到n1

# 绘图
plt.plot(np.array(coords)[:,0], np.array(coords)[:,1], 'o')  # 散点[x,y]
plt.plot(np.array(coordlist)[:,0], np.array(coordlist)[:,1], '--')  # 折线
plt.show()
目录
相关文章
|
10天前
|
测试技术 Python
探索Python中的装饰器:简化代码,增强功能
在Python的世界中,装饰器是那些能够为我们的代码增添魔力的小精灵。它们不仅让代码看起来更加优雅,还能在不改变原有函数定义的情况下,增加额外的功能。本文将通过生动的例子和易于理解的语言,带你领略装饰器的奥秘,从基础概念到实际应用,一起开启Python装饰器的奇妙旅程。
28 11
|
25天前
|
缓存 监控 测试技术
Python中的装饰器:功能扩展与代码复用的利器###
本文深入探讨了Python中装饰器的概念、实现机制及其在实际开发中的应用价值。通过生动的实例和详尽的解释,文章展示了装饰器如何增强函数功能、提升代码可读性和维护性,并鼓励读者在项目中灵活运用这一强大的语言特性。 ###
|
24天前
|
Python
探索Python中的装饰器:简化代码,提升效率
【10月更文挑战第39天】在编程的世界中,我们总是在寻找使代码更简洁、更高效的方法。Python的装饰器提供了一种强大的工具,能够让我们做到这一点。本文将深入探讨装饰器的基本概念,展示如何通过它们来增强函数的功能,同时保持代码的整洁性。我们将从基础开始,逐步深入到装饰器的高级用法,让你了解如何利用这一特性来优化你的Python代码。准备好让你的代码变得更加优雅和强大了吗?让我们开始吧!
24 1
|
25天前
|
存储 缓存 监控
掌握Python装饰器:提升代码复用性与可读性的利器
在本文中,我们将深入探讨Python装饰器的概念、工作原理以及如何有效地应用它们来增强代码的可读性和复用性。不同于传统的函数调用,装饰器提供了一种优雅的方式来修改或扩展函数的行为,而无需直接修改原始函数代码。通过实际示例和应用场景分析,本文旨在帮助读者理解装饰器的实用性,并鼓励在日常编程实践中灵活运用这一强大特性。
|
25天前
|
机器学习/深度学习 算法 数据可视化
使用Python实现深度学习模型:智能食品配送优化
使用Python实现深度学习模型:智能食品配送优化
41 2
|
27天前
|
机器学习/深度学习 数据采集 人工智能
探索机器学习:从理论到Python代码实践
【10月更文挑战第36天】本文将深入浅出地介绍机器学习的基本概念、主要算法及其在Python中的实现。我们将通过实际案例,展示如何使用scikit-learn库进行数据预处理、模型选择和参数调优。无论你是初学者还是有一定基础的开发者,都能从中获得启发和实践指导。
42 2
|
18天前
|
存储 数据挖掘 开发者
Python编程入门:从零到英雄
在这篇文章中,我们将一起踏上Python编程的奇幻之旅。无论你是编程新手,还是希望拓展技能的开发者,本教程都将为你提供一条清晰的道路,引导你从基础语法走向实际应用。通过精心设计的代码示例和练习,你将学会如何用Python解决实际问题,并准备好迎接更复杂的编程挑战。让我们一起探索这个强大的语言,开启你的编程生涯吧!
|
24天前
|
机器学习/深度学习 人工智能 TensorFlow
人工智能浪潮下的自我修养:从Python编程入门到深度学习实践
【10月更文挑战第39天】本文旨在为初学者提供一条清晰的道路,从Python基础语法的掌握到深度学习领域的探索。我们将通过简明扼要的语言和实际代码示例,引导读者逐步构建起对人工智能技术的理解和应用能力。文章不仅涵盖Python编程的基础,还将深入探讨深度学习的核心概念、工具和实战技巧,帮助读者在AI的浪潮中找到自己的位置。
|
24天前
|
机器学习/深度学习 数据挖掘 Python
Python编程入门——从零开始构建你的第一个程序
【10月更文挑战第39天】本文将带你走进Python的世界,通过简单易懂的语言和实际的代码示例,让你快速掌握Python的基础语法。无论你是编程新手还是想学习新语言的老手,这篇文章都能为你提供有价值的信息。我们将从变量、数据类型、控制结构等基本概念入手,逐步过渡到函数、模块等高级特性,最后通过一个综合示例来巩固所学知识。让我们一起开启Python编程之旅吧!
|
24天前
|
存储 Python
Python编程入门:打造你的第一个程序
【10月更文挑战第39天】在数字时代的浪潮中,掌握编程技能如同掌握了一门新时代的语言。本文将引导你步入Python编程的奇妙世界,从零基础出发,一步步构建你的第一个程序。我们将探索编程的基本概念,通过简单示例理解变量、数据类型和控制结构,最终实现一个简单的猜数字游戏。这不仅是一段代码的旅程,更是逻辑思维和问题解决能力的锻炼之旅。准备好了吗?让我们开始吧!