一、模拟退火算法简介
模拟退火算法属于一种优化算法,是局部搜索算法的扩展,和局部搜索算法不同的地方是模拟退火算法以一定的概率选择领域中目标值较大的非最优解。模拟退火算法以优化问题的求解与物理系统退火过程的相似性为基础,利用Metropolis算法适当地控制温度的下降过程来实现模拟退火,从而达到求全局优化的问题。
二、模拟退火算法应用举例
求解函数y = 11*sin(x) + 7*cos(5*x)在[-3,3]内的最大值
matlab代码:
%% SA 模拟退火: 求解函数y = 11*sin(x) + 7*cos(5*x)在[-3,3]内的最大值(动画演示)
tic
clear; clc
%% 绘制函数的图形
x = -3:0.1:3;
y = 11*sin(x) + 7*cos(5*x);
figure
plot(x,y,'b-')
hold on % 不关闭图形,继续在上面画图
%% 参数初始化
narvs = 1; % 变量个数
T0 = 100; % 初始温度
T = T0; % 迭代中温度会发生改变,第一次迭代时温度就是T0
maxgen = 200; % 最大迭代次数
Lk = 100; % 每个温度下的迭代次数
alfa = 0.95; % 温度衰减系数
x_lb = -3; % x的下界
x_ub = 3; % x的上界
%% 随机生成一个初始解
x0 = zeros(1,narvs);
for i = 1: narvs
x0(i) = x_lb(i) + (x_ub(i)-x_lb(i))*rand(1);
end
y0 = Obj_fun1(x0); % 计算当前解的函数值
h = scatter(x0,y0,'*r'); % scatter是绘制二维散点图的函数(这里返回h是为了得到图形的句柄,未来我们对其位置进行更新)
%% 定义一些保存中间过程的量,方便输出结果和画图
max_y = y0; % 初始化找到的最佳的解对应的函数值为y0
MAXY = zeros(maxgen,1); % 记录每一次外层循环结束后找到的max_y (方便画图)
%% 模拟退火过程
for iter = 1 : maxgen % 外循环, 我这里采用的是指定最大迭代次数
for i = 1 : Lk % 内循环,在每个温度下开始迭代
y = randn(1,narvs); % 生成1行narvs列的N(0,1)随机数
z = y / sqrt(sum(y.^2)); % 根据新解的产生规则计算z
x_new = x0 + z*T; % 根据新解的产生规则计算x_new的值
% 如果这个新解的位置超出了定义域,就对其进行调整
for j = 1: narvs
if x_new(j) < x_lb(j)
r = rand(1);
x_new(j) = r*x_lb(j)+(1-r)*x0(j);
elseif x_new(j) > x_ub(j)
r = rand(1);
x_new(j) = r*x_ub(j)+(1-r)*x0(j);
end
end
x1 = x_new; % 将调整后的x_new赋值给新解x1
y1 = Obj_fun1(x1); % 计算新解的函数值
if y1 > y0 % 如果新解函数值大于当前解的函数值
x0 = x1; % 更新当前解为新解
y0 = y1;
else
p = exp(-(y0 - y1)/T); % 根据Metropolis准则计算一个概率
if rand(1) < p % 生成一个随机数和这个概率比较,如果该随机数小于这个概率
x0 = x1; % 更新当前解为新解
y0 = y1;
end
end
% 判断是否要更新找到的最佳的解
if y0 > max_y % 如果当前解更好,则对其进行更新
max_y = y0; % 更新最大的y
best_x = x0; % 更新找到的最好的x
end
end
MAXY(iter) = max_y; % 保存本轮外循环结束后找到的最大的y
T = alfa*T; % 温度下降
pause(0.01) % 暂停一段时间(单位:秒)后再接着画图
h.XData = x0; % 更新散点图句柄的x轴的数据(此时解的位置在图上发生了变化)
h.YData = Obj_fun1(x0); % 更新散点图句柄的y轴的数据(此时解的位置在图上发生了变化)
end
disp('最佳的位置是:'); disp(best_x)
disp('此时最优值是:'); disp(max_y)
pause(0.5)
h.XData = []; h.YData = []; % 将原来的散点删除
scatter(best_x,max_y,'*r'); % 在最大值处重新标上散点
title(['模拟退火找到的最大值为', num2str(max_y)]) % 加上图的标题
%% 画出每次迭代后找到的最大y的图形
figure
plot(1:maxgen,MAXY,'b-');
xlabel('迭代次数');
ylabel('y的值');
toc
python代码:
import numpy as np
import pandas as pd
import matplotlib as plt
x = np.linspace(-3,3,1000)
MAXY = []
def y_value(x):
return 11*np.sin(x) + 7*np.cos(5*x)
def y_sum(y):
sum = 0
for i in y:
sum = sum+i*i
return sum
narvs = 1 #变量个数
T0 = 100 #初始温度
T = T0 #温度会随时间发生变化
maxgen = 200 #最大迭代次数即外循环迭代次数
LK = 100 #在每个温度下迭代的次数即内循环
alfa = 0.95 #温度衰减的系数
x_min = -3
x_max = 3
x0 = np.zeros((1,narvs))
y0 = y_value(x0)
max_y = y0
best_x = x0
#模拟退火过程
for ite in range(maxgen+1):
for i in range(LK+1):
y = np.random.randn(narvs)
z = y/np.sqrt(y_sum(y))
x_new = x0 + z*T
#判断新解是否超出定义域
for j in range(narvs+1):
if x_new < x_min:
r = np.random.rand()
x_new = r*x_min+(1-r)*x0
elif x_new > x_max:
r = np.random.rand()
x_new = r * x_max+(1-r)*x0
x1 = x_new
y1 = y_value(x1)
#比较新解与当前解
if y0 < y1:
x0 = x1
y0 = y1
else:
p = np.exp(-(y0-y1)/T)
if np.random.rand() < p:
x0 = x1
y0 = y1
if y0 > max_y:
max_y = y0
best_x = x0
MAXY.append(max_y)
T = alfa*T
print(y0)
print(x0)