变邻域搜索(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




















相关文章
|
7月前
|
缓存 算法 程序员
C++STL底层原理:探秘标准模板库的内部机制
🌟蒋星熠Jaxonic带你深入STL底层:从容器内存管理到红黑树、哈希表,剖析迭代器、算法与分配器核心机制,揭秘C++标准库的高效设计哲学与性能优化实践。
C++STL底层原理:探秘标准模板库的内部机制
|
7月前
|
Ubuntu API C++
C++标准库、Windows API及Ubuntu API的综合应用
总之,C++标准库、Windows API和Ubuntu API的综合应用是一项挑战性较大的任务,需要开发者具备跨平台编程的深入知识和丰富经验。通过合理的架构设计和有效的工具选择,可以在不同的操作系统平台上高效地开发和部署应用程序。
297 11
|
存储 负载均衡 算法
基于 C++ 语言的迪杰斯特拉算法在局域网计算机管理中的应用剖析
在局域网计算机管理中,迪杰斯特拉算法用于优化网络路径、分配资源和定位故障节点,确保高效稳定的网络环境。该算法通过计算最短路径,提升数据传输速率与稳定性,实现负载均衡并快速排除故障。C++代码示例展示了其在网络模拟中的应用,为企业信息化建设提供有力支持。
376 15
|
8月前
|
C++ Windows
应用程序无法正常启动(0xc0000005)?C++报错0xC0000005如何解决?使命召唤17频频出现闪退,错误代码0xC0000005(0x0)
简介: 本文介绍了Windows应用程序出现错误代码0xc0000005的解决方法,该错误多由C++运行库配置不一致或内存访问越界引起。提供包括统一运行库配置、调试排查及安装Visual C++运行库等解决方案,并附有修复工具下载链接。
2230 1
|
10月前
|
API 数据安全/隐私保护 C++
永久修改机器码工具, exe一机一码破解工具,软件机器码一键修改工具【c++代码】
程序实现了完整的机器码修改功能,包含进程查找、内存扫描、模式匹配和修改操作。代码使用
|
11月前
|
C++
爱心代码 C++
这段C++代码使用EasyX图形库生成动态爱心图案。程序通过数学公式绘制爱心形状,并以帧动画形式呈现渐变效果。运行时需安装EasyX库,教程链接:http://【EasyX图形库的安装和使用】https://www.bilibili.com/video/BV1Xv4y1p7z1。代码中定义了屏幕尺寸、颜色数组等参数,利用随机数与数学函数生成动态点位,模拟爱心扩散与收缩动画,最终实现流畅的视觉效果。
1224 0
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
编译器 C++ 容器
【c++11】c++11新特性(上)(列表初始化、右值引用和移动语义、类的新默认成员函数、lambda表达式)
C++11为C++带来了革命性变化,引入了列表初始化、右值引用、移动语义、类的新默认成员函数和lambda表达式等特性。列表初始化统一了对象初始化方式,initializer_list简化了容器多元素初始化;右值引用和移动语义优化了资源管理,减少拷贝开销;类新增移动构造和移动赋值函数提升性能;lambda表达式提供匿名函数对象,增强代码简洁性和灵活性。这些特性共同推动了现代C++编程的发展,提升了开发效率与程序性能。
508 12
|
11月前
|
人工智能 机器人 编译器
c++模板初阶----函数模板与类模板
class 类模板名private://类内成员声明class Apublic:A(T val):a(val){}private:T a;return 0;运行结果:注意:类模板中的成员函数若是放在类外定义时,需要加模板参数列表。return 0;
272 0
|
11月前
|
存储 编译器 程序员
c++的类(附含explicit关键字,友元,内部类)
本文介绍了C++中类的核心概念与用法,涵盖封装、继承、多态三大特性。重点讲解了类的定义(`class`与`struct`)、访问限定符(`private`、`public`、`protected`)、类的作用域及成员函数的声明与定义分离。同时深入探讨了类的大小计算、`this`指针、默认成员函数(构造函数、析构函数、拷贝构造、赋值重载)以及运算符重载等内容。 文章还详细分析了`explicit`关键字的作用、静态成员(变量与函数)、友元(友元函数与友元类)的概念及其使用场景,并简要介绍了内部类的特性。
432 0