最优化问题的三要素是决策变量、目标函数和约束条件。
线性规划(Linear programming),是研究线性约束条件下线性目标函数的极值问题的优化方法,常用于解决利用现有的资源得到最优决策的问题。
简单的线性规划问题可以用 Lingo软件求解,Matlab、Python 中也有求解线性规划问题的库函数或求解器,很容易学习和使用,并不需要用模拟退火算法。但是,由一般线性规划问题所衍生的整数规划、混合规划、0/1规划、二次规划、非线性规划、组合优化问题,则并不是调用某个库函数都能处理的。而模拟退火算法在很多复杂问题中具有较好的适应性,可以作为一种入门的通用智能算法来学习。
也就是说,如果只是处理线性规划问题,就不要用模拟退火算法了。但如果是现有方法无法处理的复杂优化问题,或者对某类、某个优化问题你不知道用什么方法处理了,这时用模拟退火算法还是能解决的。
本文使用惩罚函数法,分析模拟退火算法处理线性规划问题,相关内容也适用于非线性规划问题。
线性规划问题是约束优化问题,而模拟退火算法则更适合处理无约束优化问题。对于优化问题中的约束条件,模拟退火算法有几种常用的处理方法:
1. 决策变量取值的上下限约束。此类约束条件比较容易处理,只要设定初始解、新解在决策变量取值的上下限之间就可以解决。例如,(1)设置产生新解的随机数的上下限为决策变量的上下限,即 [Xmin, Xmax];(2)设置产生新解的随机数的上下限为当前解与决策变量的上下限,即 [Xnow, Xmax];(3)通过条件判断,当新解超出决策变量上下限,则令其取上下限,即 xNew = max(min(xNew, xMax), xMin)。当然,这些处理方式,都会影响随机数的概率分布,因而也影响模拟退火算法的优化性能,在此不做深入讨论。
2. 检验法处理不等式约束问题。在模拟退火算法的迭代过程中,将每次产生的新解代入每个不等式约束函数,判断是否满足约束条件;如果新解不满足约束条件,则舍弃这个新解,返回重新产生一个新解进行检验,直到产生的新解满足全部约束条件为止。这个方法的思路简单,每次迭代都在可行域内进行,但是对于约束条件众多、苛刻的复杂问题,多次产生的新解都不能满足约束条件,会使计算时间很长,甚至停滞不前。
3. 消元法处理等式约束问题。对于等式约束,很难通过随机产生的新解满足约束条件,通常不能使用检验法处理。消元法是通过解方程将等式约束中的某个决策变量表示为其它决策变量的线性关系后,代入目标函数和不等式约束条件中,从而消去该约束条件。消元法不仅解决了等式约束问题,而且减少了决策变量的数量,从而有效简化了优化问题的复杂度,是一举两得的处理方法。但是,对于非线性等式约束,求解非线性方程组也是非常困难的,消元法并不是普遍都能适用的。
4. 更为通用的处理约束条件的方法是惩罚函数法,以下进行介绍。
惩罚函数法是一类常用的处理约束条件的技术,在模拟退火算法中处理约束条件非常有效。方法的思想是将约束条件转化为惩罚函数,附加在原有的目标函数上构造新的目标函数;当不满足约束条件时,通过惩罚函数使新的目标函数变差而被舍弃。
惩罚函数法有外点法和内点法。外点法对可行域外的点(即不满足约束的点)施加惩罚,对可行域内部的点不惩罚,从而使迭代点向可行域D逼近。 内点法是在可行域内部进行搜索,约束边界起到类似围墙的作用,使目标函数无法穿过,就把搜索点限制在可行域内了,因此只适用于不等式约束。
考虑约束优化问题:
$$
min: f(X)
$$
满足限制
$$
c_i(X)\leq0
$$
惩罚函数法将问题转化成如下无约束问题
$$
min: L_k(X)=f(X)+\sigma_k\sum_{i}{g(c_i(X))}
$$
其中
$$
g(c_i(X))=max(0,c_i(X))^2
$$
在上述方程,g(ci(X)) 称为外部罚函数,sigma(k) 称为惩罚因子。在每一次迭代中,sigma(k) 都增大,然后求解该无约束问题。将每一次迭代的结果将组成一个序列,此序列的极限即为原约束问题的解。
虽然对于线性规划问题并不推荐使用模拟退火算法求解。但为了便于理解,本文仍使用之前的线性规划问题作为处理约束条件的案例。对于非线性规划问题,以及非线性约束问题,处理方法都是类似的,将在后续进行介绍。
某厂生产甲乙两种饮料,每百箱甲饮料需用原料6千克、工人10名,获利10万元;每百箱乙饮料需用原料5千克、工人20名,获利9万元。
今工厂共有原料60千克、工人150名,又由于其他条件所限甲饮料产量不超过8百箱。
(1)问如何安排生产计划,即两种饮料各生产多少使获利最大?
决策变量:
x1:甲饮料产量(单位:百箱)
x2:乙饮料产量(单位:百箱)
目标函数:
max fx = 10x1 + 9x2
约束条件:
6x1 + 5x2 <= 60
10x1 + 20x2 <= 150
取值范围:
给定条件:x1, x2 >= 0,x1 <= 8
推导条件:由 x1,x2>=0 和 10*x1+20*x2<=150 可知:0<=x1<=15;0<=x2<=7.5
因此,0 <= x1<=8,0 <= x2<=7.5
构造惩罚函数:
p1 = (max(0, 6*x1+5*x2-60))**2
p2 = (max(0, 10*x1+20*x2-150))**2
说明:如存在等式约束,例如:x1 + 2*x2 = m,也可以转化为惩罚函数:
p3 = (x1+2*x2-m)**2
P(x) = p1 + p2 + ...
构造增广目标函数:
L(x,m(k)) = min(fx) + m(k)*P(x)
m(k):惩罚因子,随迭代次数 k 逐渐增大
在模拟退火算法中,m(k) 随外循环迭代次数逐渐增大,但在内循环中应保持不变。
# 模拟退火算法 程序:惩罚函数法求解线性规划问题 # Program: SimulatedAnnealing_v2.py # Purpose: Simulated annealing algorithm for function optimization # v2.0: 使用惩罚函数法处理约束问题 # Copyright 2021 YouCans, XUPT # Crated:2021-05-01 # -*- coding: utf-8 -*- import math # 导入模块 import random # 导入模块 import pandas as pd # 导入模块 YouCans, XUPT import numpy as np # 导入模块 numpy,并简写成 np import matplotlib.pyplot as plt from datetime import datetime # 子程序:定义优化问题的目标函数 def cal_Energy(X, nVar, mk): # m(k):惩罚因子,随迭代次数 k 逐渐增大 p1 = (max(0, 6*X[0]+5*X[1]-60))**2 p2 = (max(0, 10*X[0]+20*X[1]-150))**2 fx = -(10*X[0]+9*X[1]) return fx+mk*(p1+p2) # 子程序:模拟退火算法的参数设置 def ParameterSetting(): cName = "funcOpt" # 定义问题名称 nVar = 2 # 给定自变量数量,y=f(x1,..xn) xMin = [0, 0] # 给定搜索空间的下限,x1_min,..xn_min xMax = [8, 7.5] # 给定搜索空间的上限,x1_max,..xn_max tInitial = 100.0 # 设定初始退火温度(initial temperature) tFinal = 1 # 设定终止退火温度(stop temperature) alfa = 0.98 # 设定降温参数,T(k)=alfa*T(k-1) meanMarkov = 100 # Markov链长度,也即内循环运行次数 scale = 0.5 # 定义搜索步长,可以设为固定值或逐渐缩小 return cName, nVar, xMin, xMax, tInitial, tFinal, alfa, meanMarkov, scale # 模拟退火算法 def OptimizationSSA(nVar,xMin,xMax,tInitial,tFinal,alfa,meanMarkov,scale): # ====== 初始化随机数发生器 ====== randseed = random.randint(1, 100) random.seed(randseed) # 随机数发生器设置种子,也可以设为指定整数 # ====== 随机产生优化问题的初始解 ====== xInitial = np.zeros((nVar)) # 初始化,创建数组 for v in range(nVar): # random.uniform(min,max) 在 [min,max] 范围内随机生成一个实数 xInitial[v] = random.uniform(xMin[v], xMax[v]) # 调用子函数 cal_Energy 计算当前解的目标函数值 fxInitial = cal_Energy(xInitial, nVar, 1) # m(k):惩罚因子,初值为 1 # ====== 模拟退火算法初始化 ====== xNew = np.zeros((nVar)) # 初始化,创建数组 xNow = np.zeros((nVar)) # 初始化,创建数组 xBest = np.zeros((nVar)) # 初始化,创建数组 xNow[:] = xInitial[:] # 初始化当前解,将初始解置为当前解 xBest[:] = xInitial[:] # 初始化最优解,将当前解置为最优解 fxNow = fxInitial # 将初始解的目标函数置为当前值 fxBest = fxInitial # 将当前解的目标函数置为最优值 print('x_Initial:{:.6f},{:.6f},\tf(x_Initial):{:.6f}'.format(xInitial[0], xInitial[1], fxInitial)) recordIter = [] # 初始化,外循环次数 recordFxNow = [] # 初始化,当前解的目标函数值 recordFxBest = [] # 初始化,最佳解的目标函数值 recordPBad = [] # 初始化,劣质解的接受概率 kIter = 0 # 外循环迭代次数,温度状态数 totalMar = 0 # 总计 Markov 链长度 totalImprove = 0 # fxBest 改善次数 nMarkov = meanMarkov # 固定长度 Markov链 # ====== 开始模拟退火优化 ====== # 外循环,直到当前温度达到终止温度时结束 tNow = tInitial # 初始化当前温度(current temperature) while tNow >= tFinal: # 外循环,直到当前温度达到终止温度时结束 # 在当前温度下,进行充分次数(nMarkov)的状态转移以达到热平衡 kBetter = 0 # 获得优质解的次数 kBadAccept = 0 # 接受劣质解的次数 kBadRefuse = 0 # 拒绝劣质解的次数 # ---内循环,循环次数为Markov链长度 for k in range(nMarkov): # 内循环,循环次数为Markov链长度 totalMar += 1 # 总 Markov链长度计数器 # ---产生新解 # 产生新解:通过在当前解附近随机扰动而产生新解,新解必须在 [min,max] 范围内 # 方案 1:只对 n元变量中的一个进行扰动,其它 n-1个变量保持不变 xNew[:] = xNow[:] v = random.randint(0, nVar-1) # 产生 [0,nVar-1]之间的随机数 xNew[v] = xNow[v] + scale * (xMax[v]-xMin[v]) * random.normalvariate(0, 1) # random.normalvariate(0, 1):产生服从均值为0、标准差为 1 的正态分布随机实数 xNew[v] = max(min(xNew[v], xMax[v]), xMin[v]) # 保证新解在 [min,max] 范围内 # ---计算目标函数和能量差 # 调用子函数 cal_Energy 计算新解的目标函数值 fxNew = cal_Energy(xNew, nVar, kIter) deltaE = fxNew - fxNow # ---按 Metropolis 准则接受新解 # 接受判别:按照 Metropolis 准则决定是否接受新解 if fxNew < fxNow: # 更优解:如果新解的目标函数好于当前解,则接受新解 accept = True kBetter += 1 else: # 容忍解:如果新解的目标函数比当前解差,则以一定概率接受新解 pAccept = math.exp(-deltaE / tNow) # 计算容忍解的状态迁移概率 if pAccept > random.random(): accept = True # 接受劣质解 kBadAccept += 1 else: accept = False # 拒绝劣质解 kBadRefuse += 1 # 保存新解 if accept == True: # 如果接受新解,则将新解保存为当前解 xNow[:] = xNew[:] fxNow = fxNew if fxNew < fxBest: # 如果新解的目标函数好于最优解,则将新解保存为最优解 fxBest = fxNew xBest[:] = xNew[:] totalImprove += 1 scale = scale*0.99 # 可变搜索步长,逐步减小搜索范围,提高搜索精度 # ---内循环结束后的数据整理 # 完成当前温度的搜索,保存数据和输出 pBadAccept = kBadAccept / (kBadAccept + kBadRefuse) # 劣质解的接受概率 recordIter.append(kIter) # 当前外循环次数 recordFxNow.append(round(fxNow, 4)) # 当前解的目标函数值 recordFxBest.append(round(fxBest, 4)) # 最佳解的目标函数值 recordPBad.append(round(pBadAccept, 4)) # 最佳解的目标函数值 if kIter%10 == 0: # 模运算,商的余数 print('i:{},t(i):{:.2f}, badAccept:{:.6f}, f(x)_best:{:.6f}'.\ format(kIter, tNow, pBadAccept, fxBest)) # 缓慢降温至新的温度,降温曲线:T(k)=alfa*T(k-1) tNow = tNow * alfa kIter = kIter + 1 fxBest = cal_Energy(xBest, nVar, kIter) # 由于迭代后惩罚因子增大,需随之重构增广目标函数 # ====== 结束模拟退火过程 ====== print('improve:{:d}'.format(totalImprove)) return kIter,xBest,fxBest,fxNow,recordIter,recordFxNow,recordFxBest,recordPBad # 结果校验与输出 def ResultOutput(cName,nVar,xBest,fxBest,kIter,recordFxNow,recordFxBest,recordPBad,recordIter): # ====== 优化结果校验与输出 ====== fxCheck = cal_Energy(xBest, nVar, kIter) if abs(fxBest - fxCheck)>1e-3: # 检验目标函数 print("Error 2: Wrong total millage!") return else: print("\nOptimization by simulated annealing algorithm:") for i in range(nVar): print('\tx[{}] = {:.6f}'.format(i,xBest[i])) print('\n\tf(x):{:.6f}'.format(cal_Energy(xBest,nVar,0))) # ====== 优化结果写入数据文件 ====== nowTime = datetime.now().strftime('%m%d%H%M') # '02151456' fileName = "..\data\{}_{}.dat".format(cName,nowTime)# 数据文件的地址和文件名 optRecord = { "iter":recordIter, "FxNow":recordFxNow, "FxBest":recordFxBest, "PBad":recordPBad} df_Record = pd.DataFrame(optRecord) df_Record.to_csv(fileName, index=False, encoding="utf_8_sig") with open(fileName, 'a+', encoding="utf_8_sig") as fid: fid.write("\nOptimization by simulated annealing algorithm:") for i in range(nVar): fid.write('\n\tx[{}] = {:.6f}'.format(i,xBest[i])) fid.write('\n\tf(x):{:.6f}'.format(cal_Energy(xBest,nVar,0))) print("写入数据文件: %s 完成。" % fileName) # ====== 优化结果图形化输出 ====== plt.figure(figsize=(6, 4), facecolor='#FFFFFF') # 创建一个图形窗口 plt.title('Optimization result: {}'.format(cName)) # 设置图形标题 plt.xlim((0, kIter)) # 设置 x轴范围 plt.xlabel('iter') # 设置 x轴标签 plt.ylabel('f(x)') # 设置 y轴标签 plt.plot(recordIter, recordFxNow,'b-', label='FxNow') # 绘制 FxNow 曲线 plt.plot(recordIter, recordFxBest, 'r-', label='FxBest') # 绘制 FxBest 曲线 # plt.plot(recordIter,recordPBad,'r-',label='pBadAccept') # 绘制 pBadAccept 曲线 plt.legend() # 显示图例 plt.show() return # 主程序 def main(): # 参数设置,优化问题参数定义,模拟退火算法参数设置 [cName, nVar, xMin, xMax, tInitial, tFinal, alfa, meanMarkov, scale] = ParameterSetting() # print([nVar, xMin, xMax, tInitial, tFinal, alfa, meanMarkov, scale]) # 模拟退火算法 [kIter,xBest,fxBest,fxNow,recordIter,recordFxNow,recordFxBest,recordPBad] \ = OptimizationSSA(nVar,xMin,xMax,tInitial,tFinal,alfa,meanMarkov,scale) # print(kIter, fxNow, fxBest, pBadAccept) # 结果校验与输出 ResultOutput(cName, nVar,xBest,fxBest,kIter,recordFxNow,recordFxBest,recordPBad,recordIter) if __name__ == '__main__': main()
Optimization by simulated annealing algorithm: x[0] = 6.577964 x[1] = 4.111469 f(x):-102.782857
(1)胡山鹰,陈丙珍,非线性规划问题全局优化的模拟退火法,清华大学学报,1997,37(6):5-9
(2)李歧强,具有约束指导的模拟退火算法,系统工程,2001,19(3):49-55
版权说明:
原创作品
Copyright 2021 YouCans, XUPT
Crated:2021-05-01