变邻域搜索(VNS)原理梳理和应用细节-附求解VRPTW问题C++代码

简介: 变邻域搜索(VNS)原理梳理和应用细节-附求解VRPTW问题C++代码

1、简介-introduction


变邻域搜索(Variable Neighborhood Search)是一种求解组合优化和全局优化问题的元启发式算法,它的基本思想是:在局部搜索(LS)算法的框架中系统地变化邻域结构的选择。


下面本文将介绍以下几部分内容,第二部分本文将介绍VNS的基本机制;第三部分将介绍VNS的相关拓展;第四部分将介绍VNS和其他启发式算法的混杂应用;第五部分将介绍VNS在求解VRPTW中的具体思路,最后将给出使用VNS求解VRPTW问题的代码。




2、VNS的基本机制


首先定义Nk,(k=1,...,kmax)表示事先定义好的邻域结构集合,定义 N k ( x )表示第  k个邻域结构中构造的解的集合。VNS基于三个简单的事实:


①某一个邻域结构的局部最优解不一定是另外一个邻域结构的局部最优解;


②对于所有邻域结构来说,全局最优解均是其局部最优解;


③对于许多问题情境,一个或者多个邻域的局部最优解应该彼此相近。


最后一个事实,虽然有点经验主义,但是暗示一个局部最优解通常会为全局最优解提供有效信息。



2.1 变邻域下降方法-Variable Neighborhood Descent


在VND中,事先选定的邻域结构按照一个确定的顺序进行执行,其流程如下图所示:

7c0c8d927c1f43e7b1566d4a4b6765ec.png


首先进行初始化,选择邻域结构Nk,(k=1,...,kmax),构建一个初始解,选定终止准则;之后循环执行VND,直到达到终止准则。其中VND分为两步,第一步为重置计数变量  k=1,第二步为顺序执行邻域结构,直到k=kmax。在每一次选定邻域结构进行执行的时候又可以分为两步(a)在第  k个邻域结构生成的解方案中随机选择一个解方案 x′-shaking;(b)若选择的解方案  x′优于当前解,则使用  x′替换当前解,同时重新从第一个邻域结构开始执行优化;否则选择下一个邻域结构进行解的优化。


除了上述顺序执行的方法,VND还可以进行一定的改进,使用嵌套策略来执行邻域结构。假设有3个邻域结构,  kmax=3,则嵌套结构执行方法为在外层只使用第3个邻域结构,而对于第3个邻域结构生成的每一个解方案 x′,顺序执行邻域结构1和2组成的VND。


缩减VNS(Reduced Variable Neighborhood Search)方法同上述VND方法类似,但在shaking之后,不是对 x′执行后续的邻域操作,而是对当前最优解进行后续的邻域操作。



2.2基本变邻域搜索方法-Basic Variable Neighborhood Search


VNS综合了邻域选择的确定性和随机性因素,其流程如下图所示:

c16cdc4e87604e319a1ee66df721f36d.png


基本VNS与VND的区别主要在于(2.b)中在执行完邻域操作随机选定一个解x′之后,VNS不是直接判断解 x′的优劣,而是对其进行进一步的局部搜索(LS)探索,将LS得到的解  x′′与当前解进行判断,保留较优的解。



2.3 终止准则


终止准则的选择可以有但不限于以下三种方式:


①规定一个最大的CPU运算时间;


②规定一个总的最大的迭代次数;


③规定一个最大迭代次数 m a x I t e maxIte maxIte,若在 m a x I t e maxIte maxIte之内解没有发生优化,则终止算法。




3、VNS的拓展和混杂算法



3.1 VNS的拓展


基本的VNS是一种基于descent,first improvement method和randomization的算法。在基本VNS流程的(2.c)中,可以加入一定的概率机制,当解 x ′ ′ x'' x′′次于当前解时,以一定的几率接受退化的解,来实现一种descent-ascent 机制;在选择邻域结构时,也可以加入轮盘赌的机制,优先选择优化效果较好的邻域结构进行执行;在(2.a)中可以选出最优的解 x ′ x' x′来进行后续的优化;还可以规定一个kmin和 k s t e p k_{step} kstep

k ← k + 1 k←k+1 k←k+1为 k ← k s t e p k←k_{step} k←kstep。



3.2 变邻域分解搜索-Variable Neighborhood Decomposition Search


VNDS方法将基本VNS拓展称为一种两级VNS,VNDS的步骤如下所示:


image.png


VNDS相对于基本VNS的主要区别在于(2.b)中,VNDS不是直接对于解 x′进行局部搜索LS操作,而是在(2.a)时选出一个解集合 y=x′/ x,在新的解空间  y(全部解空间的子集)中进行局部寻优,记寻找到的最优解为  y′,整个解空间中的最优解记为x′′,之后再执行(2.c)更新最优解或者执行下一次邻域结构优化。




3.3 偏变邻域搜索-Skewed VNS


SVNS用来探索距离当前解较远的另外的求解空间,以求得到不同的解结构,寻找到更优的解,SVNS的流程如下图所示:


image.png



从上述流程可以看出,SVNS相对于基本VNS,主要将(2.c)分解成为了两部分,新的(2.c)用来判断是否进行更新最优解,新的(2.d)用来判断是否进行重新开始邻域搜索还是继续进行邻域搜索,其中利用了函数  ρ(x,x′′)来判断当前解 x x x和局部最优解  x′′之间的距离,(本人理解:两个解之间的差异程度,如在VRPTW中,可以指两个路径方案用的车辆数,访问客户点的顺序差异程度等等),当) ρ(x,x′′)的值较小的时候,参数 α \alpha α取值应该较大,来使求解空间转移到差距更大的地方进行寻优。 αρ(x,x′′)的选取还可以采取学习机制或者自适应的机制进行更新。



3.4 并行变邻域搜索-Parallel VNS


在(2.b)时采用多线程的编程方式,对于某一个邻域产生的多个解方案来说,给每一个解方案开一个进程进行局部搜索LS,最终保留所有进程中求解结果最好的结果。




4、VNS和其他启发式算法的混合使用-hybrids


4.1 VNS 和 TS


有两种方式来进行VNS和TS的混合使用,第一种是在TS框架之中使用VNS,第二种是在VNS框架中使用TS。


在VNS框架中使用TS时,只需要将禁忌表机制加入到VNS搜索的过程即可,为所有的邻域搜索算子建立一个禁忌表,规定解禁步长和宽恕准则,可以有效避免backward搜索。


在TS框架中使用VNS时,只需要将TS中解的优化步骤中的LS优化变为VNS优化即可。




4.2 VNS 和 GRASP


GRASP是一种两阶段元启发式算法,在第一阶段使用贪婪随机的思想进行初始解的构建,在第二阶段使用LS或者枚举的思想进行解的优化。所以,一种顺其自然的方式来结合VNS和GRASP是在GRASP的第二阶段使用VNS来进行优化。




5、利用VNS框架求解VRPTW



首先选定邻域操作算子,本文在实现VNS时使用3种邻域操作算子,分别是2-opt inter-route算子,2-opt intra-route 算子和1-opt inter-route算子;在实现时使用基本VNS框架进行实现;终止准则本文使用了规定最大的迭代次数的终止准则。


在基本VNS框架上,本文进行了些许改进,在内循环依次执行邻域操作算子时,本文只使用2-opt inter-route算子和2-opt intra-route 算子两种算子进行当前最优解的优化,在内循环结束之后,使用1-opt inter-route算子进行LS局部搜索优化,寻找更优的解。为了避免陷入局部最优,本文使用了“整条路径”破坏的思想,来跳出局部最优解,以求找到全局最优解。





6、求解代码已经上传值GitHub



https://github.com/Dragon-wang-fei-long/Huristic-code/tree/Dragon-wang-fei-long-Heuristic-VNS




















相关文章
|
3月前
|
编译器 C++ 开发者
【C++】深入解析C/C++内存管理:new与delete的使用及原理(三)
【C++】深入解析C/C++内存管理:new与delete的使用及原理
|
2月前
|
算法 安全 C++
提高C/C++代码的可读性
提高C/C++代码的可读性
62 4
|
3月前
|
存储 并行计算 安全
C++多线程应用
【10月更文挑战第29天】C++ 中的多线程应用广泛,常见场景包括并行计算、网络编程中的并发服务器和图形用户界面(GUI)应用。通过多线程可以显著提升计算速度和响应能力。示例代码展示了如何使用 `pthread` 库创建和管理线程。注意事项包括数据同步与互斥、线程间通信和线程安全的类设计,以确保程序的正确性和稳定性。
|
3月前
|
C++
C++番外篇——虚拟继承解决数据冗余和二义性的原理
C++番外篇——虚拟继承解决数据冗余和二义性的原理
50 1
|
3月前
|
Linux C语言 C++
vsCode远程执行c和c++代码并操控linux服务器完整教程
这篇文章提供了一个完整的教程,介绍如何在Visual Studio Code中配置和使用插件来远程执行C和C++代码,并操控Linux服务器,包括安装VSCode、安装插件、配置插件、配置编译工具、升级glibc和编写代码进行调试的步骤。
425 0
vsCode远程执行c和c++代码并操控linux服务器完整教程
|
3月前
|
存储 编译器 C++
【C++篇】揭开 C++ STL list 容器的神秘面纱:从底层设计到高效应用的全景解析(附源码)
【C++篇】揭开 C++ STL list 容器的神秘面纱:从底层设计到高效应用的全景解析(附源码)
85 2
|
3月前
|
存储 算法 C++
【搜索算法】 跳马问题(C/C++)
【搜索算法】 跳马问题(C/C++)
|
3月前
|
人工智能 算法 Java
【搜索算法】数字游戏(C/C++)
【搜索算法】数字游戏(C/C++)
|
2月前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
60 2
|
2月前
|
存储 编译器 C++
【c++】类和对象(下)(取地址运算符重载、深究构造函数、类型转换、static修饰成员、友元、内部类、匿名对象)
本文介绍了C++中类和对象的高级特性,包括取地址运算符重载、构造函数的初始化列表、类型转换、static修饰成员、友元、内部类及匿名对象等内容。文章详细解释了每个概念的使用方法和注意事项,帮助读者深入了解C++面向对象编程的核心机制。
110 5