Skip to content

Commit 15e74db

Browse files
unknownhinayuki64
unknown
authored andcommitted
Add method option in robustlq.py
1 parent ba191fb commit 15e74db

File tree

2 files changed

+50
-43
lines changed

2 files changed

+50
-43
lines changed

quantecon/robustlq.py

+20-7
Original file line numberDiff line numberDiff line change
@@ -152,7 +152,7 @@ def b_operator(self, P):
152152

153153
return F, new_P
154154

155-
def robust_rule(self):
155+
def robust_rule(self, method='doubling'):
156156
"""
157157
This method solves the robust control problem by tricking it
158158
into a stacked LQ problem, as described in chapter 2 of Hansen-
@@ -165,6 +165,12 @@ def robust_rule(self):
165165
166166
And the value function is :math:`-x'Px`
167167
168+
Parameters
169+
----------
170+
method : str, optional(default='doubling')
171+
Solution method used in solving the associated Riccati
172+
equation, str in {'doubling', 'qz'}.
173+
168174
Returns
169175
-------
170176
F : array_like(float, ndim=2)
@@ -189,7 +195,7 @@ def robust_rule(self):
189195
lq = LQ(-beta*I*theta, R, A, C, beta=beta)
190196

191197
# == Solve and convert back to robust problem == #
192-
P, f, d = lq.stationary_values()
198+
P, f, d = lq.stationary_values(method=method)
193199
F = np.zeros((self.k, self.n))
194200
K = -f[:k, :]
195201

@@ -199,7 +205,7 @@ def robust_rule(self):
199205
lq = LQ(Qa, R, A, Ba, beta=beta)
200206

201207
# == Solve and convert back to robust problem == #
202-
P, f, d = lq.stationary_values()
208+
P, f, d = lq.stationary_values(method=method)
203209
F = f[:k, :]
204210
K = -f[k:f.shape[0], :]
205211

@@ -254,14 +260,17 @@ def robust_rule_simple(self, P_init=None, max_iter=80, tol=1e-8):
254260

255261
return F, K, P
256262

257-
def F_to_K(self, F):
263+
def F_to_K(self, F, method='doubling'):
258264
"""
259265
Compute agent 2's best cost-minimizing response K, given F.
260266
261267
Parameters
262268
----------
263269
F : array_like(float, ndim=2)
264270
A k x n array
271+
method : str, optional(default='doubling')
272+
Solution method used in solving the associated Riccati
273+
equation, str in {'doubling', 'qz'}.
265274
266275
Returns
267276
-------
@@ -276,18 +285,21 @@ def F_to_K(self, F):
276285
A2 = self.A - dot(self.B, F)
277286
B2 = self.C
278287
lq = LQ(Q2, R2, A2, B2, beta=self.beta)
279-
neg_P, neg_K, d = lq.stationary_values()
288+
neg_P, neg_K, d = lq.stationary_values(method=method)
280289

281290
return -neg_K, -neg_P
282291

283-
def K_to_F(self, K):
292+
def K_to_F(self, K, method='doubling'):
284293
"""
285294
Compute agent 1's best value-maximizing response F, given K.
286295
287296
Parameters
288297
----------
289298
K : array_like(float, ndim=2)
290299
A j x n array
300+
method : str, optional(default='doubling')
301+
Solution method used in solving the associated Riccati
302+
equation, str in {'doubling', 'qz'}.
291303
292304
Returns
293305
-------
@@ -302,7 +314,7 @@ def K_to_F(self, K):
302314
Q1 = self.Q
303315
R1 = self.R - self.beta * self.theta * dot(K.T, K)
304316
lq = LQ(Q1, R1, A1, B1, beta=self.beta)
305-
P, F, d = lq.stationary_values()
317+
P, F, d = lq.stationary_values(method=method)
306318

307319
return F, P
308320

@@ -388,3 +400,4 @@ def evaluate_F(self, F):
388400
o_F = (ho + beta * tr) / (1 - beta)
389401

390402
return K_F, P_F, d_F, O_F, o_F
403+

quantecon/tests/test_robustlq.py

+30-36
Original file line numberDiff line numberDiff line change
@@ -51,9 +51,7 @@ def setUp(self):
5151
self.rblq_test = RBLQ(Q, R, A, B, C, beta, theta)
5252
self.rblq_test_pf = RBLQ(Q_pf, R, A, B_pf, C, beta, theta)
5353
self.lq_test = LQ(Q, R, A, B, C, beta)
54-
55-
self.Fr, self.Kr, self.Pr = self.rblq_test.robust_rule()
56-
self.Fr_pf, self.Kr_pf, self.Pr_pf = self.rblq_test_pf.robust_rule()
54+
self.methods = ['doubling', 'qz']
5755

5856
def tearDown(self):
5957
del self.rblq_test
@@ -65,52 +63,48 @@ def test_pure_forecasting(self):
6563
def test_robust_rule_vs_simple(self):
6664
rblq = self.rblq_test
6765
rblq_pf = self.rblq_test_pf
68-
Fr, Kr, Pr = self.Fr, self.Kr, self.Pr
69-
Fr_pf, Kr_pf, Pr_pf = self.Fr_pf, self.Kr_pf, self.Pr_pf
7066

71-
Fs, Ks, Ps = rblq.robust_rule_simple(P_init=Pr, tol=1e-12)
72-
Fs_pf, Ks_pf, Ps_pf = rblq_pf.robust_rule_simple(P_init=Pr_pf, tol=1e-12)
67+
for method in self.methods:
68+
Fr, Kr, Pr = self.rblq_test.robust_rule(method=method)
69+
Fr_pf, Kr_pf, Pr_pf = self.rblq_test_pf.robust_rule(method=method)
70+
71+
Fs, Ks, Ps = rblq.robust_rule_simple(P_init=Pr, tol=1e-12)
72+
Fs_pf, Ks_pf, Ps_pf = rblq_pf.robust_rule_simple(P_init=Pr_pf, tol=1e-12)
7373

74-
assert_allclose(Fr, Fs, rtol=1e-4)
75-
assert_allclose(Kr, Ks, rtol=1e-4)
76-
assert_allclose(Pr, Ps, rtol=1e-4)
74+
assert_allclose(Fr, Fs, rtol=1e-4)
75+
assert_allclose(Kr, Ks, rtol=1e-4)
76+
assert_allclose(Pr, Ps, rtol=1e-4)
7777

78-
atol = 1e-10
79-
assert_allclose(Fr_pf, Fs_pf, rtol=1e-4)
80-
assert_allclose(Kr_pf, Ks_pf, rtol=1e-4, atol=atol)
81-
assert_allclose(Pr_pf, Ps_pf, rtol=1e-4, atol=atol)
78+
atol = 1e-10
79+
assert_allclose(Fr_pf, Fs_pf, rtol=1e-4)
80+
assert_allclose(Kr_pf, Ks_pf, rtol=1e-4, atol=atol)
81+
assert_allclose(Pr_pf, Ps_pf, rtol=1e-4, atol=atol)
8282

8383

8484
def test_f2k_and_k2f(self):
8585
rblq = self.rblq_test
86-
Fr, Kr, Pr = self.Fr, self.Kr, self.Pr
87-
88-
K_f2k, P_f2k = rblq.F_to_K(Fr)
89-
F_k2f, P_k2f = rblq.K_to_F(Kr)
9086

91-
assert_allclose(K_f2k, Kr, rtol=1e-4)
92-
assert_allclose(F_k2f, Fr, rtol=1e-4)
93-
assert_allclose(P_f2k, P_k2f, rtol=1e-4)
87+
for method in self.methods:
88+
Fr, Kr, Pr = self.rblq_test.robust_rule(method=method)
89+
K_f2k, P_f2k = rblq.F_to_K(Fr, method=method)
90+
F_k2f, P_k2f = rblq.K_to_F(Kr, method=method)
91+
assert_allclose(K_f2k, Kr, rtol=1e-4)
92+
assert_allclose(F_k2f, Fr, rtol=1e-4)
93+
assert_allclose(P_f2k, P_k2f, rtol=1e-4)
9494

9595
def test_evaluate_F(self):
9696
rblq = self.rblq_test
97-
Fr, Kr, Pr = self.Fr, self.Kr, self.Pr
98-
99-
Kf, Pf, df, Of, of = rblq.evaluate_F(Fr)
100-
101-
# In the future if we wanted, we could check more things, but I
102-
# think the other pieces are basically just plugging these into
103-
# equations so if these hold then the others should be correct
104-
# as well.
105-
assert_allclose(Pf, Pr)
106-
assert_allclose(Kf, Kr)
107-
108-
109-
110-
111-
97+
for method in self.methods:
98+
Fr, Kr, Pr = self.rblq_test.robust_rule(method=method)
11299

100+
Kf, Pf, df, Of, of = rblq.evaluate_F(Fr)
113101

102+
# In the future if we wanted, we could check more things, but I
103+
# think the other pieces are basically just plugging these into
104+
# equations so if these hold then the others should be correct
105+
# as well.
106+
assert_allclose(Pf, Pr)
107+
assert_allclose(Kf, Kr)
114108

115109
if __name__ == '__main__':
116110
suite = unittest.TestLoader().loadTestsFromTestCase(TestRBLQControl)

0 commit comments

Comments
 (0)