模拟退火算法应用——求解二元函数的最小值(极小值)

仅作自己学习使用


一、问题

在这里插入图片描述

二、代码

clear
clc

T1 = cputime;
xmax = 5;
xmin = -5;
ymax = 5;
ymin = -5;
L = 20;     % 马尔科夫链长度
dt = 0.998; % 降温系数
S = 0.02;   % 步长因子
T = 200;    % 初始温度
TZ = 1e-8;  % 容差
Tmin = 0.01;% 最低温度
P = 0;      % Metropolis接受准则中接受的点的个数
PreX = rand*(xmax - xmin) + xmin;   % (设置初始x的位置)上一个搜索的x值
PreY = rand*(ymax - ymin) + ymin;   % (设置初始y的位置)上一个搜索的y值
PreBestX = PreX;    % 上一个最优秀的x值
PreBestY = PreY;    % 上一个最优秀的y值
PreX = rand*(xmax - xmin) + xmin;   % 上一个搜索的x值
PreY = rand*(ymax - ymin) + ymin;   % 上一个搜索的y值
BestX = PreX;       % 最优秀的X值
BestY = PreY;       % 最优秀的y值
trace = [];         % 用于记录历代最优解的函数值
deta = abs(eval(BestX,BestY) - eval(PreBestX,PreBestY));    % 上一次最小值和这次最小值的差的绝对值
while (deta>TZ) && (T>0.01)
    % 在当前温度下迭代L次数
    for i = 1: L
        % 在当前点附件随机的选取下一点
        P = 0;
        while P==0
            NextX = PreX + S*(rand*(xmax-xmin) + xmin);
            NextY = PreY + S*(rand*(ymax-ymin) + ymin);
            if (NextX >= xmin && NextX <= xmax && NextY >= ymin && NextY <= ymax)
                P = 1;
            end
        end

        % 判断选取的这个(x,y)对应的函数值是否比上一个选出来的最优值更优秀
        if(eval(NextX,NextY)<eval(BestX,BestY))
            % 保留上一个最优解
            PreBestX = BestX;
            PreBestY = BestY;
            % 更新最优解
            BestX = NextX;
            BestY = NextY;
        end

        % Metropolis接受准测 接受过程
        if(eval(NextX,NextY) < eval(PreX,PreY))
            % 当前解更优秀,接受新解
            PreX = NextX;
            PreY = NextY;
            P = P +1;
        else
            % 当前解更差,概率接受更差的解
            P1 = exp((eval(PreX,PreY)-eval(NextX,NextY))/T); % 接受的概率
            if P1 > rand
                PreX = NextX;
                PreY = NextY;
                P = P + 1;
            end
        end
        trace = [trace eval(BestX,BestY)];
    end
    deta = abs(eval(BestX,BestY) - eval(PreBestX,PreBestY));
    % 本次退火过程结束,温度下降
    T = T * dt;
end
T2 = cputime;
timeConsume = T2 - T1;
%% 绘图适配值最优化曲线
figure(1)
disp("最小值点在:")
BestX
BestY
disp("最小值为:")
eval(BestX,BestY)
plot(trace,LineWidth=1,Color=[0.68 0.52 0.96])
xlabel("迭代次数")
ylabel("目标函数值")
title("适配值最优化曲线","时间消耗:" + timeConsume )
%% 做出函数图像
figure(2)
x = -5:0.01:5;
y = -5:0.01:5;
N = length(x);
for i = 1 : N
    for j = 1 : N
        z(i,j) = 5*cos(x(i)*y(j)) + x(i)*y(j) + y(j)^3;
    end
end
colormap('jet')
mesh(x,y,z)
grid on
xlabel('x')
ylabel('y')
zlabel('f(x,y)')
hold on 
plot3(BestX,BestY,eval(BestX,BestY),'r*')
text(BestX,BestY,eval(BestX,BestY), '最小值点', 'FontSize', 15, 'HorizontalAlignment', 'left');

%% 评估函数(目标函数)
function result = eval(x,y)
    result = 5 * cos(x*y) + x*y + y^3;
end

三、效果

3.1 适配值最优化曲线

适配值最优化曲线

3.2 多次求解结果在函数图像中的表现

在这里插入图片描述
图中五角星就是求解出来的极值点,发现仍然会有部分会落在局部极值点的情况,陷入了局部最优。

四、问题

       跑过的朋友可以发现,如果按照上述代码,画出来的极值点应该是这样的:
在这里插入图片描述
那么我做了什么调整呢:我把绘图点的x,y坐标交换了一下:

plot3(BestX,BestY,eval(BestX,BestY),'rp',MarkerSize=10,LineWidth=2)

为什么会交换呢?因为我发现x,y的值求反了,BestX里边存的是Y的坐标,BestY里边存放的是X坐标;

还有一个问题是,关于x,y的随机跳动:

NextX = PreX + S*(rand*(xmax-xmin) + xmin);
NextY = PreY + S*(rand*(ymax-ymin) + ymin);

它是与温度无关的,在我求解一维函数的极值问题中我也提到了。

请知道bug的朋友在评论区留言,我将及时改正!

  • 10
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
以下是一个简单的遗传算法求解二元函数最小值的 Matlab 代码示例: ```matlab % 遗传算法求解二元函数最小值 % 目标函数:f(x,y) = 100*(y-x^2)^2 + (1-x)^2 % 取值范围:-5 <= x,y <= 5 clear clc % 初始化参数 popSize = 100; % 种群大小 chromLen = 32; % 染色体长度 pc = 0.7; % 交叉概率 pm = 0.01; % 变异概率 maxGen = 200; % 最大迭代次数 % 生成初始种群 pop = randi([0,1],popSize,chromLen); % 循环迭代 for i = 1:maxGen % 适应度计算 x = -5 + bi2de(pop(:,1:16))/2^16*10; y = -5 + bi2de(pop(:,17:32))/2^16*10; fitness = 100*(y-x.^2).^2 + (1-x).^2; % 最优解 [bestFit,idx] = min(fitness); bestX = x(idx); bestY = y(idx); % 选择 cumFitness = cumsum(fitness)/sum(fitness); newPop = zeros(popSize,chromLen); for j = 1:popSize idx = find(cumFitness >= rand,1); newPop(j,:) = pop(idx,:); end % 交叉 for j = 1:2:popSize if rand < pc cpos = randi(chromLen-1); newPop(j,[cpos+1:chromLen]) = pop(j+1,[cpos+1:chromLen]); newPop(j+1,[cpos+1:chromLen]) = pop(j,[cpos+1:chromLen]); end end % 变异 for j = 1:popSize for k = 1:chromLen if rand < pm newPop(j,k) = 1-newPop(j,k); end end end pop = newPop; % 显示结果 disp(['迭代次数:',num2str(i),',最优解:',num2str(bestFit),',x:',num2str(bestX),',y:',num2str(bestY)]); end ``` 其中,`bi2de` 函数用于将二进制数转化为十进制数,`randi` 函数用于生成随机整数,`cumsum` 函数用于计算累加和,`rand` 函数用于生成随机数。在循环迭代的过程中,首先计算种群中每个个体的适应度,然后根据适应度进行选择、交叉和变异操作,最后更新种群。在每次迭代结束后,输出当前迭代次数、最优解及其对应的 x 和 y 值。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

亲爱的老吉先森

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值