Matlab数学建模算法之模拟退火算法(SA)详解

news2024/9/22 19:32:00

🔗 运行环境:Matlab

🚩 撰写作者:左手の明天

🥇 精选专栏:《python》

🔥  推荐专栏:《算法研究》

🔐#### 防伪水印——左手の明天 ####🔐

💗 大家好🤗🤗🤗,我是左手の明天!好久不见💗

💗今天分享matlab数学建模算法——模拟退火算法💗

📆  最近更新:2023 年 12 月 24 日,左手の明天的第 310 篇原创博客

📚 更新于专栏:matlab

🔐#### 防伪水印——左手の明天 ####🔐


目录

一、模拟退火算法

1 基本思想

2 基本步骤

二、算法流程​​​​​​​

三、解决局部最优解

四、模拟退火算法在数学建模的应用

五、旅行商问题的matlab实现

1 旅行商问题

2 算法设计流程

(1)TSP问题的解空间和初始解

(2)新解的产生

(3)目标函数

(4)目标函数差

(5)Metropolis准则

3 matlab实现

六、MATLAB 模拟退火算法的示例代码

七、模拟退火算法优缺点


一、模拟退火算法

模拟退火算法(Simulated Annealing,SA)是一种模拟固体降温过程的最优化算法。其模拟的过程是首先将固体加温至某一温度,固体内部的粒子随温度上升慢慢变为无序的状态,内能增大,然后让其慢慢冷却,温度下降时,内部的粒子慢慢趋于有序,达到一种平衡态,最后达到常温时成为基态,此时内能减为最小。

模拟退火算法的基本原理分为三部分:初始解、解空间和目标函数。初始解是算法迭代的起点,试验表明,模拟退火算法是健壮的,即最终解的求得最优解并不十分依赖初始解的选取,从而可任意选择一个初始解。当然,如果初始解选择得当可以加快找到全局最优解。解空间一般是离散的可行解的集合。

1 基本思想

模拟退火算法的基本思想是将待求解的问题视为一个能量系统,系统的当前状态为解的状态,能量值为解的目标函数值,系统的状态转移依赖于Metropolis准则

在模拟退火过程中,首先从某一初始解出发,将其视为系统的初始状态,然后通过随机选择不同的状态转移方式,不断迭代产生新的状态。在每一步转移中,根据目标函数值的差异和温度参数的大小,决定是否接受新状态作为当前状态。如果新状态的目标函数值更优,则一定概率上接受新状态;如果新状态的目标函数值更劣,则根据概率判断是否接受新状态。

模拟退火算法的精髓在于引入了温度参数,通过不断降低温度来引导系统从劣解逐步向最优解过渡。在这个过程中,通过Metropolis准则控制状态转移概率,使得算法在迭代过程中能够跳出局部最优解,向全局最优解靠近。

总的来说,模拟退火算法是一种基于概率的优化算法,通过模拟固体降温的过程,利用随机性和概率性寻找全局最优解,具有较好的鲁棒性和通用性。

2 基本步骤

模拟退火算法的基本步骤如下:

(1)初始化参数。包括初始温度、降温速率、终止温度和初始解等。

(2)产生新解。在当前解的邻域内产生一个新解。

(3)接受新解。计算当前解与新解之间的差异,如果新解更优,则接受它;否则,以一定的概率接受它。

(4)降温。根据设定的降温速率降低温度。

(5)终止判断。如果温度降低到终止温度以下,则停止搜索,输出最优解。

二、算法流程

三、解决局部最优解

模拟退火算法通过以下方式解决局部最优解的问题:

  1. 引入温度参数:模拟退火算法中的温度参数控制着算法的搜索过程。在高温状态下,算法更倾向于接受较差的解,这样可以探索更广阔的解空间;随着温度的降低,算法逐渐趋于保守,只接受更优的解。这种温度的变化过程有助于算法跳出局部最优解,向全局最优解过渡。
  2. Metropolis准则:模拟退火算法中的Metropolis准则是决定是否接受新状态的关键。当新状态比当前状态更优时,一定概率上接受新状态;当新状态更劣时,根据概率判断是否接受新状态。这样可以避免算法陷入局部最优解,而是通过概率的方式探索整个解空间。
  3. 随机性:模拟退火算法中的随机性使得算法在搜索过程中能够探索更多的解空间,而不是只停留在局部最优解附近。这种随机性有助于算法跳出局部最优解,向全局最优解靠近。

综上所述,模拟退火算法通过引入温度参数、Metropolis准则和随机性等方式,解决了局部最优解的问题。这种基于概率的优化算法能够跳出局部最优解,向全局最优解靠近,从而在各种优化问题中取得了较好的效果。

四、模拟退火算法在数学建模的应用

以下是模拟退火算法在数学建模竞赛中的一些应用场景:

  1. 旅行商问题(TSP):在这个经典问题中,模拟退火算法可以用来找到一条使得旅行商访问所有城市的总距离最短的路径。
  2. 背包问题:模拟退火算法可以用来解决背包问题,即在满足背包容量限制的前提下,找到一种物品组合使得背包中物品总价值最大。
  3. 图的着色问题:模拟退火算法可以用来解决图的最小着色问题,即给图的顶点分配最少的颜色,使得相邻的顶点颜色不同。
  4. 聚类问题:模拟退火算法可以用来解决数据聚类问题,即在给定数据集和聚类数量的前提下,找到一种聚类方式使得类内距离最小,类间距离最大。
  5. 车辆路径问题(VRP):模拟退火算法可以用来解决车辆路径问题,即在满足车辆容量和行驶距离限制的前提下,找到一种物品配送路径使得配送成本最小。
  6. 调度问题:模拟退火算法可以用来解决生产调度、车间调度等问题,目标是在满足生产约束的前提下,最小化生产时间、等待时间或者其他相关指标。

五、旅行商问题的matlab实现

1 旅行商问题

一名商人要到n个不同的城市去推销商品,每2个城市l和j之间的距离为d,如何选择一条路径使得商人每个城市走一遍后回到起点所走的路径最短。

例如:有52个城市,已知每个城市的坐标,求每个城市走一遍后回到起点,所走的路径最短。

2 算法设计流程

(1)TSP问题的解空间和初始解

(2)新解的产生

二变换法:任选序号u,v(设u<v<n),变换u,v之间的访问顺序

三变换法:任选序号u,v,w(设u<=v<w),将u,v之间的路径插到w之后访问

(3)目标函数

(4)目标函数差

计算变换前的解和变换后目标函数的差值:

(5)Metropolis准则

以新解与当前解的目标函数差定义接受概率,即

3 matlab实现

clear	
	clc
	a = 0.99;	% 温度衰减函数的参数
	t0 = 97; tf = 3; t = t0;
	Markov_length = 10000;	% Markov链长度
	coordinates = [
1	 565.0	 575.0;	2	  25.0	 185.0;	3	 345.0	 750.0;	
4	 945.0	 685.0;	5	 845.0	 655.0;	6	 880.0	 660.0;	
7	  25.0	 230.0;	8	 525.0	1000.0;	9	 580.0	1175.0;	
10	 650.0	1130.0;	11	1605.0	 620.0;	12	1220.0	 580.0;	
13	1465.0	 200.0;	14	1530.0	   5.0;	15	 845.0	 680.0;	
16	 725.0	 370.0;	17	 145.0	 665.0;	18	 415.0	 635.0;	
19	 510.0	 875.0;	20	 560.0	 365.0;	21	 300.0	 465.0;	
22	 520.0	 585.0;	23	 480.0	 415.0;	24	 835.0	 625.0;	
25	 975.0	 580.0;	26	1215.0	 245.0;	27	1320.0	 315.0;	
28	1250.0	 400.0;	29	 660.0	 180.0;	30	 410.0	 250.0;	
31	 420.0	 555.0;	32	 575.0	 665.0;	33	1150.0	1160.0;	
34	 700.0	 580.0;	35	 685.0	 595.0;	36	 685.0	 610.0;	
37	 770.0	 610.0;	38	 795.0	 645.0;	39	 720.0	 635.0;	
40	 760.0	 650.0;	41	 475.0	 960.0;	42	  95.0	 260.0;	
43	 875.0	 920.0;	44	 700.0	 500.0;	45	 555.0	 815.0;	
46	 830.0	 485.0;	47	1170.0	  65.0;	48	 830.0	 610.0;	
49	 605.0	 625.0;	50	 595.0	 360.0;	51	1340.0	 725.0;	
52	1740.0	 245.0;	
];
	coordinates(:,1) = [];
	amount = size(coordinates,1); 	% 城市的数目
	% 通过向量化的方法计算距离矩阵
	dist_matrix = zeros(amount, amount);
	coor_x_tmp1 = coordinates(:,1) * ones(1,amount);
	coor_x_tmp2 = coor_x_tmp1';
	coor_y_tmp1 = coordinates(:,2) * ones(1,amount);
	coor_y_tmp2 = coor_y_tmp1';
	dist_matrix = sqrt((coor_x_tmp1-coor_x_tmp2).^2 + ...
					(coor_y_tmp1-coor_y_tmp2).^2);

	sol_new = 1:amount;         % 产生初始解
% sol_new是每次产生的新解;sol_current是当前解;sol_best是冷却中的最好解;
	E_current = inf;E_best = inf; 		% E_current是当前解对应的回路距离;
% E_new是新解的回路距离;
% E_best是最优解的
	sol_current = sol_new; sol_best = sol_new;          
	p = 1;

	while t>=tf
		for r=1:Markov_length		% Markov链长度
			% 产生随机扰动
			if (rand < 0.5)	% 随机决定是进行两交换还是三交换
				% 两交换
				ind1 = 0; ind2 = 0;
				while (ind1 == ind2)
					ind1 = ceil(rand.*amount);
					ind2 = ceil(rand.*amount);
				end
				tmp1 = sol_new(ind1);
				sol_new(ind1) = sol_new(ind2);
				sol_new(ind2) = tmp1;
			else
				% 三交换
				ind1 = 0; ind2 = 0; ind3 = 0;
				while (ind1 == ind2) || (ind1 == ind3) ...
					|| (ind2 == ind3) || (abs(ind1-ind2) == 1)
					ind1 = ceil(rand.*amount);
					ind2 = ceil(rand.*amount);
					ind3 = ceil(rand.*amount);
				end
				tmp1 = ind1;tmp2 = ind2;tmp3 = ind3;
				% 确保ind1 < ind2 < ind3
				if (ind1 < ind2) && (ind2 < ind3)
					;
				elseif (ind1 < ind3) && (ind3 < ind2)
					ind2 = tmp3;ind3 = tmp2;
				elseif (ind2 < ind1) && (ind1 < ind3)
					ind1 = tmp2;ind2 = tmp1;
				elseif (ind2 < ind3) && (ind3 < ind1) 
					ind1 = tmp2;ind2 = tmp3; ind3 = tmp1;
				elseif (ind3 < ind1) && (ind1 < ind2)
					ind1 = tmp3;ind2 = tmp1; ind3 = tmp2;
				elseif (ind3 < ind2) && (ind2 < ind1)
					ind1 = tmp3;ind2 = tmp2; ind3 = tmp1;
				end
				
				tmplist1 = sol_new((ind1+1):(ind2-1));
				sol_new((ind1+1):(ind1+ind3-ind2+1)) = ...
					sol_new((ind2):(ind3));
				sol_new((ind1+ind3-ind2+2):ind3) = ...
					tmplist1;
			end

			%检查是否满足约束
			
			% 计算目标函数值(即内能)
			E_new = 0;
			for i = 1 : (amount-1)
				E_new = E_new + ...
					dist_matrix(sol_new(i),sol_new(i+1));
			end
			% 再算上从最后一个城市到第一个城市的距离
			E_new = E_new + ...
				dist_matrix(sol_new(amount),sol_new(1));
			
			if E_new < E_current
				E_current = E_new;
				sol_current = sol_new;
				if E_new < E_best
% 把冷却过程中最好的解保存下来
					E_best = E_new;
					sol_best = sol_new;
				end
			else
				% 若新解的目标函数值小于当前解的,
				% 则仅以一定概率接受新解
				if rand < exp(-(E_new-E_current)./t)
					E_current = E_new;
					sol_current = sol_new;
				else	
					sol_new = sol_current;
				end
			end
		end
		t=t.*a;		% 控制参数t(温度)减少为原来的a倍
	end

	disp('最优解为:')
	disp(sol_best)
	disp('最短距离:')
	disp(E_best)

六、MATLAB 模拟退火算法的示例代码

以下是一个简单的 MATLAB 模拟退火算法的示例代码:

function [x_best, f_best] = simulated_annealing(f, lb, ub, init_temp, alpha, max_iter)
% SIMULATED_ANNEALING 模拟退火算法
%   f: 目标函数
%   lb: 变量的下界
%   ub: 变量的上界
%   init_temp: 初始温度
%   alpha: 温度衰减系数
%   max_iter: 最大迭代次数
 
% 初始化
x = lb + (ub-lb).*rand(size(lb)); % 随机生成初始解
f_x = f(x); % 计算目标函数值
temp = init_temp; % 初始温度
x_best = x; % 最佳解
f_best = f_x; % 最佳目标函数值
 
for iter = 1:max_iter
    % 生成新解
    x_new = x + randn(size(x)).*exp(-alpha.*temp); % 根据温度生成新解
    f_new = f(x_new); % 计算新解的目标函数值
    
    % 接受新解的准则
    if f_new < f_x || rand < exp(-alpha.*temp*(f_new-f_x))
        x = x_new; % 更新解
        f_x = f_new; % 更新目标函数值
        if f_x < f_best % 更新最佳解和最佳目标函数值
            x_best = x;
            f_best = f_x;
        end
    end
    temp = max(temp - alpha, 0.01); % 温度衰减
end
end

使用该函数的示例代码:

function y = f(x)
% 目标函数,这里选择简单的平方和函数 y = sum(x.^2)
y = sum(x.^2);
end
 
lb = -10; % 变量的下界为 -10
ub = 10; % 变量的上界为 10
init_temp = 1000; % 初始温度为 1000
alpha = 0.95; % 温度衰减系数为 0.95
max_iter = 1000; % 最大迭代次数为 1000
 
[x_best, f_best] = simulated_annealing(@f, lb, ub, init_temp, alpha, max_iter);
fprintf('最佳解:%f\n', x_best);
fprintf('最佳目标函数值:%f\n', f_best);

七、模拟退火算法优缺点

模拟退火算法的优点包括:

  1. 通用性和简单性:模拟退火算法的实现比较简单,通用性强,适合解决各种不同类型的问题,尤其是一维和多维的优化问题。
  2. 概率保证性:模拟退火算法具有概率意义上的全局最优解,即对于一般的问题,它最终都能收敛到全局最优解,而不仅仅是局部最优解。
  3. 鲁棒性强:模拟退火算法对初始解的依赖性不强,因此对问题的敏感度较低,具有较强的鲁棒性。
  4. 能够有效避免局部最优解:通过引入随机因素,模拟退火算法能够跳出局部最优解,向全局最优解靠近。

然而,模拟退火算法也存在一些缺点:

  1. 计算量大:模拟退火算法需要进行大量的迭代和计算,尤其在问题规模较大时,计算量会显著增加,导致算法的运行时间较长。
  2. 对参数敏感:模拟退火算法的参数选择对其性能影响较大,如果参数选择不当,可能会导致算法性能不佳。
  3. 对某些问题收敛速度慢:对于一些复杂的问题,模拟退火算法可能需要较长时间才能收敛到全局最优解。
  4. 参数降温过程复杂:模拟退火算法中的降温过程需要精心设计,如果降温过程不合理,可能会导致算法性能不佳。

综上所述,模拟退火算法在解决各种优化问题时具有一定的优势和适用性,但也存在一些需要改进的方面。在实际应用中,需要根据具体问题选择合适的算法参数和实现方式,以获得最佳的优化效果。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1381196.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

分布式事务:构建无障碍的云原生应用的完美解决方案

目录 一、前言 二、分布式事务概述 2.1 什么是分布式事务 2.2 分布式事务的挑战 2.3 分布式事务的分类 三、传统解决方案分析 3.1 两阶段提交协议&#xff08;2PC&#xff09; 3.2 三阶段提交协议&#xff08;3PC&#xff09; 3.3 补偿事务 3.4 其他传统解决方案 四…

什么是梅尼埃病?

梅尼埃病&#xff08;Meniere’s Disease&#xff0c;MD&#xff09;作为耳鼻咽喉科常见的眩晕疾病&#xff0c;是一种原因不明的、以膜迷路积水为主要病理特征的内耳疾病&#xff0c;临床表现为反复的发作性眩晕、波动性听力下降、耳鸣和耳闷胀感&#xff0c;发病率约为5~20/1…

paypal贝宝怎么绑卡支付

一、PayPal是什么 PayPal是一个很多国家地区通用的支付渠道&#xff0c;我们可以把它理解为一项在线服务&#xff0c;相当于美国版的支付宝。你可以通过PayPal进行汇款和收款&#xff0c;相比传统的电汇和西联那类的汇款方式&#xff0c;PayPal更加简单和容易&#xff0c;被很…

WebRTC入门:基础的核心协议与概念(二十三)

简介: CSDN博客专家,专注Android/Linux系统,分享多mic语音方案、音视频、编解码等技术,与大家一起成长! 优质专栏:Audio工程师进阶系列【原创干货持续更新中……】🚀 优质专栏:多媒体系统工程师系列【原创干货持续更新中……】🚀 人生格言: 人生从来没有捷径,只…

如何将.NET 8.0的ASP.NET Core Web API部署成Windows服务

写在前面 前面写了一篇关于将.NET应用转换成Windows服务的方法&#xff0c;其实真正的目的是为了探索如何将Asp.Net Core Web Api 部署成Windows 服务。基于上一篇的基础&#xff0c;只需把创建 WebApplication 的代码放到 BackgroundService 的ExecuteAsync方法中即可。 其中…

redis夯实之路-集群详解

Redis有单机模式和集群模式。 集群是 Redis 提供的分布式数据库方案&#xff0c;集群通过分片( sharding )来实现数据共享&#xff0c;并提供复制和故障转移。集群模式可以有多个 master 。使用集群模式可以进一步提升 Redis 性能&#xff0c;分布式部署实现高可用性&#xff…

快速了解——逻辑回归及模型评估方法

一、逻辑回归 应用场景&#xff1a;解决二分类问题 1、sigmoid函数 1. 公式&#xff1a; 2. 作用&#xff1a;把 (-∞&#xff0c;∞) 映射到 (0&#xff0c; 1) 3. 数学性质&#xff1a;单调递增函数&#xff0c;拐点在x0&#xff0c;y0.5的位置 4. 导函数公式&#xff1a;f…

MySQL 和 Redis 如何保证数据一致性,通过MySQL的binlog实现

1、简介 MySQL 和 Redis 如何保证数据一致性&#xff0c;目前大多讨论的是先更新Redis后更新MySQL&#xff0c;还是先更新MySQL 后更新Redis&#xff0c;这两种方式在实际的应用场景中都不能确保数据的完全一致性&#xff0c;在某些情况下会出现问题&#xff0c;本文介绍使用 C…

Linux Debian12使用VSCode和Python搭建flask开发环境

一、安装VSCode 在Linux Debian12系统上安装VSCode教程可以参考网上相关教程。 二、安装Python 打开VSCode&#xff0c;安装python和python扩展包&#xff0c;如下图所示&#xff1a; 三、创建Python虚拟环境 1.新建文件夹testFlask 2.用vscode打开文件夹testFlask&#xf…

C++/WinRT 简介

C/WinRT 是 Windows 运行时 (WinRT) API 的完全标准新式 C17 语言投影&#xff0c;以基于标头文件的库的形式实现&#xff0c;旨在为你提供对新式 Windows API 的一流访问。 利用 C/WinRT&#xff0c;你可以采用任何符合标准的 C17 编译器创作和使用 Windows 运行时 API。 Wind…

[VSCode] VSCode 常用快捷键

文章目录 VSCode 源代码编辑器VSCode 常用快捷键分类汇总01 编辑02 导航03 调试04 其他05 重构06 测试07 扩展08 选择09 搜索10 书签11 多光标12 代码片段13 其他 VSCode 源代码编辑器 官网&#xff1a;https://code.visualstudio.com/ 下载地址&#xff1a;https://code.visua…

谷歌最新医学领域LLM大模型:AMIE

2024年1月11日Google 研究院发布最新医疗大模型AMIE&#xff1a;用于诊断医学推理和对话的研究人工智能系统。 文章链接&#xff1a;Articulate Medical Intelligence Explorer (AMIE) giuthub&#xff1a;目前代码未开源 关于大模型之前有过一篇总结&#xff1a;大语言模型(L…

C++11 14 17内存管理

智能指针 unique_ptr 初始化 访问和移动赋值 重置和移动内存资源 自定义删除器 shared_ptr 原理 自定义删除器 分配器allocator和new重载 new表达式原理 operator new delete placement new new (buf) 是一种 "placement new" 的使用方式&#xff0c;它允许在已…

Qt QSQlite数据库插入字符串中存在单个双引号或单个单引号解决方案

1. 前言 当进行数据库写入或更新时&#xff0c;有时会遇到存在字符串中包含单个双引号或者单引号。 2. 单引号和双引号""作用 在数据库中&#xff0c;字符串常量时需要用一对英文单引号或英文双引号""将字符串常量括起来。 比如&#xff1a; select * …

vivado 使用源文件

使用源文件 概述 源文件包括从AMD IP添加的设计源、知识产权&#xff08;IP&#xff09;源目录、RTL设计源、从系统添加的数字信号处理&#xff08;DSP&#xff09;源生成器工具和IP子系统&#xff0c;也称为块设计&#xff0c;由IP集成商创建AMD Vivado的功能™ 设计套件。源…

微机原理常考填空总结

hello大家好我是吃个西瓜嘤&#xff0c;这篇节只总结微机原理常考填空题都是干货展示常出现的易错点以及微机原理注意事项。 以下仅代表个人发言 #微机原理 正文开始&#xff1a; 1&#xff0c;区分JZ&#xff0c;JNZ技巧 也就是D70用JZ&#xff1b;D71用JNZ。 JZ;条件ZF1时…

Java SE入门及基础(10)

switch选择结构 1. 概念 switch 表示开关的意思&#xff0c;为了帮助理解&#xff0c;下面以线路为例&#xff0c;进行解释说明 上图中表示一条带有多个开关的线路&#xff0c;当开关打开时&#xff0c;该开关所控制的灯即被点亮。 2. 语法规则 switch ( 表达式 ){ //…

秒懂百科,C++如此简单丨第十二天:ASCLL码

目录 必看信息 Everyday English &#x1f4dd;ASCLL码是什么&#xff1f; &#x1f4dd;ASCLL码表 &#x1f4dd;利用ASCLL码实现大写转小写 &#x1f4dd;小试牛刀 总结 必看信息 ▶本篇文章由爱编程的小芒果原创&#xff0c;未经许可&#xff0c;严禁转载。 ▶本篇文…

springBoot-Dev tools、Lombok、Spring initializr

1、lombok &#xff08;idea已经有快捷键实现&#xff09; 可以简化Bean的开发&#xff0c;使用注解的形式替代get、set方法&#xff0c;toString方法&#xff0c;全参数构造器&#xff0c;无参构造器&#xff0c;重写equals、hashCode方法。 没卵用&#xff0c;知道即可 1.1配…

让java程序就像脚本一样去写工具

背景&#xff1a; 接触了各种语言之后发现&#xff0c;java还是比go&#xff0c;.netcore之类的简单&#xff0c;成熟&#xff0c;我最终选择了jenkinsshelljava去部署我们的代码&#xff0c;此时很多人可能去使用js或者python之类的去写部署逻辑&#xff0c;毕竟java每次打包…