优化算法(直接搜索法)¶
本章节将讲解优化算法的使用,包括 Nelder-Mead
算法跟 Powell
算法,它们都是一种直接搜索算法。我们在 QPanda
中实现了这两个算法,OriginNelderMead
和 OriginPowell
,
这两个类都继承自 AbstractOptimizer
。
接口介绍¶
我们可以通过优化器工厂生成指定类型的优化器,例如我们指定它的类型为Nelder-Mead
optimizer = OptimizerFactory.makeOptimizer(OptimizerType.NELDER_MEAD)
#optimizer = OptimizerFactory.makeOptimizer('Nelder-Mead')
我们需要向优化器注册一个计算损失值的函数和待优化参数。
init_para = [0, 0]
optimizer.registerFunc(lossFunc, init_para)
然后设置结束条件,我们可以设置变量及函数值的收敛阈值,函数最大可调用次数,和优化迭代次数。只要满足上述结束条件,则优化结束。
optimizer.setXatol(1e-6)
optimizer.setFatol(1e-6)
optimizer.setMaxFCalls(200)
optimizer.setMaxIter(200)
然后通过exec接口执行优化,通过getResult接口获得优化后的结果。
optimizer.exec()
result = optimizer.getResult()
print(result.message)
print(" Current function value: ", result.fun_val)
print(" Iterations: ", result.iters)
print(" Function evaluations: ", result.fcalls)
print(" Optimized para: W: ", result.para[0], " b: ", result.para[1])
实例¶
给定一些散列点,我们来拟合一条直线,使得散列点到直线的距离和最小。定义直线的函数的表达式为 y = w*x + b
,接下来我们将通过使用优化算法得到w和b的优化值。 首先定义求期望的函数
from pyqpanda import *
import numpy as np
x = np.array([3.3, 4.4, 5.5, 6.71, 6.93, 4.168, 9.779, 6.182, 7.59,
2.167, 7.042, 10.791, 5.313, 7.997, 5.654, 9.27,3.1])
y = np.array([1.7, 2.76, 2.09, 3.19, 1.694, 1.573, 3.366, 2.596, 2.53,
1.221, 2.827, 3.465, 1.65, 2.904, 2.42, 2.94,1.3])
def lossFunc(para,grad,inter,fcall):
y_ = np.zeros(len(y))
for i in range(len(y)):
y_[i] = para[0] * x[i] + para[1]
loss = 0
for i in range(len(y)):
loss += (y_[i] - y[i])**2/len(y)
return ("", loss)
我们使用 Nelder-Mead
算法进行优化
optimizer = OptimizerFactory.makeOptimizer('Nelder-Mead')
init_para = [0, 0]
optimizer.registerFunc(lossFunc, init_para)
optimizer.setXatol(1e-6)
optimizer.setFatol(1e-6)
optimizer.setMaxIter(200)
optimizer.exec()
result = optimizer.getResult()
print(result.message)
print(" Current function value: ", result.fun_val)
print(" Iterations: ", result.iters)
print(" Function evaluations: ", result.fcalls)
print(" Optimized para: W: ", result.para[0], " b: ", result.para[1])

我们将散列点和拟合的直线进行绘图
import matplotlib.pyplot as plt
w = result.para[0]
b = result.para[1]
plt.plot(x, y, 'o', label = 'Training data')
plt.plot(x, w*x + b, 'r', label = 'Fitted line')
plt.legend()
plt.show()
