X-Git-Url: http://gitweb.michael.orlitzky.com/?a=blobdiff_plain;f=mjo%2Fcone%2Fcone.py;h=d33a1c5bc317bf7a627f964ca11b78c45045b08a;hb=fdc03da648dd989527ff4c12ccce04c990869e3b;hp=84c3adb57d8396735460f5aefe5d669416e18ff1;hpb=2c784c920c6fcd0e15d1061b8ef04f67d9ade0c7;p=sage.d.git diff --git a/mjo/cone/cone.py b/mjo/cone/cone.py index 84c3adb..d33a1c5 100644 --- a/mjo/cone/cone.py +++ b/mjo/cone/cone.py @@ -83,7 +83,7 @@ def motzkin_decomposition(K): REFERENCES: .. [Stoer-Witzgall] J. Stoer and C. Witzgall. Convexity and - Optimization in Finite Dimensions I. Springer-Verlag, New + Optimization in Finite Dimensions I. Springer-Verlag, New York, 1970. EXAMPLES: @@ -152,18 +152,19 @@ def motzkin_decomposition(K): sage: S.is_equivalent(expected_S) True """ - linspace_gens = [ copy(b) for b in K.linear_subspace().basis() ] - linspace_gens += [ -b for b in linspace_gens ] + # The lines() method only returns one generator per line. For a true + # line, we also need a generator pointing in the opposite direction. + S_gens = [ direction*gen for direction in [1,-1] for gen in K.lines() ] + S = Cone(S_gens, K.lattice()) - S = Cone(linspace_gens, K.lattice()) - - # Since ``S`` is a subspace, its dual is its orthogonal complement - # (albeit in the wrong lattice). + # Since ``S`` is a subspace, the rays of its dual generate its + # orthogonal complement. S_perp = Cone(S.dual(), K.lattice()) P = K.intersection(S_perp) return (P,S) + def positive_operator_gens(K): r""" Compute generators of the cone of positive operators on this cone. @@ -216,12 +217,46 @@ def positive_operator_gens(K): TESTS: - A positive operator on a cone should send its generators into the cone:: + Each positive operator generator should send the generators of the + cone into the cone:: + + sage: set_random_seed() + sage: K = random_cone(max_ambient_dim=5) + sage: pi_of_K = positive_operator_gens(K) + sage: all([ K.contains(P*x) for P in pi_of_K for x in K ]) + True + + Each positive operator generator should send a random element of the + cone into the cone:: + + sage: set_random_seed() + sage: K = random_cone(max_ambient_dim=5) + sage: pi_of_K = positive_operator_gens(K) + sage: all([ K.contains(P*K.random_element()) for P in pi_of_K ]) + True + + A random element of the positive operator cone should send the + generators of the cone into the cone:: sage: set_random_seed() sage: K = random_cone(max_ambient_dim=5) sage: pi_of_K = positive_operator_gens(K) - sage: all([K.contains(p*x) for p in pi_of_K for x in K.rays()]) + sage: L = ToricLattice(K.lattice_dim()**2) + sage: pi_cone = Cone([ g.list() for g in pi_of_K ], lattice=L) + sage: P = matrix(K.lattice_dim(), pi_cone.random_element().list()) + sage: all([ K.contains(P*x) for x in K ]) + True + + A random element of the positive operator cone should send a random + element of the cone into the cone:: + + sage: set_random_seed() + sage: K = random_cone(max_ambient_dim=5) + sage: pi_of_K = positive_operator_gens(K) + sage: L = ToricLattice(K.lattice_dim()**2) + sage: pi_cone = Cone([ g.list() for g in pi_of_K ], lattice=L) + sage: P = matrix(K.lattice_dim(), pi_cone.random_element().list()) + sage: K.contains(P*K.random_element()) True The dimension of the cone of positive operators is given by the