无约束最优化方法

最后更新于:2022-04-01 16:06:47

## 无约束最优化方法 最优化(Optimization)是工程技术、经济管理、 科学研究、社会生活中经常遇到的问题, 如:结构设计,资源分配,生产计划,运输方案等等。 最优化: 在一定条件下,寻求使目标最大(小)的决策解决优化问题的手段 • 经验积累,主观判断 • 作试验,比优劣 • 建立数学模型,求解最优策略 ### 优化问题的数学模型 无约束优化问题标准形式: 求n维决策变量x=[x1,x2,…,xn]T在可行域Ω中 使得目标函数f(x)→min,x⊆Ω⊆Rn ⇒ ![这里写图片描述](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-07-25_5795bdc2f0216.jpg "") **可行解**只满足(2),**最优解**满足(1)(2) **无约束优化**只有(1),**约束优化**需要(1)(2) ### 无约束优化的基本方法 ### 求解无约束优化的基本思路:[下山法](http://blog.csdn.net/u013007900/article/details/43525733) #### 基本思想: 和爬山法基本思想一样,在Rn中某一点,确定一个**搜索放向**然后沿着该方向的**移动步长**,得到使目标函数下降的最新点。 #### 迭代步骤: Step 1 初始化:初始点x0,终止准则等 Step 2 迭代改进:方向dk,步长αk xk+1=xk+(αd)k 且 f(xk+1)<f(xk) Step 3 终止检验:得到近优解或k+1⇒k转2> 选择dk,αk使得f下降更快⇒利用不同的算法 ### 最速下降法(梯度法) 泰勒级数: ![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-07-25_5795bdc317591.png "") 其中Δx可正可负,但必须充分接近于0。 X沿D方向移动步长α后,变为X+αD。由泰勒展开式:![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-07-25_5795bdc32ec6e.png "") 目标函数:![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-07-25_5795bdc3439d9.png "") α确定的情况下即最小化:![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-07-25_5795bdc3547a3.png "") 向量的内积何时最小?当然是两向量方向相反时。所以X移动的方向应该和梯度的方向相反。 **下降方向:**∇fT(xk)dk<0 **最速下降方向:**dk=−∇f(xk)负梯度方向 **迭代改进形式:**xk+1=xk−∇f(xk) 算法特点:在初始状态改进较快,在最优解附近改进缓慢。 求函数f(x1,x2)=x21+4x22在极小点,以初始点X0=(1,1)T。 ~~~ #include"matrix.h"//大家自己YY一个矩阵类吧 #include<iostream> #include<iomanip> #include<cmath> #include<limits> #include<cassert> using namespace std; const int SIZE=2; const double ZETA=0.001; inline double func(Matrix<double> &X){ assert(SIZE==X.getRows()); double x1=X.get(0,0); double x2=X.get(1,0); return x1*x1+4*x2*x2; } inline Matrix<double> gradient(Matrix<double> &X){ assert(SIZE==X.getRows()); double x1=X.get(0,0); double x2=X.get(1,0); Matrix<double> rect(SIZE,1); rect.put(0,0,2*x1); rect.put(1,0,8*x2); return rect; } inline Matrix<double> Hesse(Matrix<double> &X){ Matrix<double> rect(SIZE,SIZE); rect.put(0,0,2); rect.put(0,1,0); rect.put(1,0,0); rect.put(1,1,8); return rect; } int main(int agrc,char *argv[]){ Matrix<double> X(SIZE,1); X.put(0,0,1); X.put(1,0,1); int iteration=0; double value=func(X); double newValue=numeric_limits<double>::max(); while(++iteration){ Matrix<double> G=gradient(X); double factor=((G.getTranspose()*G).get(0,0))/((G.getTranspose()*Hesse(X)*G).get(0,0)); for(int i=0;i<G.getRows();++i) for(int j=0;j<G.getColumns();++j) G.put(i,j,G.get(i,j)*factor); Matrix<double> newX=X-G; //cout<<"X=["<<newX.get(0,0)<<","<<newX.get(1,0)<<"]"<<endl; newValue=func(newX); if(fabs(newValue-value)<ZETA) break; else{ X=newX; value=newValue; } //cout<<"本次迭代找到的极小值"<<value<<endl; } cout<<"迭代次数"<<iteration<<endl; cout<<"极小值"<<value<<endl; return 0; } ~~~ > 这份算法使用的是利用Hesse矩阵和泰勒公式计算最优步长 若f(X)具有二阶连续偏导,由泰勒展开式可得: ![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-07-25_5795bdc368a0b.png "") H是f(X)的Hesse矩阵。 ![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-07-25_5795bdc37ab36.png "") 可得最优步长: ![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-07-25_5795bdc38d688.png "") g是f(X)的梯度矩阵。 此时: ![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-07-25_5795bdc39f52f.png "") 可见最速下降法中最优步长不仅与梯度有关,而且与Hesse矩阵有关。 ### 牛顿迭代法 将f(xk+1)在xk点作泰勒展开至二阶项,用d替代dk,有 f(xk+1)=f(xk+d)=f(xk)+∇fT(xk)d+12dT∇2f(xk)d 求d使得f(xk+1)极小⇒右端对d的导数为0⇒∇f(xk)+∇2f(xk)d=0 可得到> **牛顿方程**∇2f(xk)d = −∇f(xk) **牛顿方向**dk = −(∇2f(xk))−1∇f(xk) **迭代格式**xk+1 = xk−(∇2f(xk))−1∇f(xk) 其中∇2f(xk)为[Hessen矩阵](http://baike.baidu.com/link?url=E2wdH1BhTKFos1JYd_NhHvkmGAyABsHJDIgSyygvzTuoAs41tvb4OGzzWThbUMwobdBAEPpS7nzia-GQUHHf7a) 可见x的搜索方向是−(∇2f(xk))−1∇f(xk),函数值在此方向上下降,就需要它与梯度方向相反,即−∇f(xk)(∇2f(xk))−1∇f(xk)<0,所以要求在每一个迭代点上Hessian矩阵必须是正定的。 算法特点: 牛顿法是二次收敛的,并且收敛阶数是2。一般目标函数在最优点附近呈现为二次函数,于是可以想像最优点附近用牛顿迭代法收敛是比较快的。而在开始搜索的几步,我们用梯度下降法收敛是比较快的。将两个方法融合起来可以达到满意的效果。 收敛快是牛顿迭代法最大的优点,但也有致命的缺点:Hessian矩阵及其逆的求解计算量大,更何况在某个迭代点Xk处Hessian矩阵的逆可能根本就不存在(即Hessian 矩阵奇异),这样无法求得Xk+1。 求f(x1,x2)=(x1−2)4+(x1−2x2)2的极小点,初始点取X=(0,3)T ~~~ #include"matrix.h" #include<iostream> #include<iomanip> #include<cmath> #include<limits> #include<cassert> using namespace std; const int SIZE=2; const double ZETA=0.001; inline double func(Matrix<double> &X){ assert(SIZE==X.getRows()); double x1=X.get(0,0); double x2=X.get(1,0); return pow(x1-2,4)+pow(x1-2*x2,2); } inline Matrix<double> gradient(Matrix<double> &X){ assert(SIZE==X.getRows()); double x1=X.get(0,0); double x2=X.get(1,0); Matrix<double> rect(SIZE,1); rect.put(0,0,4*pow(x1-2,3)+2*(x1-2*x2)); rect.put(1,0,-4*(x1-2*x2)); return rect; } inline Matrix<double> Hesse(Matrix<double> &X){ Matrix<double> rect(SIZE,SIZE); double x1=X.get(0,0); double x2=X.get(1,0); rect.put(0,0,12*pow(x1-2,2)+2); rect.put(0,1,-4); rect.put(1,0,-4); rect.put(1,1,8); return rect; } int main(int agrc,char *argv[]){ Matrix<double> X(SIZE,1); X.put(0,0,0); X.put(1,0,3); int iteration=0; double value=func(X); double newValue=numeric_limits<double>::max(); while(++iteration){ Matrix<double> G=gradient(X); Matrix<double> H=Hesse(X); Matrix<double> newX=X-H.getInverse()*G; cout<<"X=["<<newX.get(0,0)<<","<<newX.get(1,0)<<"]"<<endl; newValue=func(newX); if(fabs(newValue-value)<ZETA) break; else{ X=newX; value=newValue; } cout<<"本次迭代找到的极小值"<<value<<endl; } cout<<"迭代次数"<<iteration<<endl; cout<<"极小值"<<value<<endl; return 0; } ~~~ ### 拟牛顿法 Hessian矩阵在拟牛顿法中是不计算的,拟牛顿法是构造与Hessian矩阵相似的正定矩阵,这个构造方法,使用了目标函数的梯度(一阶导数)信息和两个点的“位移”(Xk−Xk−1)来实现。 有人会说,是不是用Hessian矩阵的近似矩阵来代替Hessian矩阵,会导致求解效果变差呢?事实上,效果反而通常会变好。 拟牛顿法与牛顿法的迭代过程一样,仅仅是各个Hessian矩阵的求解方法不一样。 在远离极小值点处,Hessian矩阵一般不能保证正定,使得目标函数值不降反升。而拟牛顿法可以使目标函数值沿下降方向走下去,并且到了最后,在极小值点附近,可使构造出来的矩阵与Hessian矩阵“很像”了,这样,拟牛顿法也会具有牛顿法的二阶收敛性。 对目标函数f(X)做二阶泰勒展开: f(x)=f(xk+1)+(x−xk+1)T∇f(xk+1)+12(x−xk+1)TG(xk+1)(x−xk+1) 其中∇2f(xk) ⇔G(xk+1) 上式两边对x求导: ∇f(x)=∇f(xk+1)+G(xk+1)(x−xk+1) 当x=xk的时候,有 G−1(xk+1)[∇f(xk+1)−∇f(xk)]=xk+1−xk 这里我们用Hk表示在点xk处的Hessian矩阵的**逆矩阵** Hk+1[∇fd(xk+1)−∇f(xk)]=xk+1−xk ......(5) (5)式就是**拟牛顿方程** 下面提供两种拟牛顿法 ![这里写图片描述](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-07-25_5795bdc3b5dd4.jpg "") ### 非线性最小二乘(Least Square)拟合 问题: 给定(ti,yi),i=1,…n, 拟合一个函数y=f(t,x), 其中x为待定的参数向量, f 对x非线性。 记误差ri(x)=yi−f(ti,x)     r(x)=(r1(x),…,rn(x))T 优化模型: minxR(x)=rT(x)r(x) 记r(x)的雅克比行列式为J(x)=(∂ri∂xj)n×m ∇R=2J(x)Tr(x)∇2R=2J(x)TJ(x)+2SS=∑ni=1ri(x)∇2ri(x)∇2ri(x)=(∂2ri∂xk∂xl)n×m 牛顿法要计算Hessian矩阵,其中S计算量大 若f 对x线性, 则化为线性最小二乘拟合, 此时 S=0 特定算法考虑如何忽略或近似矩阵S #### Gauss-Newton算法(GN):忽略矩阵S ∇R=2J(x)Tr(x)∇2R=2J(x)TJ(x) 牛顿方程:∇2f(xk)d=−∇f(xk) f用R代替,下降方向dk满足 J(x)TJ(x)dk=−J(x)Tr(x) GN算法收敛性特点: S=∑ni=1ri(x) 收敛性依赖f对x的线性程度,即偏差r的大小(”小残差问题“) #### LM算法: Levenbery (1944), Marquardt (1963) G-N算法修正 J(x)TJ(x)dk=−J(x)Tr(x) ⇓ (J(x)TJ(x)+αkI)dk=−J(x)Tr(x) 其中αk>0为修正参数,防止JTJ出现病态。 LM算法特点 dk位于GN方向(αk很小)和负梯度方向(αk很大)之间 LM算法的另一种观点 min∥r(xk)+J(xk)(x−xk)∥2 s.t.     ∥x−xk∥2≤hk    (∗)置信域 其解为 (J(x)TJ(x)+αkI)dk=−J(x)Tr(x) 若GN方向满足约束(*),αk=0,否则ak>0 置信域算法特点: 先确定最长步长hk,再确定方向dk
';