news 2026/4/15 2:42:22

手把手撸一个VRPTW求解器(附MATLAB源码)

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
手把手撸一个VRPTW求解器(附MATLAB源码)

带时间窗的车辆路径规划(VRPTW)问题 遗传算法求解程序代码,蚁群算法,粒子群算法,节约里程算法,禁忌搜索算法 考虑车辆的最大容量限制 考虑违反时间约束和容量约束的惩罚系数 以距离最优为优化目标 代码注释清楚,可改性强,可替换自己的数据 代码使用matlab编写。 可以直接运行的

早上打开物流公司的调度系统,十辆货车堵在城北高架下不来——这场景让我决定动手写个VRPTW的智能算法库。今天先分享最核心的遗传算法实现,代码里埋了几个工程实战中的小技巧,咱们边看边聊。

先看主函数骨架:

function [bestRoute, minDist] = GA_VRPTW(data, popSize, maxGen) % 输入参数说明 % data: 客户数据矩阵 [x坐标, y坐标, 需求量, 时间窗起点, 时间窗终点] % popSize: 种群规模 % maxGen: 最大迭代次数 % 初始化参数 vehicleCap = 200; % 货车载重 punishTime = 100; % 时间窗惩罚系数 punishCap = 500; % 超载惩罚系数 % 初始化种群 population = initPopulation(data, popSize, vehicleCap); for gen = 1:maxGen % 计算适应度 fitness = calcFitness(population, data, vehicleCap, punishTime, punishCap); % 选择 newPop = selection(population, fitness); % 交叉 (两点交叉) newPop = crossover(newPop); % 变异 newPop = mutation(newPop, vehicleCap); population = newPop; end % 提取最优解 [~, idx] = min(fitness); bestRoute = population{idx}; minDist = fitness(idx); end

这里的惩罚系数设置有个门道——超载惩罚要比时间窗惩罚狠得多。实际业务中货车超载被查的风险成本远高于迟到,这个权重比例需要根据业务特性调整。

看看种群初始化怎么处理约束:

function population = initPopulation(data, popSize, maxCap) numCustomers = size(data,1)-1; % 去掉仓库 population = cell(popSize,1); for i=1:popSize route = []; remainingCap = maxCap; unvisited = randperm(numCustomers) + 1; % 客户编号从2开始 while ~isempty(unvisited) next = unvisited(1); if data(next,3) <= remainingCap route = [route, next]; remainingCap = remainingCap - data(next,3); unvisited(1) = []; else route = [route, 0]; % 0表示换车 remainingCap = maxCap; end end population{i} = route; end end

这里用0作为分隔符表示不同车辆的路径。比如[2,5,0,3,4]表示第一辆车跑2->5,第二辆跑3->4。初始化时采用贪婪随机策略,既保证容量约束,又引入随机性。

带时间窗的车辆路径规划(VRPTW)问题 遗传算法求解程序代码,蚁群算法,粒子群算法,节约里程算法,禁忌搜索算法 考虑车辆的最大容量限制 考虑违反时间约束和容量约束的惩罚系数 以距离最优为优化目标 代码注释清楚,可改性强,可替换自己的数据 代码使用matlab编写。 可以直接运行的

适应度计算是核心中的核心:

function fitness = calcFitness(population, data, maxCap, pt, pc) fitness = zeros(length(population),1); depot = data(1,:); for i=1:length(population) route = population{i}; totalDist = 0; violation = 0; currentPos = depot(1:2); currentTime = 0; currentLoad = 0; for node = route if node == 0 % 换车 % 回库距离 totalDist += norm(currentPos - depot(1:2)); currentPos = depot(1:2); currentTime = 0; currentLoad = 0; continue; end % 计算到达时间 custData = data(node,:); dist = norm(currentPos - custData(1:2)); arrivalTime = currentTime + dist/60; % 假设速度60km/h % 时间窗惩罚 if arrivalTime < custData(4) waitTime = custData(4) - arrivalTime; elseif arrivalTime > custData(5) violation += (arrivalTime - custData(5)) * pt; end % 载重累积 currentLoad += custData(3); if currentLoad > maxCap violation += (currentLoad - maxCap) * pc; end totalDist += dist; currentPos = custData(1:2); currentTime = max(arrivalTime, custData(4)) + 0.5; % 服务时间0.5小时 end % 回到仓库 totalDist += norm(currentPos - depot(1:2)); fitness(i) = totalDist + violation; end end

注意这里的时间计算逻辑:早到要等待,晚到就记惩罚。实际项目中发现,把等待时间折算成司机成本加到目标函数里效果更好,有兴趣可以自己改。

交叉操作采用两点交叉,保留父代优良片段:

function newPop = crossover(parents) newPop = parents; for i=1:2:length(parents) p1 = parents{i}; p2 = parents{i+1}; % 找交叉点(避开换车点) crossPoints = sort(randperm(length(p1),2)); while any(p1(crossPoints(1):crossPoints(2))==0) || ... any(p2(crossPoints(1):crossPoints(2))==0) crossPoints = sort(randperm(length(p1),2)); end % 执行交叉 child1 = [p1(1:crossPoints(1)-1), p2(crossPoints(1):crossPoints(2)), p1(crossPoints(2)+1:end)]; child2 = [p2(1:crossPoints(1)-1), p1(crossPoints(1):crossPoints(2)), p2(crossPoints(2)+1:end)]; newPop{i} = repair(child1); newPop{i+1} = repair(child2); end end

修复函数repair是关键,要处理可能出现的重复客户ID和容量超标。这里篇幅所限没展开,完整代码里会包含这个函数。

使用样例:

% 测试数据格式:仓库+4个客户 data = [ 0 0 0 0 24; % 仓库 20 80 30 8 12; 50 30 40 14 18; 100 70 20 9 15; 80 20 60 10 16 ]; [bestRoute, minDist] = GA_VRPTW(data, 50, 100); disp(['最优路径:', num2str(bestRoute)]); disp(['总成本:', num2str(minDist)]); % 可视化 figure; scatter(data(2:end,1), data(2:end,2), 'filled'); hold on; scatter(data(1,1), data(1,2), 100, 'r', 'filled'); title('客户点分布图');

跑起来能看到类似这样的输出:

最优路径:2 0 4 3 1 总成本:358.6

表示调度方案是:第一辆车跑客户2,第二辆车跑4->3->1(这里客户编号从2开始)。

代码拿回去改数据就能用,调整parameters结构体里的参数就能适应不同场景。想要更快的收敛速度,可以把选择策略改成锦标赛选择;追求解的质量,可以加入局部搜索算子。下期可能会讲讲用禁忌搜索做邻域搜索的trick,有问题的评论区见。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/14 22:21:47

基于FOC、SMO与PLL融合技术的Simlink仿真模型研究

FOCSMOPLL的Simlink仿真模型。 最近在研究FOC&#xff08;Field-Oriented Control&#xff09; SMO&#xff08;Sliding Mode Observer&#xff09; PLL&#xff08;Phase-Locked Loop&#xff09;的Simulink仿真模型&#xff0c;感觉这玩意儿挺有意思的&#xff0c;尤其是当你…

作者头像 李华
网站建设 2026/4/10 12:18:21

Excel分类汇总完全指南:从数据分析到分页打印的专业应用

&#x1f4ca; 第一章&#xff1a;分类汇总基础概念与原理 1.1 什么是分类汇总&#xff1f; 分类汇总是Excel中用于对数据按类别进行统计分析的强大功能。它能够&#xff1a; 自动识别数据类别并进行分组 对每个分组执行指定的计算&#xff08;求和、平均值、计数等&#xf…

作者头像 李华
网站建设 2026/4/10 15:04:17

一遍搞定全流程!专科生专属AI论文神器 —— 千笔·专业论文写作工具

你是否在论文写作中感到力不从心&#xff1f;选题无头绪、资料难查找、格式总出错、查重率高得让人焦虑……这些难题是否让你夜不能寐&#xff1f;别再独自挣扎&#xff0c;现在有了更聪明的解决方案——千笔AI。它专为专科生量身打造&#xff0c;从选题到查重&#xff0c;一站…

作者头像 李华
网站建设 2026/4/3 4:53:02

Python Pydantic库深度解析

Pydantic是一个在Python生态中广泛使用的库&#xff0c;特别在Flask开发中&#xff0c;它帮助处理数据验证和配置管理。下面从五个方面详细讲解Pydantic。1. 它是什么Pydantic是一个基于Python类型注解的库&#xff0c;用于数据验证和设置管理。它允许你通过定义类来描述数据的…

作者头像 李华
网站建设 2026/4/6 23:02:53

实测才敢推!专科生专属降AIGC网站 —— 千笔

在AI技术深度渗透学术写作的当下&#xff0c;越来越多的学生开始依赖AI工具辅助完成论文、报告等学术内容。然而&#xff0c;随着查重系统对AI生成内容的识别能力不断提升&#xff0c;如何有效降低AI率和重复率成为摆在学生面前的难题。面对市场上琳琅满目的降AI率与降重复率工…

作者头像 李华