当前位置: 首页 > news >正文

禁忌搜索算法(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实现

禁忌搜索算法是一种现代启发式搜索方案&#xff0c;主要用于解决组合优化问题。该算法由George F. Lugeral于1986年首次提出&#xff0c;旨在增强局部搜索算法的性能&#xff0c;避免其陷入局部最优解。禁忌搜索利用一个称为“禁忌表”的数据结构&#xff0c;记住最近访问的解决…...

Meta发布Llama 3.1 405B模型:开源与闭源模型之争的新篇章

引言 在人工智能领域&#xff0c;开源与闭源模型之争一直是热点话题。近日&#xff0c;Meta发布了最新的Llama 3.1 405B模型&#xff0c;以其强大的性能和庞大的参数规模&#xff0c;成为了开源模型中的佼佼者。本文将详细介绍Llama 3.1 405B模型的性能、功能及其在开源领域的…...

Linux网络协议深度解析:从IP到TCP/IP堆栈

Linux网络协议深度解析是一个复杂而详细的主题&#xff0c;它涵盖了从基本的数据包传输到复杂的协议交互。以下是对"Linux网络协议深度解析&#xff1a;从IP到TCP/IP堆栈"这一主题的简要解析&#xff1a; IP协议&#xff08;Internet Protocol&#xff09; •作用:…...

AWS DMS MySQL为源端,如何在更改分区的时候避免报错

问题描述&#xff1a; 文档[1]中描述MySQL compatible Databases作为DMS任务的源端&#xff0c;不支持MySQL 分区表的 DDL 更改。 在源端MySQL进行分区添加时&#xff0c;日志里会出现如下报错&#xff1a; [SOURCE_CAPTURE ]W: Cannot change partition in table members…...

Java从基础到高级特性及应用

Java&#xff0c;作为一门历史悠久且广泛应用的编程语言&#xff0c;自1995年问世以来&#xff0c;便以其跨平台性、面向对象、自动内存管理等特点&#xff0c;在软件开发领域占据了举足轻重的地位。从桌面应用到企业级系统&#xff0c;从移动开发到云计算服务&#xff0c;Java…...

JavaScript(17)——事件监听

什么是事件&#xff1f; 事件是在编程时系统内发生的动作或发生的事情&#xff0c;比如用户在网页上单击一个按钮 什么是事件监听&#xff1f; 就是让程序检测是否有事件产生&#xff0c;一旦有事件触发&#xff0c;就立刻调用一个函数做出响应&#xff0c;也称为绑定事件或…...

Dav_笔记11:SQL Tuning Overview-sql调优 之 4

开发高效的SQL语句 本节介绍了提高SQL语句效率的方法: ■验证优化程序统计信息 ■审查执行计划 ■重构SQL语句 ■重组索引 ■修改或禁用触发器和约束 ■重组数据 ■随着时间的推移维护执行计划 ■尽可能少地访问数据 验证优化程序统计信息 查询优化器在确定最佳执行…...

vue3引入openlayers

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

大数据管理中心设计规划方案(可编辑的43页PPT)

引言&#xff1a;随着企业业务的快速发展&#xff0c;数据量急剧增长&#xff0c;传统数据管理方式已无法满足高效处理和分析大数据的需求。建立一个集数据存储、处理、分析、可视化于一体的大数据管理中心&#xff0c;提升数据处理能力&#xff0c;加速业务决策过程&#xff0…...

Android --- 广播

广播是什么&#xff1f; 一种相互通信&#xff0c;传递信息的机制&#xff0c;组件内、进程间&#xff08;App之间&#xff09; 如何使用广播&#xff1f; 组成部分 发送者-发送广播 与启动其他四大组件一样&#xff0c;广播发送也是使用intent发送。 设置action&#xff…...

AR 眼镜之-蓝牙电话-实现方案

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

stl-set

目录 目录 内部自动有序、不含重复元素 关于能不能自己造一个cmp&#xff0c;还挺复杂。 访问&#xff1a;只能用迭代器且受限 添加元素&#xff1a;没有pushback&#xff0c;用insert 复杂度&#xff1a;ologn ​编辑 查找元素find&#xff08;&#xff09;&#xff1…...

【Stable Diffusion】(基础篇五)—— 使用SD提升分辨率

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

5.CSS学习(浮动)

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

Spring Cloud微服务项目统一封装数据响应体

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

java算法day20

java算法day20 701.二叉搜索树中的插入操作450.删除二叉搜索树中的节点108 将有序数组转换为二叉搜索树 本次的题目都是用递归函数的返回值来完成&#xff0c;多熟悉这样的用法&#xff0c;很方便。 其实我感觉&#xff0c;涉及构造二叉树的题目&#xff0c;用递归函数的返回值…...

web自动化测试-python+selenium+unitest

文章目录 Web自动化测试工具1. 主流的Web自动化测试工具2. Selenium家族史 Web自动化测试环境搭建基于Python环境搭建示例&#xff1a;通过程序启动浏览器&#xff0c;并打开百度首页&#xff0c;暂停3秒&#xff0c;关闭浏览器 页面元素定位1. 如何进行元素定位&#xff1f;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工程

前言&#xff1a;因为大模型的流行&#xff0c;衍生出了一个小领域“Prompt工程”&#xff0c;不知道大家会不会跟小编一样&#xff0c;不就是写提示吗&#xff0c;这有什么难的&#xff0c;不过大家还是不要小瞧了Prompt工程&#xff0c;现在很多大模型把会“Prompt工程”作为…...

Vim 调用外部命令学习笔记

Vim 外部命令集成完全指南 文章目录 Vim 外部命令集成完全指南核心概念理解命令语法解析语法对比 常用外部命令详解文本排序与去重文本筛选与搜索高级 grep 搜索技巧文本替换与编辑字符处理高级文本处理编程语言处理其他实用命令 范围操作示例指定行范围处理复合命令示例 实用技…...

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造&#xff0c;完美适配AGV和无人叉车。同时&#xff0c;集成以太网与语音合成技术&#xff0c;为各类高级系统&#xff08;如MES、调度系统、库位管理、立库等&#xff09;提供高效便捷的语音交互体验。 L…...

突破不可导策略的训练难题:零阶优化与强化学习的深度嵌合

强化学习&#xff08;Reinforcement Learning, RL&#xff09;是工业领域智能控制的重要方法。它的基本原理是将最优控制问题建模为马尔可夫决策过程&#xff0c;然后使用强化学习的Actor-Critic机制&#xff08;中文译作“知行互动”机制&#xff09;&#xff0c;逐步迭代求解…...

FastAPI 教程:从入门到实践

FastAPI 是一个现代、快速&#xff08;高性能&#xff09;的 Web 框架&#xff0c;用于构建 API&#xff0c;支持 Python 3.6。它基于标准 Python 类型提示&#xff0c;易于学习且功能强大。以下是一个完整的 FastAPI 入门教程&#xff0c;涵盖从环境搭建到创建并运行一个简单的…...

【解密LSTM、GRU如何解决传统RNN梯度消失问题】

解密LSTM与GRU&#xff1a;如何让RNN变得更聪明&#xff1f; 在深度学习的世界里&#xff0c;循环神经网络&#xff08;RNN&#xff09;以其卓越的序列数据处理能力广泛应用于自然语言处理、时间序列预测等领域。然而&#xff0c;传统RNN存在的一个严重问题——梯度消失&#…...

【机器视觉】单目测距——运动结构恢复

ps&#xff1a;图是随便找的&#xff0c;为了凑个封面 前言 在前面对光流法进行进一步改进&#xff0c;希望将2D光流推广至3D场景流时&#xff0c;发现2D转3D过程中存在尺度歧义问题&#xff0c;需要补全摄像头拍摄图像中缺失的深度信息&#xff0c;否则解空间不收敛&#xf…...

CVE-2020-17519源码分析与漏洞复现(Flink 任意文件读取)

漏洞概览 漏洞名称&#xff1a;Apache Flink REST API 任意文件读取漏洞CVE编号&#xff1a;CVE-2020-17519CVSS评分&#xff1a;7.5影响版本&#xff1a;Apache Flink 1.11.0、1.11.1、1.11.2修复版本&#xff1a;≥ 1.11.3 或 ≥ 1.12.0漏洞类型&#xff1a;路径遍历&#x…...

Scrapy-Redis分布式爬虫架构的可扩展性与容错性增强:基于微服务与容器化的解决方案

在大数据时代&#xff0c;海量数据的采集与处理成为企业和研究机构获取信息的关键环节。Scrapy-Redis作为一种经典的分布式爬虫架构&#xff0c;在处理大规模数据抓取任务时展现出强大的能力。然而&#xff0c;随着业务规模的不断扩大和数据抓取需求的日益复杂&#xff0c;传统…...

Python爬虫实战:研究Restkit库相关技术

1. 引言 1.1 研究背景与意义 在当今信息爆炸的时代,互联网上存在着海量的有价值数据。如何高效地采集这些数据并将其应用于实际业务中,成为了许多企业和开发者关注的焦点。网络爬虫技术作为一种自动化的数据采集工具,可以帮助我们从网页中提取所需的信息。而 RESTful API …...

在Zenodo下载文件 用到googlecolab googledrive

方法&#xff1a;Figshare/Zenodo上的数据/文件下载不下来&#xff1f;尝试利用Google Colab &#xff1a;https://zhuanlan.zhihu.com/p/1898503078782674027 参考&#xff1a; 通过Colab&谷歌云下载Figshare数据&#xff0c;超级实用&#xff01;&#xff01;&#xff0…...