禁忌搜索算法(Tabu Search,TS)及其Python和MATLAB实现
禁忌搜索算法是一种现代启发式搜索方案,主要用于解决组合优化问题。该算法由George F. Lugeral于1986年首次提出,旨在增强局部搜索算法的性能,避免其陷入局部最优解。禁忌搜索利用一个称为“禁忌表”的数据结构,记住最近访问的解决方案,从而禁止在短期内回到这些解,借此探索更广泛的解空间并寻求更优解。
### 一、背景
在许多组合优化问题中,比如旅行商问题、调度问题、背包问题等,寻找最优解往往是非常复杂的。传统的启发式算法,如爬山算法,往往容易陷入局部最优解,导致整体解的优化性能不足。禁忌搜索在此背景下应运而生,目的在于通过记忆和策略来引导搜索过程,从而跳出局部最优化的困境,接近全局最优解。
### 二、原理
禁忌搜索算法的核心思想是利用禁忌表记录近期的解,在搜索过程中避免再次访问这些解。该算法可视为改进的局部搜索,允许进行“非对称”的搜索,即尽管某些解在短期内被禁止,算法仍可以探索其他潜在的解。
#### 1. 解决方案表示
通常用一个向量或一个集合表示问题的解。例如,在旅行商问题中,可以用一个城市的排列来表示一个路线解。
#### 2. 邻域结构
一个解决方案的邻域是通过对当前解进行小的变更而形成的一组解决方案。邻域结构的设计非常重要,它直接影响到搜索的效率和效果。
#### 3. 禁忌表
禁忌表是禁忌搜索的重要组成部分,主要用于记录一定数量的“禁忌”解。它通常采用先进先出(FIFO)策略,删除最早的纪录,以保持大小恒定。禁忌表的长度是一个参数,影响着算法的性能。
#### 4. 目标函数
目标函数用于评估每个解决方案的质量。禁忌搜索算法通过最大化或最小化目标函数,来引导搜索过程。
### 三、实现过程
禁忌搜索的实现过程一般包括以下几个步骤:
#### 1. 初始化
- 选定初始解,并计算其目标函数值。
- 初始化禁忌表为空。
- 设定禁忌表的大小和最大迭代次数。
#### 2. 主循环
在指定的迭代次数内执行以下步骤:
- **邻域生成**:根据当前解生成解的邻域。
- **评估邻域解**:计算邻域中所有解的目标函数值,并找出最优解。
- **禁忌检查**:检查该邻域解是否在禁忌表中。如果是,则判断是否是最优解;如果不是,则更新当前解为邻域最优解。
- **更新禁忌表**:将当前解或某个特定的属性(如交换的元素)加入禁忌表,确保在之后的搜索中不再回到该解。
- **记录最优解**:如果当前解优于历史记录,更新最优解。
#### 3. 终止条件
- 根据事先设定的终止条件,如达到最大迭代次数或在一定时间内未找到新解,来结束搜索。
#### 4. 输出结果
- 输出最优解及其目标函数值。
### 四、流程示意图
```
开始 → 初始化 (初始解、目标函数、禁忌表) →
↓
主循环 (达到最大迭代次数?)
↙ ↘
是 否
↓ ↓
输出结果 邻域生成 →
评估邻域解 →
禁忌检查 →
更新禁忌表 →
记录最优解 →
返回主循环
```
### 五、算法性能分析
禁忌搜索算法的性能常常取决于多个因素,如禁忌表的大小、邻域结构的设计以及目标函数的计算复杂度。良好的邻域结构和适当的禁忌表大小能够在巨大的解空间中有效地引导搜索过程。此外,禁忌搜索的多样性和灵活性使其可以与其他算法(如遗传算法、模拟退火等)结合,形成混合算法。
### 六、应用领域
禁忌搜索被广泛应用于各种领域,包括但不限于:
1. **旅行商问题**:解决路径最优化。
2. **调度问题**:如制造与任务调度。
3. **资源分配**:最大化利益或最小化成本。
4. **图着色问题**:解决图的最小着色问题。
5. **路径规划**:自动驾驶与机器人导航等领域。
### 七、结论
禁忌搜索算法是一种强大的优化工具,能够有效地解决大量组合优化问题。通过使用禁忌表、邻域结构等机制,它克服了传统局部搜索的局限性,探索更广泛的解空间。禁忌搜索算法的灵活性及其与其他方法的结合能力,使得其在实际应用中具有重要的价值。随着计算技术的发展,禁忌搜索算法仍将继续在各类优化问题中发挥重要作用。
Python:
import numpy as np
class TabuSearch:
def __init__(self, objective_function, initial_solution, tabu_size, max_iterations):
self.objective_function = objective_function
self.current_solution = initial_solution
self.best_solution = initial_solution
self.tabu_list = []
self.tabu_size = tabu_size
self.max_iterations = max_iterations
def find_neighbors(self, solution):
neighbors = []
# Example of generating neighbors by swapping two elements
for i in range(len(solution)):
for j in range(i + 1, len(solution)):
neighbor = solution.copy()
neighbor[i], neighbor[j] = neighbor[j], neighbor[i]
neighbors.append(neighbor)
return neighbors
def run(self):
for _ in range(self.max_iterations):
neighbors = self.find_neighbors(self.current_solution)
best_neighbor = None
best_value = float('inf')
for neighbor in neighbors:
if neighbor not in self.tabu_list:
neighbor_value = self.objective_function(neighbor)
if neighbor_value < best_value:
best_value = neighbor_value
best_neighbor = neighbor
if best_neighbor is not None:
self.current_solution = best_neighbor
if self.objective_function(self.current_solution) < self.objective_function(self.best_solution):
self.best_solution = self.current_solution
self.tabu_list.append(self.current_solution)
if len(self.tabu_list) > self.tabu_size:
self.tabu_list.pop(0)
return self.best_solution, self.objective_function(self.best_solution)
# Example usage
def objective_function(solution):
return sum(solution) # Minimize the sum for example
initial_solution = [3, 1, 4, 1, 5]
tabu_search = TabuSearch(objective_function, initial_solution, tabu_size=5, max_iterations=100)
best_solution, best_value = tabu_search.run()
print("Best Solution:", best_solution)
print("Best Value:", best_value)
MATLAB:
classdef TabuSearch
properties
objective_function
current_solution
best_solution
tabu_list
tabu_size
max_iterations
end
methods
function obj = TabuSearch(objective_function, initial_solution, tabu_size, max_iterations)
obj.objective_function = objective_function;
obj.current_solution = initial_solution;
obj.best_solution = initial_solution;
obj.tabu_list = {};
obj.tabu_size = tabu_size;
obj.max_iterations = max_iterations;
end
function neighbors = find_neighbors(obj, solution)
neighbors = [];
n = length(solution);
% Generate neighbors by swapping two elements
for i = 1:n
for j = i+1:n
neighbor = solution;
neighbor([i, j]) = neighbor([j, i]);
neighbors = [neighbors; neighbor];
end
end
end
function [best_solution, best_value] = run(obj)
for iter = 1:obj.max_iterations
neighbors = obj.find_neighbors(obj.current_solution);
best_neighbor = [];
best_value = Inf;
for i = 1:size(neighbors, 1)
neighbor = neighbors(i, :);
if ~ismember(neighbor, obj.tabu_list, 'rows')
neighbor_value = obj.objective_function(neighbor);
if neighbor_value < best_value
best_value = neighbor_value;
best_neighbor = neighbor;
end
end
end
if ~isempty(best_neighbor)
obj.current_solution = best_neighbor;
if obj.objective_function(obj.current_solution) < obj.objective_function(obj.best_solution)
obj.best_solution = obj.current_solution;
end
obj.tabu_list{end+1} = obj.current_solution; %#ok<*AGROW>
if length(obj.tabu_list) > obj.tabu_size
obj.tabu_list(1) = [];
end
end
end
best_solution = obj.best_solution;
best_value = obj.objective_function(best_solution);
end
end
end
% Example usage
objective_function = @(solution) sum(solution); % Minimize the sum for example
initial_solution = [3, 1, 4, 1, 5];
tabu_search = TabuSearch(objective_function, initial_solution, 5, 100);
[best_solution, best_value] = tabu_search.run();
disp('Best Solution:');
disp(best_solution);
disp('Best Value:');
disp(best_value);
说明
邻域生成:在 Python 和 MATLAB 示例中,使用交换两个元素的方法生成邻域解。根据具体问题,其他邻域生成策略也可以应用。
目标函数:示例中使用的目标函数是求解数组元素的和。可以根据需要替换为其他目标函数。
禁忌列表:用于存放最近访问过的解,以避免将它们纳入后续搜索。
这两个实现提供了禁忌搜索的基本框架,可以根据实际需求修改和扩充。具体的优化问题和目标函数可以自行设计。
相关文章:
禁忌搜索算法(Tabu Search,TS)及其Python和MATLAB实现
禁忌搜索算法是一种现代启发式搜索方案,主要用于解决组合优化问题。该算法由George F. Lugeral于1986年首次提出,旨在增强局部搜索算法的性能,避免其陷入局部最优解。禁忌搜索利用一个称为“禁忌表”的数据结构,记住最近访问的解决…...
Meta发布Llama 3.1 405B模型:开源与闭源模型之争的新篇章
引言 在人工智能领域,开源与闭源模型之争一直是热点话题。近日,Meta发布了最新的Llama 3.1 405B模型,以其强大的性能和庞大的参数规模,成为了开源模型中的佼佼者。本文将详细介绍Llama 3.1 405B模型的性能、功能及其在开源领域的…...
Linux网络协议深度解析:从IP到TCP/IP堆栈
Linux网络协议深度解析是一个复杂而详细的主题,它涵盖了从基本的数据包传输到复杂的协议交互。以下是对"Linux网络协议深度解析:从IP到TCP/IP堆栈"这一主题的简要解析: IP协议(Internet Protocol) •作用:…...

AWS DMS MySQL为源端,如何在更改分区的时候避免报错
问题描述: 文档[1]中描述MySQL compatible Databases作为DMS任务的源端,不支持MySQL 分区表的 DDL 更改。 在源端MySQL进行分区添加时,日志里会出现如下报错: [SOURCE_CAPTURE ]W: Cannot change partition in table members…...
Java从基础到高级特性及应用
Java,作为一门历史悠久且广泛应用的编程语言,自1995年问世以来,便以其跨平台性、面向对象、自动内存管理等特点,在软件开发领域占据了举足轻重的地位。从桌面应用到企业级系统,从移动开发到云计算服务,Java…...

JavaScript(17)——事件监听
什么是事件? 事件是在编程时系统内发生的动作或发生的事情,比如用户在网页上单击一个按钮 什么是事件监听? 就是让程序检测是否有事件产生,一旦有事件触发,就立刻调用一个函数做出响应,也称为绑定事件或…...
Dav_笔记11:SQL Tuning Overview-sql调优 之 4
开发高效的SQL语句 本节介绍了提高SQL语句效率的方法: ■验证优化程序统计信息 ■审查执行计划 ■重构SQL语句 ■重组索引 ■修改或禁用触发器和约束 ■重组数据 ■随着时间的推移维护执行计划 ■尽可能少地访问数据 验证优化程序统计信息 查询优化器在确定最佳执行…...

vue3引入openlayers
安装ol包 OpenLayers作为 ol npm包提供,它提供了官方支持的API的所有模块。 官方地址:ol npm install ol模块和子模块约定 具有CamelCase名称的OpenLayers模块提供类作为默认导出,并且可能包含其他常量或函数作为命名导出: i…...

大数据管理中心设计规划方案(可编辑的43页PPT)
引言:随着企业业务的快速发展,数据量急剧增长,传统数据管理方式已无法满足高效处理和分析大数据的需求。建立一个集数据存储、处理、分析、可视化于一体的大数据管理中心,提升数据处理能力,加速业务决策过程࿰…...

Android --- 广播
广播是什么? 一种相互通信,传递信息的机制,组件内、进程间(App之间) 如何使用广播? 组成部分 发送者-发送广播 与启动其他四大组件一样,广播发送也是使用intent发送。 设置actionÿ…...

AR 眼镜之-蓝牙电话-实现方案
目录 📂 前言 AR 眼镜系统版本 蓝牙电话 来电铃声 1. 🔱 技术方案 1.1 结构框图 1.2 方案介绍 1.3 实现方案 步骤一:屏蔽原生蓝牙电话相关功能 步骤二:自定义蓝牙电话实现 2. 💠 屏蔽原生蓝牙电话相关功能 …...

stl-set
目录 目录 内部自动有序、不含重复元素 关于能不能自己造一个cmp,还挺复杂。 访问:只能用迭代器且受限 添加元素:没有pushback,用insert 复杂度:ologn 编辑 查找元素find()࿱…...

【Stable Diffusion】(基础篇五)—— 使用SD提升分辨率
使用SD提升分辨率 本系列博客笔记主要参考B站nenly同学的视频教程,传送门:B站第一套系统的AI绘画课!零基础学会Stable Diffusion,这绝对是你看过的最容易上手的AI绘画教程 | SD WebUI 保姆级攻略_哔哩哔哩_bilibili 在前期作画的…...

5.CSS学习(浮动)
浮动(float) 是一种传统的网页布局方式,通过浮动,可以使元素脱离文档流的控制,使其横向排列。 其编写在CSS样式中。 float:none(默认值) 元素不浮动。 float:left 设置的元素在其包含…...

Spring Cloud微服务项目统一封装数据响应体
在微服务架构下,处理服务之间的通信和数据一致性是一个重要的挑战。为了提高开发效率、保证数据的一致性及简化前端开发,统一封装数据响应体是一种非常有效的实践。本文博主将介绍如何在 Spring Cloud 微服务项目中统一封装数据响应体,并分享…...

java算法day20
java算法day20 701.二叉搜索树中的插入操作450.删除二叉搜索树中的节点108 将有序数组转换为二叉搜索树 本次的题目都是用递归函数的返回值来完成,多熟悉这样的用法,很方便。 其实我感觉,涉及构造二叉树的题目,用递归函数的返回值…...
web自动化测试-python+selenium+unitest
文章目录 Web自动化测试工具1. 主流的Web自动化测试工具2. Selenium家族史 Web自动化测试环境搭建基于Python环境搭建示例:通过程序启动浏览器,并打开百度首页,暂停3秒,关闭浏览器 页面元素定位1. 如何进行元素定位?2.…...
LeetCode题练习与总结:组合两个表--175
一、题目描述 SQL Schema > Pandas Schema > 表: Person ---------------------- | 列名 | 类型 | ---------------------- | PersonId | int | | FirstName | varchar | | LastName | varchar | ---------------------- personId 是该表的主…...

数据结构:二叉搜索树(简单C++代码实现)
目录 前言 1. 二叉搜索树的概念 2. 二叉搜索树的实现 2.1 二叉树的结构 2.2 二叉树查找 2.3 二叉树的插入和中序遍历 2.4 二叉树的删除 3. 二叉搜索树的应用 3.1 KV模型实现 3.2 应用 4. 二叉搜索树分析 总结 前言 本文将深入探讨二叉搜索树这一重要的数据结构。二…...
深入理解Prompt工程
前言:因为大模型的流行,衍生出了一个小领域“Prompt工程”,不知道大家会不会跟小编一样,不就是写提示吗,这有什么难的,不过大家还是不要小瞧了Prompt工程,现在很多大模型把会“Prompt工程”作为…...
基于大模型的 UI 自动化系统
基于大模型的 UI 自动化系统 下面是一个完整的 Python 系统,利用大模型实现智能 UI 自动化,结合计算机视觉和自然语言处理技术,实现"看屏操作"的能力。 系统架构设计 #mermaid-svg-2gn2GRvh5WCP2ktF {font-family:"trebuchet ms",verdana,arial,sans-…...

K8S认证|CKS题库+答案| 11. AppArmor
目录 11. AppArmor 免费获取并激活 CKA_v1.31_模拟系统 题目 开始操作: 1)、切换集群 2)、切换节点 3)、切换到 apparmor 的目录 4)、执行 apparmor 策略模块 5)、修改 pod 文件 6)、…...

SCAU期末笔记 - 数据分析与数据挖掘题库解析
这门怎么题库答案不全啊日 来简单学一下子来 一、选择题(可多选) 将原始数据进行集成、变换、维度规约、数值规约是在以下哪个步骤的任务?(C) A. 频繁模式挖掘 B.分类和预测 C.数据预处理 D.数据流挖掘 A. 频繁模式挖掘:专注于发现数据中…...

Opencv中的addweighted函数
一.addweighted函数作用 addweighted()是OpenCV库中用于图像处理的函数,主要功能是将两个输入图像(尺寸和类型相同)按照指定的权重进行加权叠加(图像融合),并添加一个标量值&#x…...
spring:实例工厂方法获取bean
spring处理使用静态工厂方法获取bean实例,也可以通过实例工厂方法获取bean实例。 实例工厂方法步骤如下: 定义实例工厂类(Java代码),定义实例工厂(xml),定义调用实例工厂ÿ…...
C++中string流知识详解和示例
一、概览与类体系 C 提供三种基于内存字符串的流,定义在 <sstream> 中: std::istringstream:输入流,从已有字符串中读取并解析。std::ostringstream:输出流,向内部缓冲区写入内容,最终取…...
汇编常见指令
汇编常见指令 一、数据传送指令 指令功能示例说明MOV数据传送MOV EAX, 10将立即数 10 送入 EAXMOV [EBX], EAX将 EAX 值存入 EBX 指向的内存LEA加载有效地址LEA EAX, [EBX4]将 EBX4 的地址存入 EAX(不访问内存)XCHG交换数据XCHG EAX, EBX交换 EAX 和 EB…...
Go 并发编程基础:通道(Channel)的使用
在 Go 中,Channel 是 Goroutine 之间通信的核心机制。它提供了一个线程安全的通信方式,用于在多个 Goroutine 之间传递数据,从而实现高效的并发编程。 本章将介绍 Channel 的基本概念、用法、缓冲、关闭机制以及 select 的使用。 一、Channel…...

深度学习水论文:mamba+图像增强
🧀当前视觉领域对高效长序列建模需求激增,对Mamba图像增强这方向的研究自然也逐渐火热。原因在于其高效长程建模,以及动态计算优势,在图像质量提升和细节恢复方面有难以替代的作用。 🧀因此短时间内,就有不…...

vulnyx Blogger writeup
信息收集 arp-scan nmap 获取userFlag 上web看看 一个默认的页面,gobuster扫一下目录 可以看到扫出的目录中得到了一个有价值的目录/wordpress,说明目标所使用的cms是wordpress,访问http://192.168.43.213/wordpress/然后查看源码能看到 这…...