使用Scipy.Optimize.Minimize进行矩阵约束

时间:2015-01-08 22:58:24

标签: python optimization numpy matrix scipy

我尝试使用python的scipy.optimize来使用SLSQP算法最小化某些功能。优化似乎工作得很好,不受约束,只有一个矩阵约束,但是当我添加第二个矩阵约束时,我得到一个错误。

import nlopt
import numpy as np
from scipy.optimize import minimize

n = 3
n_sim = 10000
risk_aversion = 5

Y = np.random.normal(loc=0, scale=1, size=(n_sim, n))

mu_Y = np.mean(Y, axis=0).reshape(1, n)
sigma_Y = np.cov(Y, rowvar=0, bias=1)

x0 = np.ones((n, 1)) / n

Aeq = np.ones((1, n))
beq = 1
A =  np.array([[1, 1, 0], [-1, -1, 0]])
b = np.array([[0.25], [-0.5]])

def func(x, mu, sigma, risk_aversion):
    return -np.dot(mu, x) + 0.5 * risk_aversion * np.dot(np.dot(x.T, sigma), x)
def func_deriv(x, sigma, risk_aversion):
    return risk_aversion * np.dot(sigma, x)

c_ = {'type': 'eq', 'fun' : lambda x: np.dot(Aeq, x) - beq, 'jac' : lambda x: Aeq}
b_ = [(0,1) for i in range(n)]

这个带有一个约束和边界的版本可以正常工作

res = minimize(lambda x: func(x, mu_Y, sigma_Y, risk_aversion), x0, 
               jac=lambda x: func_deriv(x, sigma_Y, risk_aversion), 
               constraints=c_, bounds=b_, method='SLSQP', options={'disp': True})

然而,这两个约束版本给我一个错误

d_ = (c_,
        {'type': 'ineq', 'fun' : lambda x: np.dot(A, x) - b, 'jac' : lambda x: A})

res2 = minimize(lambda x: func(x, mu_Y, sigma_Y, risk_aversion), x0, 
                jac=lambda x: func_deriv(x, sigma_Y, risk_aversion), 
                constraints=d_, bounds=b_, method='SLSQP', options={'disp': True})

1 个答案:

答案 0 :(得分:0)

如果我使用

A =  np.array([-1, -1, 0])
b = -0.5

而不是我原来的,然后优化工作。这表明我不能做一个约束矩阵(就像我在Matlab中使用fmincon),除非有人可以另外确认。但是,我可以继续添加约束,以便它可以工作。约束必须是字典元组的形式,所以我把它放在一起

d = []
for i in range(0, b.size):
    d.append({'type': 'ineq', 'fun' : lambda x: np.dot(A[i, ], x) - b[i], 'jac' : lambda x: A[i, ]})
d_ = tuple(d)
c_tuple = c_,
d_ = d_ + c_tuple

似乎工作正常。