【matlab 路径规划】基于改进遗传粒子群算法的药店配送路径优化

一 背景介绍

本文分享的是一个基于订单合并的订单分配和路径规划联合优化,主要背景是骑手根据客户需求,从药店取药之后进行配送,配送的过程中考虑路径的长度、客户的服务时间窗、车辆的固定成本等要素,经过建模和优化得到最优的配送方案。

二 模型介绍

2.1基本假设

配送的具体流程和现实情况,建立的数学模型基于以下假设条件:
(1)O2O 药品零售平台旗下的各个门店能够满足已下单顾客的需求量,即不存在供不应
求的情况。
(2)已知消费者下单商品数量、地理位置及时间窗和每个消费者的需求量不会发生变化
(3)骑手在每个配送点服务时间恒定且相同,由于服务时间较短所以忽略不计。
(4)骑手从药店出发,中途不可返回药店取货,完成所有的配送任务后需要返回药店。
(5)在骑手对各配送点进行配送的过程中,不考虑交通堵塞、车辆故障、天气恶劣等突
发状况的影响。

2.2目标函数

在这里插入图片描述

2.3 约束条件

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

算法介绍

遗传算法是一种模拟自然进化过程的优化算法,用于解决优化问题。它模拟了生物进化的过程,通过对优良个体的选择、交叉和变异,逐步优化解的质量,最终找到最优解。

遗传算法的基本步骤包括:

  1. 初始化种群:随机生成一组初始解作为种群,通常采用随机数生成的方式。

  2. 适应度评价:根据问题的具体要求,采用适应度函数对每个个体进行评估,得到其适应度值。

  3. 选择操作:根据个体的适应度值,按照一定的选择概率选择优良个体作为父代,通常采用轮盘赌选择方法。

  4. 交叉操作:从选出的父代个体中选取一对个体,通过某种交叉方式生成新的个体。

  5. 变异操作:对新生成的个体进行一定的变异操作,改变其基因的值,增加种群的多样性。

  6. 更新种群:将新生成的个体加入到种群中,得到更新后的种群。

  7. 终止条件判断:判断是否满足终止条件,如达到最大迭代次数或找到满足要求的解。

  8. 返回最优解:返回种群中适应度最好的个体作为最优解。

遗传算法通过迭代优化的方式,不断改进解的质量,寻找到全局最优解或较好的局部最优解。它在解决复杂问题、搜索空间大的问题等方面具有很好的性能。

四 算例分析

算例1 本文使用30个节点的算例,1个配送节点 29个需求节点(分为三个优先级)
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

车辆编号.1: 0 -> 7 -> 1 -> 12 -> 15 -> 24 -> 22 -> 11 -> 27 -> 26 -> 29 ->
25 -> 0 到达时间节点: 0 - 4.7 - 9.9 - 11.4 - 12.4 - 14.9 - 15.6 - 17.4 -
19.6 - 24 - 26.7 - 28.4 - 33.7 min 行驶距离: 8413.36 m, 总时间: 33.7 min; 行驶成本 (C1): 21.03, 惩罚成本 (C2): 50.88
------------------------------------------------------------- 车辆编号.2: 0 -> 8 -> 19 -> 0 到达时间节点: 0 - 7.9 - 10.2 - 19.7 min 行驶距离: 4931.47
m, 总时间: 19.7 min; 行驶成本 (C1): 12.33, 惩罚成本 (C2): 29.59
------------------------------------------------------------- 车辆编号.3: 0 -> 18 -> 13 -> 4 -> 5 -> 16 -> 28 -> 0 到达时间节点: 0 - 2.5 - 6 - 7.5 -
10.1 - 13.6 - 15.1 - 16.6 min 行驶距离: 4138.40 m, 总时间: 16.6 min; 行驶成本 (C1): 10.35, 惩罚成本 (C2): 29.81
------------------------------------------------------------- 车辆编号.4: 0 -> 10 -> 6 -> 9 -> 20 -> 0 到达时间节点: 0 - 5.9 - 10 - 12.4 - 15.5 -
20.1 min 行驶距离: 5020.18 m, 总时间: 20.1 min; 行驶成本 (C1): 12.55, 惩罚成本 (C2): 33.81
------------------------------------------------------------- 车辆编号.5: 0 -> 23 -> 17 -> 14 -> 21 -> 30 -> 0 到达时间节点: 0 - 6.2 - 7.2 - 8.2 -
11.8 - 20.5 - 22.8 min 行驶距离: 5695.44 m, 总时间: 22.8 min; 行驶成本 (C1): 14.24, 惩罚成本 (C2): 37.93
------------------------------------------------------------- 车辆编号.6: 0 -> 2 -> 3 -> 0 到达时间节点: 0 - 1.5 - 5.7 - 9.5 min 行驶距离: 2363.65 m,
总时间: 9.5 min; 行驶成本 (C1): 5.91, 惩罚成本 (C2): 14.18

算例2 本文使用10个节点的算例,1个配送节点 9个需求节点(分为三个优先级)

在这里插入图片描述
**

车辆编号.1: 0 -> 2 -> 3 -> 1 -> 5 -> 4 -> 7 -> 8 -> 9 -> 6 -> 0 到达时间节点:
0 - 1.5 - 5.7 - 13.8 - 15.3 - 16.9 - 18.5 - 22.1 - 27.2 - 29.5 - 32.4
min 行驶距离: 8090.30 m, 总时间: 32.4 min; 行驶成本 (C1): 20.23, 惩罚成本 (C2):
54.26

**

六 项目分享

部分源码

clc
clear
close all
tic % 保存当前时间

dataloader
%% 初始化问题参数
CustomerNum = size(Position, 1) - 1; % 需求点个数

%% 需求点绘图
figure
hold on
xx = Position(:, 1);
yy = Position(:, 2);
idx1 = find(order_priority == 1);
idx2 = find(order_priority == 2);
idx3 = find(order_priority == 3);
scatter(xx(idx1), yy(idx1), 25, 'filled', 'go', 'DisplayName', '第一优先级')
scatter(xx(idx2), yy(idx2), 25, 'filled', 'bo', 'DisplayName', '第二优先级')
scatter(xx(idx3), yy(idx3), 25, 'filled', 'yo', 'DisplayName', '第三优先级')
scatter(xx(1), yy(1), 200, 'filled', 'rp', 'DisplayName', '药店')
legend
title('需求点散点图')

%% 初始化算法参数
NIND = 1000; % 粒子数量
MAXGEN = 100; % 最大迭代次数
mutation_prob = 0.05; % 变异概率
crossover_prob = 0.8; % 交叉概率
tournament_size = 5; % 锦标赛规模

%% 为预分配内存而初始化的0矩阵
Population = zeros(NIND, CustomerNum * 2 + 1); % 预分配内存,用于存储种群数据
PopDistance = zeros(NIND, 1); % 预分配矩阵内存
GbestDisByGen = zeros(MAXGEN, 1); % 预分配矩阵内存

penalty_costs = zeros(NIND, 1);
travel_costs = zeros(NIND, 1);
vehicle_costs = zeros(NIND, 1);
total_distances = zeros(NIND, 1);
penalty_orders = cell(NIND, 1);

for i = 1:NIND
    %% 初始化各粒子
    Population(i, :) = InitPop(CustomerNum, Distance, setting); % 使用GRASP算法生成TSP路径

    %% 计算路径长度
    PopDistance(i) = CalcDis(Population(i,:),Distance,TimeWindow,order_priority,setting); % 计算路径长度
end
%% 存储Pbest数据
Pbest = Population; % 初始化Pbest为当前粒子集合
PbestDistance = PopDistance; % 初始化Pbest的目标函数值为当前粒子集合的目标函数值

%% 存储Gbest数据
[mindis, index] = min(PbestDistance); % 获得Pbest中
Gbest = Pbest(index, :); % 初始Gbest粒子
GbestDistance = mindis; % 初始Gbest粒子的目标函数值

%% 开始迭代
gen = 1;

while gen <= MAXGEN
    %% 选择算子(锦标赛选择)
    new_population = zeros(size(Population));
    for i = 1:NIND
        new_population(i, :) = Selection(Population, PopDistance, tournament_size); % 锦标赛选择
    end
    Population = new_population;

    %% 每个粒子更新
    for i = 1:NIND
        %% 粒子与Pbest交叉
        if rand < crossover_prob
            Population(i, 2:end-1) = Crossover(Population(i, 2:end-1), Pbest(i, 2:end-1)); % 交叉
        end

        % 新路径长度变短则记录至Pbest
         PopDistance(i) = CalcDis(Population(i,:),Distance,TimeWindow,order_priority,setting); % 计算路径长度
        if PopDistance(i) < PbestDistance(i) % 若新路径长度变短
            Pbest(i, :) = Population(i, :); % 更新Pbest
            PbestDistance(i) = PopDistance(i); % 更新Pbest距离
        end

        %% 根据Pbest更新Gbest
        [mindis, index] = min(PbestDistance); % 找出Pbest中最短距离
        if mindis < GbestDistance % 若Pbest中最短距离小于Gbest距离
            Gbest = Pbest(index, :); % 更新Gbest
            GbestDistance = mindis; % 更新Gbest距离
        end

        %% 粒子与Gbest交叉
        if rand < crossover_prob
            Population(i, 2:end-1) = Crossover(Population(i, 2:end-1), Gbest(2:end-1));
        end

        % 新路径长度变短则记录至Pbest
        PopDistance(i) = CalcDis(Population(i,:),Distance,TimeWindow,order_priority,setting); % 计算路径长度
        if PopDistance(i) < PbestDistance(i) % 若新路径长度变短
            Pbest(i, :) = Population(i, :); % 更新Pbest
            PbestDistance(i) = PopDistance(i); % 更新Pbest距离
        end

        %% 粒子自身变异
        if rand < mutation_prob
            Population(i, :) = Mutate(Population(i, :), Distance); % 传递Distance矩阵
        end

        % 新路径长度变短则记录至Pbest
        PopDistance(i) = CalcDis(Population(i,:),Distance,TimeWindow,order_priority,setting); % 计算路径长度
        if PopDistance(i) < PbestDistance(i) % 若新路径长度变短
            Pbest(i, :) = Population(i, :); % 更新Pbest
            PbestDistance(i) = PopDistance(i); % 更新Pbest距离
        end

        %% 根据Pbest更新Gbest
        [mindis, index] = min(PbestDistance); % 找出Pbest中最短距离
        if mindis < GbestDistance % 若Pbest中最短距离小于Gbest距离
            Gbest = Pbest(index, :); % 更新Gbest
            GbestDistance = mindis; % 更新Gbest距离
        end
    end

    %% 显示此代信息
    fprintf('迭代次数 = %d, 最小成本 = %.2f   \n', gen, GbestDistance)

    %% 存储此代最短距离
    GbestDisByGen(gen) = GbestDistance;

    %% 更新迭代次数
    gen = gen + 1;
end

% 删去路径中多余1
for i = 1:length(Gbest) - 1
    if Gbest(i) == Gbest(i + 1)
        Gbest(i) = 0; % 相邻位都为1时前一个置零
    end
end
Gbest(Gbest == 0) = []; % 删去多余零元素

Gbest = Gbest - 1; % 编码各减1,与文中的编码一致

%% 计算结果数据输出到命令行
disp('-------------------------------------------------------------')
toc % 显示运行时间
TextOutput(Gbest, Distance, TimeWindow, setting); % 显示最优路径
disp('-------------------------------------------------------------')

%% 迭代图
figure
plot(GbestDisByGen, 'LineWidth', 2) % 展示目标函数值历史变化
xlim([1 gen - 1]) % 设置 x 坐标轴范围
set(gca, 'LineWidth', 1)
xlabel('迭代次数')
ylabel('最小成本')
title('遗传粒子群迭代曲线图')

%% 绘制实际路线
DrawPath(Gbest, Position, idx1, idx2, idx3)

本项目是典型的考虑车辆容量,车辆行驶距离,客户时间窗的车辆路径规划问题。使用了性能相对较好的遗传粒子群算法(GAPSO),代码使用模块化编程,主函数框架相对固定,能够兼容不同类型的优化模型。
需要完整项目源码或者需要定制项目的朋友欢迎咨询。


http://www.niftyadmin.cn/n/5536527.html

相关文章

LabVIEW自动探头外观检测

开发了一套基于LabVIEW的软件系统&#xff0c;结合视觉检测技术&#xff0c;实现探头及连接器外观的自动检测。通过使用高分辨率工业相机、光源和机械手臂&#xff0c;系统能够自动定位并检测探头表面的细微缺陷&#xff0c;如划痕、残胶、异色、杂物等。系统支持多种探头形态&…

在Ubuntu 16.04上安装和配置ownCloud的方法

前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到网站。 简介 ownCloud 是一个文件共享服务器&#xff0c;允许您将个人内容&#xff08;如文档和图片&#xff09;存储在一个类似 Dropbox 的集…

推动高效能:东芝TB67H301FTG全桥直流电机驱动IC

在如今高度自动化的时代&#xff0c;电子产品的性能和效率成为了工程师们关注的焦点。东芝的TB67H301FTG全桥直流电机驱动IC应运而生&#xff0c;以其卓越的技术和可靠性&#xff0c;成为众多应用的理想选择。无论是在机器人、家用电器、工业自动化&#xff0c;还是在其他需要精…

微信小程序的自定义组件

小程序目前已经支持组件化开发&#xff0c;可以将页面中的功能模块抽取成自定义组件&#xff0c;以便在不同的页面中重复使用&#xff1b; 也可以将复杂的页面拆分成多个低耦合的模块&#xff0c;有助于代码维护 常见组件 开发中常见的组件有两种&#xff1a; 1. 公共组件&am…

C# Winform PropertyGrid显示中文

主要原理是在枚举上添加DescriptionAttribute属性&#xff0c;然后通过反射将其显示出来 方法1&#xff1a;继承StringConverter类 public class EnumConvertor : StringConverter {public override bool GetStandardValuesSupported(ITypeDescriptorContext context){return…

基于STM32F103C8T6的同步电机驱动-CubeMX配置与IQmath调用

基于STM32F103C8T6的同步电机驱动-CubeMX配置与IQmath调用 一、功能描述: 上位机通过CAN总线实现对电机的运动控制,主要包含三种模式:位置模式、速度模式以及力矩模式。驱动器硬件核心为STM32F103C8T6,带相电压采集电路以及母线电压采集电路。其中供电电压12V。 PWM中心对…

步进电机(STM32+28BYJ-48)

一、简介 步进电动机&#xff08;stepping motor&#xff09;把电脉冲信号变换成角位移以控制转子转动的执行机构。在自动控制装置中作为执行器。每输入一个脉冲信号&#xff0c;步进电动机前进一步&#xff0c;故又称脉冲电动机。步进电动机多用于数字式计算机的外部设备&…

STM32 HAL库实现硬件IIC通信

文章目录 一. 前言二. 关于IIC通信三. IIC通信过程四. STM32实现硬件IIC通信五. 关于硬件IIC的Bug 一. 前言 最近正在DIY一款智能电池&#xff0c;需要使用STM32F030F4P6和TI的电池管理芯片BQ40Z50进行SMBUS通信。SMBUS本质上就是IIC通信&#xff0c;项目用到STM32CubeMXHAL库…