Base class for polyhedra: Methods regarding the combinatorics of a polyhedron¶
Excluding methods relying on sage.graphs
.
- class sage.geometry.polyhedron.base3.Polyhedron_base3(parent, Vrep, Hrep, Vrep_minimal=None, Hrep_minimal=None, pref_rep=None, mutable=False, **kwds)[source]¶
Bases:
Polyhedron_base2
Methods related to the combinatorics of a polyhedron.
See
sage.geometry.polyhedron.base.Polyhedron_base
.- a_maximal_chain()[source]¶
Return a maximal chain of the face lattice in increasing order.
EXAMPLES:
sage: P = polytopes.cube() sage: P.a_maximal_chain() [A -1-dimensional face of a Polyhedron in ZZ^3, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices, A 3-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 8 vertices] sage: P = polytopes.cube() sage: chain = P.a_maximal_chain(); chain [A -1-dimensional face of a Polyhedron in ZZ^3, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices, A 3-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 8 vertices] sage: [face.ambient_V_indices() for face in chain] [(), (5,), (0, 5), (0, 3, 4, 5), (0, 1, 2, 3, 4, 5, 6, 7)]
>>> from sage.all import * >>> P = polytopes.cube() >>> P.a_maximal_chain() [A -1-dimensional face of a Polyhedron in ZZ^3, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices, A 3-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 8 vertices] >>> P = polytopes.cube() >>> chain = P.a_maximal_chain(); chain [A -1-dimensional face of a Polyhedron in ZZ^3, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices, A 3-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 8 vertices] >>> [face.ambient_V_indices() for face in chain] [(), (5,), (0, 5), (0, 3, 4, 5), (0, 1, 2, 3, 4, 5, 6, 7)]
P = polytopes.cube() P.a_maximal_chain() P = polytopes.cube() chain = P.a_maximal_chain(); chain [face.ambient_V_indices() for face in chain]
- adjacency_matrix(algorithm=None)[source]¶
Return the binary matrix of vertex adjacencies.
INPUT:
algorithm
– string (optional); specify whether the face generator starts with facets or vertices:'primal'
– start with the facets'dual'
– start with the verticesNone
– choose automatically
EXAMPLES:
sage: polytopes.simplex(4).vertex_adjacency_matrix() [0 1 1 1 1] [1 0 1 1 1] [1 1 0 1 1] [1 1 1 0 1] [1 1 1 1 0]
>>> from sage.all import * >>> polytopes.simplex(Integer(4)).vertex_adjacency_matrix() [0 1 1 1 1] [1 0 1 1 1] [1 1 0 1 1] [1 1 1 0 1] [1 1 1 1 0]
polytopes.simplex(4).vertex_adjacency_matrix()
The rows and columns of the vertex adjacency matrix correspond to the
Vrepresentation()
objects: vertices, rays, and lines. The \((i,j)\) matrix entry equals \(1\) if the \(i\)-th and \(j\)-th V-representation object are adjacent.Two vertices are adjacent if they are the endpoints of an edge, that is, a one-dimensional face. For unbounded polyhedra this clearly needs to be generalized and we define two V-representation objects (see
sage.geometry.polyhedron.constructor
) to be adjacent if they together generate a one-face. There are three possible combinations:Two vertices can bound a finite-length edge.
A vertex and a ray can generate a half-infinite edge starting at the vertex and with the direction given by the ray.
A vertex and a line can generate an infinite edge. The position of the vertex on the line is arbitrary in this case, only its transverse position matters. The direction of the edge is given by the line generator.
For example, take the half-plane:
sage: half_plane = Polyhedron(ieqs=[(0,1,0)]) sage: half_plane.Hrepresentation() (An inequality (1, 0) x + 0 >= 0,)
>>> from sage.all import * >>> half_plane = Polyhedron(ieqs=[(Integer(0),Integer(1),Integer(0))]) >>> half_plane.Hrepresentation() (An inequality (1, 0) x + 0 >= 0,)
half_plane = Polyhedron(ieqs=[(0,1,0)]) half_plane.Hrepresentation()
Its (non-unique) V-representation consists of a vertex, a ray, and a line. The only edge is spanned by the vertex and the line generator, so they are adjacent:
sage: half_plane.Vrepresentation() (A line in the direction (0, 1), A ray in the direction (1, 0), A vertex at (0, 0)) sage: half_plane.vertex_adjacency_matrix() [0 0 1] [0 0 0] [1 0 0]
>>> from sage.all import * >>> half_plane.Vrepresentation() (A line in the direction (0, 1), A ray in the direction (1, 0), A vertex at (0, 0)) >>> half_plane.vertex_adjacency_matrix() [0 0 1] [0 0 0] [1 0 0]
half_plane.Vrepresentation() half_plane.vertex_adjacency_matrix()
In one dimension higher, that is for a half-space in 3 dimensions, there is no one-dimensional face. Hence nothing is adjacent:
sage: Polyhedron(ieqs=[(0,1,0,0)]).vertex_adjacency_matrix() [0 0 0 0] [0 0 0 0] [0 0 0 0] [0 0 0 0]
>>> from sage.all import * >>> Polyhedron(ieqs=[(Integer(0),Integer(1),Integer(0),Integer(0))]).vertex_adjacency_matrix() [0 0 0 0] [0 0 0 0] [0 0 0 0] [0 0 0 0]
Polyhedron(ieqs=[(0,1,0,0)]).vertex_adjacency_matrix()
EXAMPLES:
In a bounded polygon, every vertex has precisely two adjacent ones:
sage: P = Polyhedron(vertices=[(0, 1), (1, 0), (3, 0), (4, 1)]) sage: for v in P.Vrep_generator(): ....: print("{} {}".format(P.adjacency_matrix().row(v.index()), v)) (0, 1, 0, 1) A vertex at (0, 1) (1, 0, 1, 0) A vertex at (1, 0) (0, 1, 0, 1) A vertex at (3, 0) (1, 0, 1, 0) A vertex at (4, 1)
>>> from sage.all import * >>> P = Polyhedron(vertices=[(Integer(0), Integer(1)), (Integer(1), Integer(0)), (Integer(3), Integer(0)), (Integer(4), Integer(1))]) >>> for v in P.Vrep_generator(): ... print("{} {}".format(P.adjacency_matrix().row(v.index()), v)) (0, 1, 0, 1) A vertex at (0, 1) (1, 0, 1, 0) A vertex at (1, 0) (0, 1, 0, 1) A vertex at (3, 0) (1, 0, 1, 0) A vertex at (4, 1)
P = Polyhedron(vertices=[(0, 1), (1, 0), (3, 0), (4, 1)]) for v in P.Vrep_generator(): print("{} {}".format(P.adjacency_matrix().row(v.index()), v))
If the V-representation of the polygon contains vertices and one ray, then each V-representation object is adjacent to two V-representation objects:
sage: P = Polyhedron(vertices=[(0, 1), (1, 0), (3, 0), (4, 1)], ....: rays=[(0,1)]) sage: for v in P.Vrep_generator(): ....: print("{} {}".format(P.adjacency_matrix().row(v.index()), v)) (0, 1, 0, 0, 1) A ray in the direction (0, 1) (1, 0, 1, 0, 0) A vertex at (0, 1) (0, 1, 0, 1, 0) A vertex at (1, 0) (0, 0, 1, 0, 1) A vertex at (3, 0) (1, 0, 0, 1, 0) A vertex at (4, 1)
>>> from sage.all import * >>> P = Polyhedron(vertices=[(Integer(0), Integer(1)), (Integer(1), Integer(0)), (Integer(3), Integer(0)), (Integer(4), Integer(1))], ... rays=[(Integer(0),Integer(1))]) >>> for v in P.Vrep_generator(): ... print("{} {}".format(P.adjacency_matrix().row(v.index()), v)) (0, 1, 0, 0, 1) A ray in the direction (0, 1) (1, 0, 1, 0, 0) A vertex at (0, 1) (0, 1, 0, 1, 0) A vertex at (1, 0) (0, 0, 1, 0, 1) A vertex at (3, 0) (1, 0, 0, 1, 0) A vertex at (4, 1)
P = Polyhedron(vertices=[(0, 1), (1, 0), (3, 0), (4, 1)], rays=[(0,1)]) for v in P.Vrep_generator(): print("{} {}".format(P.adjacency_matrix().row(v.index()), v))
If the V-representation of the polygon contains vertices and two distinct rays, then each vertex is adjacent to two V-representation objects (which can now be vertices or rays). The two rays are not adjacent to each other:
sage: P = Polyhedron(vertices=[(0, 1), (1, 0), (3, 0), (4, 1)], ....: rays=[(0,1), (1,1)]) sage: for v in P.Vrep_generator(): ....: print("{} {}".format(P.adjacency_matrix().row(v.index()), v)) (0, 1, 0, 0, 0) A ray in the direction (0, 1) (1, 0, 1, 0, 0) A vertex at (0, 1) (0, 1, 0, 0, 1) A vertex at (1, 0) (0, 0, 0, 0, 1) A ray in the direction (1, 1) (0, 0, 1, 1, 0) A vertex at (3, 0)
>>> from sage.all import * >>> P = Polyhedron(vertices=[(Integer(0), Integer(1)), (Integer(1), Integer(0)), (Integer(3), Integer(0)), (Integer(4), Integer(1))], ... rays=[(Integer(0),Integer(1)), (Integer(1),Integer(1))]) >>> for v in P.Vrep_generator(): ... print("{} {}".format(P.adjacency_matrix().row(v.index()), v)) (0, 1, 0, 0, 0) A ray in the direction (0, 1) (1, 0, 1, 0, 0) A vertex at (0, 1) (0, 1, 0, 0, 1) A vertex at (1, 0) (0, 0, 0, 0, 1) A ray in the direction (1, 1) (0, 0, 1, 1, 0) A vertex at (3, 0)
P = Polyhedron(vertices=[(0, 1), (1, 0), (3, 0), (4, 1)], rays=[(0,1), (1,1)]) for v in P.Vrep_generator(): print("{} {}".format(P.adjacency_matrix().row(v.index()), v))
The vertex adjacency matrix has base ring integers. This way one can express various counting questions:
sage: P = polytopes.cube() sage: Q = P.stack(P.faces(2)[0]) sage: M = Q.vertex_adjacency_matrix() sage: sum(M) (4, 4, 3, 3, 4, 4, 4, 3, 3) sage: G = Q.vertex_graph() # needs sage.graphs sage: G.degree() # needs sage.graphs [4, 4, 3, 3, 4, 4, 4, 3, 3]
>>> from sage.all import * >>> P = polytopes.cube() >>> Q = P.stack(P.faces(Integer(2))[Integer(0)]) >>> M = Q.vertex_adjacency_matrix() >>> sum(M) (4, 4, 3, 3, 4, 4, 4, 3, 3) >>> G = Q.vertex_graph() # needs sage.graphs >>> G.degree() # needs sage.graphs [4, 4, 3, 3, 4, 4, 4, 3, 3]
P = polytopes.cube() Q = P.stack(P.faces(2)[0]) M = Q.vertex_adjacency_matrix() sum(M) G = Q.vertex_graph() # needs sage.graphs G.degree() # needs sage.graphs
- bounded_edges()[source]¶
Return the bounded edges (excluding rays and lines).
OUTPUT: a generator for pairs of vertices, one pair per edge
EXAMPLES:
sage: p = Polyhedron(vertices=[[1,0],[0,1]], rays=[[1,0],[0,1]]) sage: [ e for e in p.bounded_edges() ] [(A vertex at (0, 1), A vertex at (1, 0))] sage: for e in p.bounded_edges(): print(e) (A vertex at (0, 1), A vertex at (1, 0))
>>> from sage.all import * >>> p = Polyhedron(vertices=[[Integer(1),Integer(0)],[Integer(0),Integer(1)]], rays=[[Integer(1),Integer(0)],[Integer(0),Integer(1)]]) >>> [ e for e in p.bounded_edges() ] [(A vertex at (0, 1), A vertex at (1, 0))] >>> for e in p.bounded_edges(): print(e) (A vertex at (0, 1), A vertex at (1, 0))
p = Polyhedron(vertices=[[1,0],[0,1]], rays=[[1,0],[0,1]]) [ e for e in p.bounded_edges() ] for e in p.bounded_edges(): print(e)
- combinatorial_polyhedron()[source]¶
Return the combinatorial type of
self
.See
sage.geometry.polyhedron.combinatorial_polyhedron.base.CombinatorialPolyhedron
.EXAMPLES:
sage: polytopes.cube().combinatorial_polyhedron() A 3-dimensional combinatorial polyhedron with 6 facets sage: polytopes.cyclic_polytope(4,10).combinatorial_polyhedron() A 4-dimensional combinatorial polyhedron with 35 facets sage: Polyhedron(rays=[[0,1], [1,0]]).combinatorial_polyhedron() A 2-dimensional combinatorial polyhedron with 2 facets
>>> from sage.all import * >>> polytopes.cube().combinatorial_polyhedron() A 3-dimensional combinatorial polyhedron with 6 facets >>> polytopes.cyclic_polytope(Integer(4),Integer(10)).combinatorial_polyhedron() A 4-dimensional combinatorial polyhedron with 35 facets >>> Polyhedron(rays=[[Integer(0),Integer(1)], [Integer(1),Integer(0)]]).combinatorial_polyhedron() A 2-dimensional combinatorial polyhedron with 2 facets
polytopes.cube().combinatorial_polyhedron() polytopes.cyclic_polytope(4,10).combinatorial_polyhedron() Polyhedron(rays=[[0,1], [1,0]]).combinatorial_polyhedron()
- f_vector(num_threads=None, parallelization_depth=None, algorithm=None)[source]¶
Return the f-vector.
INPUT:
num_threads
– integer (optional); specify the number of threads; otherwise determined byncpus()
parallelization_depth
– integer (optional); specify how deep in the lattice the parallelization is donealgorithm
– string (optional); specify whether the face generator starts with facets or vertices:'primal'
– start with the facets'dual'
– start with the verticesNone
– choose automatically
OUTPUT:
Return a vector whose \(i\)-th entry is the number of \(i-2\)-dimensional faces of the polytope.
Note
The
vertices
as given byvertices()
do not need to correspond to \(0\)-dimensional faces. If a polyhedron contains \(k\) lines they correspond to \(k\)-dimensional faces. See example below.EXAMPLES:
sage: p = Polyhedron(vertices=[[1, 2, 3], [1, 3, 2], ....: [2, 1, 3], [2, 3, 1], [3, 1, 2], [3, 2, 1], [0, 0, 0]]) sage: p.f_vector() (1, 7, 12, 7, 1) sage: polytopes.cyclic_polytope(4,10).f_vector() (1, 10, 45, 70, 35, 1) sage: polytopes.hypercube(5).f_vector() (1, 32, 80, 80, 40, 10, 1)
>>> from sage.all import * >>> p = Polyhedron(vertices=[[Integer(1), Integer(2), Integer(3)], [Integer(1), Integer(3), Integer(2)], ... [Integer(2), Integer(1), Integer(3)], [Integer(2), Integer(3), Integer(1)], [Integer(3), Integer(1), Integer(2)], [Integer(3), Integer(2), Integer(1)], [Integer(0), Integer(0), Integer(0)]]) >>> p.f_vector() (1, 7, 12, 7, 1) >>> polytopes.cyclic_polytope(Integer(4),Integer(10)).f_vector() (1, 10, 45, 70, 35, 1) >>> polytopes.hypercube(Integer(5)).f_vector() (1, 32, 80, 80, 40, 10, 1)
p = Polyhedron(vertices=[[1, 2, 3], [1, 3, 2], [2, 1, 3], [2, 3, 1], [3, 1, 2], [3, 2, 1], [0, 0, 0]]) p.f_vector() polytopes.cyclic_polytope(4,10).f_vector() polytopes.hypercube(5).f_vector()
Polyhedra with lines do not have \(0\)-faces:
sage: Polyhedron(ieqs=[[1,-1,0,0],[1,1,0,0]]).f_vector() (1, 0, 0, 2, 1)
>>> from sage.all import * >>> Polyhedron(ieqs=[[Integer(1),-Integer(1),Integer(0),Integer(0)],[Integer(1),Integer(1),Integer(0),Integer(0)]]).f_vector() (1, 0, 0, 2, 1)
Polyhedron(ieqs=[[1,-1,0,0],[1,1,0,0]]).f_vector()
However, the method
Polyhedron_base.vertices()
returns two points that belong to theVrepresentation
:sage: P = Polyhedron(ieqs=[[1,-1,0],[1,1,0]]) sage: P.vertices() (A vertex at (1, 0), A vertex at (-1, 0)) sage: P.f_vector() (1, 0, 2, 1)
>>> from sage.all import * >>> P = Polyhedron(ieqs=[[Integer(1),-Integer(1),Integer(0)],[Integer(1),Integer(1),Integer(0)]]) >>> P.vertices() (A vertex at (1, 0), A vertex at (-1, 0)) >>> P.f_vector() (1, 0, 2, 1)
P = Polyhedron(ieqs=[[1,-1,0],[1,1,0]]) P.vertices() P.f_vector()
- face_generator(face_dimension=None, algorithm=None)[source]¶
Return an iterator over the faces of given dimension.
If dimension is not specified return an iterator over all faces.
INPUT:
face_dimension
– integer (default:None
); yield only faces of this dimension if specifiedalgorithm
– string (optional); specify whether to start with facets or vertices:'primal'
– start with the facets'dual'
– start with the verticesNone
– choose automatically
OUTPUT:
A
FaceIterator_geom
. This class iterates over faces asPolyhedronFace
. Seeface
for details. The order is random but fixed.EXAMPLES:
sage: P = polytopes.cube() sage: it = P.face_generator() sage: it Iterator over the faces of a 3-dimensional polyhedron in ZZ^3 sage: list(it) [A 3-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 8 vertices, A -1-dimensional face of a Polyhedron in ZZ^3, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices] sage: P = polytopes.hypercube(4) sage: list(P.face_generator(2))[:4] [A 2-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 4 vertices, A 2-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 4 vertices, A 2-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 4 vertices, A 2-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 4 vertices]
>>> from sage.all import * >>> P = polytopes.cube() >>> it = P.face_generator() >>> it Iterator over the faces of a 3-dimensional polyhedron in ZZ^3 >>> list(it) [A 3-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 8 vertices, A -1-dimensional face of a Polyhedron in ZZ^3, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices] >>> P = polytopes.hypercube(Integer(4)) >>> list(P.face_generator(Integer(2)))[:Integer(4)] [A 2-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 4 vertices, A 2-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 4 vertices, A 2-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 4 vertices, A 2-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 4 vertices]
P = polytopes.cube() it = P.face_generator() it list(it) P = polytopes.hypercube(4) list(P.face_generator(2))[:4]
If a polytope has more facets than vertices, the dual mode is chosen:
sage: P = polytopes.cross_polytope(3) sage: list(P.face_generator()) [A 3-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 6 vertices, A -1-dimensional face of a Polyhedron in ZZ^3, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 3 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 3 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 3 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 3 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 3 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 3 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 3 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 3 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices]
>>> from sage.all import * >>> P = polytopes.cross_polytope(Integer(3)) >>> list(P.face_generator()) [A 3-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 6 vertices, A -1-dimensional face of a Polyhedron in ZZ^3, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 3 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 3 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 3 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 3 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 3 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 3 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 3 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices, A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 3 vertices, A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices]
P = polytopes.cross_polytope(3) list(P.face_generator())
The face iterator can also be slightly modified. In non-dual mode we can skip subfaces of the current (proper) face:
sage: P = polytopes.cube() sage: it = P.face_generator(algorithm='primal') sage: _ = next(it), next(it) sage: face = next(it) sage: face.ambient_H_indices() (5,) sage: it.ignore_subfaces() sage: face = next(it) sage: face.ambient_H_indices() (4,) sage: it.ignore_subfaces() sage: [face.ambient_H_indices() for face in it] [(3,), (2,), (1,), (0,), (2, 3), (1, 3), (1, 2, 3), (1, 2), (0, 2), (0, 1, 2), (0, 1)]
>>> from sage.all import * >>> P = polytopes.cube() >>> it = P.face_generator(algorithm='primal') >>> _ = next(it), next(it) >>> face = next(it) >>> face.ambient_H_indices() (5,) >>> it.ignore_subfaces() >>> face = next(it) >>> face.ambient_H_indices() (4,) >>> it.ignore_subfaces() >>> [face.ambient_H_indices() for face in it] [(3,), (2,), (1,), (0,), (2, 3), (1, 3), (1, 2, 3), (1, 2), (0, 2), (0, 1, 2), (0, 1)]
P = polytopes.cube() it = P.face_generator(algorithm='primal') _ = next(it), next(it) face = next(it) face.ambient_H_indices() it.ignore_subfaces() face = next(it) face.ambient_H_indices() it.ignore_subfaces() [face.ambient_H_indices() for face in it]
In dual mode we can skip supfaces of the current (proper) face:
sage: P = polytopes.cube() sage: it = P.face_generator(algorithm='dual') sage: _ = next(it), next(it) sage: face = next(it) sage: face.ambient_V_indices() (7,) sage: it.ignore_supfaces() sage: next(it) A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex sage: face = next(it) sage: face.ambient_V_indices() (5,) sage: it.ignore_supfaces() sage: [face.ambient_V_indices() for face in it] [(4,), (3,), (2,), (1,), (0,), (1, 6), (3, 4), (2, 3), (0, 3), (0, 1, 2, 3), (1, 2), (0, 1)]
>>> from sage.all import * >>> P = polytopes.cube() >>> it = P.face_generator(algorithm='dual') >>> _ = next(it), next(it) >>> face = next(it) >>> face.ambient_V_indices() (7,) >>> it.ignore_supfaces() >>> next(it) A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex >>> face = next(it) >>> face.ambient_V_indices() (5,) >>> it.ignore_supfaces() >>> [face.ambient_V_indices() for face in it] [(4,), (3,), (2,), (1,), (0,), (1, 6), (3, 4), (2, 3), (0, 3), (0, 1, 2, 3), (1, 2), (0, 1)]
P = polytopes.cube() it = P.face_generator(algorithm='dual') _ = next(it), next(it) face = next(it) face.ambient_V_indices() it.ignore_supfaces() next(it) face = next(it) face.ambient_V_indices() it.ignore_supfaces() [face.ambient_V_indices() for face in it]
In non-dual mode, we cannot skip supfaces:
sage: it = P.face_generator(algorithm='primal') sage: _ = next(it), next(it) sage: next(it) A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices sage: it.ignore_supfaces() Traceback (most recent call last): ... ValueError: only possible when in dual mode
>>> from sage.all import * >>> it = P.face_generator(algorithm='primal') >>> _ = next(it), next(it) >>> next(it) A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices >>> it.ignore_supfaces() Traceback (most recent call last): ... ValueError: only possible when in dual mode
it = P.face_generator(algorithm='primal') _ = next(it), next(it) next(it) it.ignore_supfaces()
In dual mode, we cannot skip subfaces:
sage: it = P.face_generator(algorithm='dual') sage: _ = next(it), next(it) sage: next(it) A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex sage: it.ignore_subfaces() Traceback (most recent call last): ... ValueError: only possible when not in dual mode
>>> from sage.all import * >>> it = P.face_generator(algorithm='dual') >>> _ = next(it), next(it) >>> next(it) A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex >>> it.ignore_subfaces() Traceback (most recent call last): ... ValueError: only possible when not in dual mode
it = P.face_generator(algorithm='dual') _ = next(it), next(it) next(it) it.ignore_subfaces()
We can only skip sub-/supfaces of proper faces:
sage: it = P.face_generator(algorithm='primal') sage: next(it) A 3-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 8 vertices sage: it.ignore_subfaces() Traceback (most recent call last): ... ValueError: iterator not set to a face yet
>>> from sage.all import * >>> it = P.face_generator(algorithm='primal') >>> next(it) A 3-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 8 vertices >>> it.ignore_subfaces() Traceback (most recent call last): ... ValueError: iterator not set to a face yet
it = P.face_generator(algorithm='primal') next(it) it.ignore_subfaces()
See also
ALGORITHM:
See
FaceIterator
.
- faces(face_dimension)[source]¶
Return the faces of given dimension.
INPUT:
face_dimension
– integer; the dimension of the faces whose representation will be returned
OUTPUT:
A tuple of
PolyhedronFace
. See modulesage.geometry.polyhedron.face
for details. The order is random but fixed.See also
face_generator()
,facet()
.EXAMPLES:
Here we find the vertex and face indices of the eight three-dimensional facets of the four-dimensional hypercube:
sage: p = polytopes.hypercube(4) sage: list(f.ambient_V_indices() for f in p.faces(3)) [(0, 5, 6, 7, 8, 9, 14, 15), (1, 4, 5, 6, 10, 13, 14, 15), (1, 2, 6, 7, 8, 10, 11, 15), (8, 9, 10, 11, 12, 13, 14, 15), (0, 3, 4, 5, 9, 12, 13, 14), (0, 2, 3, 7, 8, 9, 11, 12), (1, 2, 3, 4, 10, 11, 12, 13), (0, 1, 2, 3, 4, 5, 6, 7)] sage: face = p.faces(3)[3] sage: face.ambient_Hrepresentation() (An inequality (1, 0, 0, 0) x + 1 >= 0,) sage: face.vertices() (A vertex at (-1, -1, 1, -1), A vertex at (-1, -1, 1, 1), A vertex at (-1, 1, -1, -1), A vertex at (-1, 1, 1, -1), A vertex at (-1, 1, 1, 1), A vertex at (-1, 1, -1, 1), A vertex at (-1, -1, -1, 1), A vertex at (-1, -1, -1, -1))
>>> from sage.all import * >>> p = polytopes.hypercube(Integer(4)) >>> list(f.ambient_V_indices() for f in p.faces(Integer(3))) [(0, 5, 6, 7, 8, 9, 14, 15), (1, 4, 5, 6, 10, 13, 14, 15), (1, 2, 6, 7, 8, 10, 11, 15), (8, 9, 10, 11, 12, 13, 14, 15), (0, 3, 4, 5, 9, 12, 13, 14), (0, 2, 3, 7, 8, 9, 11, 12), (1, 2, 3, 4, 10, 11, 12, 13), (0, 1, 2, 3, 4, 5, 6, 7)] >>> face = p.faces(Integer(3))[Integer(3)] >>> face.ambient_Hrepresentation() (An inequality (1, 0, 0, 0) x + 1 >= 0,) >>> face.vertices() (A vertex at (-1, -1, 1, -1), A vertex at (-1, -1, 1, 1), A vertex at (-1, 1, -1, -1), A vertex at (-1, 1, 1, -1), A vertex at (-1, 1, 1, 1), A vertex at (-1, 1, -1, 1), A vertex at (-1, -1, -1, 1), A vertex at (-1, -1, -1, -1))
p = polytopes.hypercube(4) list(f.ambient_V_indices() for f in p.faces(3)) face = p.faces(3)[3] face.ambient_Hrepresentation() face.vertices()
You can use the
index()
method to enumerate vertices and inequalities:sage: def get_idx(rep): return rep.index() sage: [get_idx(_) for _ in face.ambient_Hrepresentation()] [4] sage: [get_idx(_) for _ in face.ambient_Vrepresentation()] [8, 9, 10, 11, 12, 13, 14, 15] sage: [ ([get_idx(_) for _ in face.ambient_Vrepresentation()], ....: [get_idx(_) for _ in face.ambient_Hrepresentation()]) ....: for face in p.faces(3) ] [([0, 5, 6, 7, 8, 9, 14, 15], [7]), ([1, 4, 5, 6, 10, 13, 14, 15], [6]), ([1, 2, 6, 7, 8, 10, 11, 15], [5]), ([8, 9, 10, 11, 12, 13, 14, 15], [4]), ([0, 3, 4, 5, 9, 12, 13, 14], [3]), ([0, 2, 3, 7, 8, 9, 11, 12], [2]), ([1, 2, 3, 4, 10, 11, 12, 13], [1]), ([0, 1, 2, 3, 4, 5, 6, 7], [0])]
>>> from sage.all import * >>> def get_idx(rep): return rep.index() >>> [get_idx(_) for _ in face.ambient_Hrepresentation()] [4] >>> [get_idx(_) for _ in face.ambient_Vrepresentation()] [8, 9, 10, 11, 12, 13, 14, 15] >>> [ ([get_idx(_) for _ in face.ambient_Vrepresentation()], ... [get_idx(_) for _ in face.ambient_Hrepresentation()]) ... for face in p.faces(Integer(3)) ] [([0, 5, 6, 7, 8, 9, 14, 15], [7]), ([1, 4, 5, 6, 10, 13, 14, 15], [6]), ([1, 2, 6, 7, 8, 10, 11, 15], [5]), ([8, 9, 10, 11, 12, 13, 14, 15], [4]), ([0, 3, 4, 5, 9, 12, 13, 14], [3]), ([0, 2, 3, 7, 8, 9, 11, 12], [2]), ([1, 2, 3, 4, 10, 11, 12, 13], [1]), ([0, 1, 2, 3, 4, 5, 6, 7], [0])]
def get_idx(rep): return rep.index() [get_idx(_) for _ in face.ambient_Hrepresentation()] [get_idx(_) for _ in face.ambient_Vrepresentation()] [ ([get_idx(_) for _ in face.ambient_Vrepresentation()], [get_idx(_) for _ in face.ambient_Hrepresentation()]) for face in p.faces(3) ]
- facet_adjacency_matrix(algorithm=None)[source]¶
Return the adjacency matrix for the facets.
INPUT:
algorithm
– string (optional); specify whether the face generator starts with facets or vertices:'primal'
– start with the facets'dual'
– start with the verticesNone
– choose automatically
EXAMPLES:
sage: s4 = polytopes.simplex(4, project=True) sage: s4.facet_adjacency_matrix() [0 1 1 1 1] [1 0 1 1 1] [1 1 0 1 1] [1 1 1 0 1] [1 1 1 1 0] sage: p = Polyhedron(vertices=[(0,0),(1,0),(0,1)]) sage: p.facet_adjacency_matrix() [0 1 1] [1 0 1] [1 1 0]
>>> from sage.all import * >>> s4 = polytopes.simplex(Integer(4), project=True) >>> s4.facet_adjacency_matrix() [0 1 1 1 1] [1 0 1 1 1] [1 1 0 1 1] [1 1 1 0 1] [1 1 1 1 0] >>> p = Polyhedron(vertices=[(Integer(0),Integer(0)),(Integer(1),Integer(0)),(Integer(0),Integer(1))]) >>> p.facet_adjacency_matrix() [0 1 1] [1 0 1] [1 1 0]
s4 = polytopes.simplex(4, project=True) s4.facet_adjacency_matrix() p = Polyhedron(vertices=[(0,0),(1,0),(0,1)]) p.facet_adjacency_matrix()
The facet adjacency matrix has base ring integers. This way one can express various counting questions:
sage: P = polytopes.cube() sage: Q = P.stack(P.faces(2)[0]) sage: M = Q.facet_adjacency_matrix() sage: sum(M) (4, 4, 4, 4, 3, 3, 3, 3, 4)
>>> from sage.all import * >>> P = polytopes.cube() >>> Q = P.stack(P.faces(Integer(2))[Integer(0)]) >>> M = Q.facet_adjacency_matrix() >>> sum(M) (4, 4, 4, 4, 3, 3, 3, 3, 4)
P = polytopes.cube() Q = P.stack(P.faces(2)[0]) M = Q.facet_adjacency_matrix() sum(M)
- facets()[source]¶
Return the facets of the polyhedron.
Facets are the maximal nontrivial faces of polyhedra. The empty face and the polyhedron itself are trivial.
A facet of a \(d\)-dimensional polyhedron is a face of dimension \(d-1\). For \(d \neq 0\) the converse is true as well.
OUTPUT:
A tuple of
PolyhedronFace
. Seeface
for details. The order is random but fixed.See also
EXAMPLES:
Here we find the eight three-dimensional facets of the four-dimensional hypercube:
sage: p = polytopes.hypercube(4) sage: p.facets() (A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices)
>>> from sage.all import * >>> p = polytopes.hypercube(Integer(4)) >>> p.facets() (A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices)
p = polytopes.hypercube(4) p.facets()
This is the same result as explicitly finding the three-dimensional faces:
sage: dim = p.dimension() sage: p.faces(dim-1) (A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices)
>>> from sage.all import * >>> dim = p.dimension() >>> p.faces(dim-Integer(1)) (A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices, A 3-dimensional face of a Polyhedron in ZZ^4 defined as the convex hull of 8 vertices)
dim = p.dimension() p.faces(dim-1)
The
0
-dimensional polyhedron does not have facets:sage: P = Polyhedron([[0]]) sage: P.facets() ()
>>> from sage.all import * >>> P = Polyhedron([[Integer(0)]]) >>> P.facets() ()
P = Polyhedron([[0]]) P.facets()
- greatest_common_subface_of_Hrep(*Hrepresentatives)[source]¶
Return the largest face that is contained in
Hrepresentatives
.INPUT:
Hrepresentatives
– facets or indices of Hrepresentatives; the indices are assumed to be the indices of theHrepresentation()
OUTPUT: a
PolyhedronFace
EXAMPLES:
sage: P = polytopes.permutahedron(5) sage: P.meet_of_Hrep() A 4-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 120 vertices sage: P.meet_of_Hrep(1) A 3-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 24 vertices sage: P.meet_of_Hrep(4) A 3-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 12 vertices sage: P.meet_of_Hrep(1,3,7) A 1-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 2 vertices sage: P.meet_of_Hrep(1,3,7).ambient_H_indices() (0, 1, 3, 7)
>>> from sage.all import * >>> P = polytopes.permutahedron(Integer(5)) >>> P.meet_of_Hrep() A 4-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 120 vertices >>> P.meet_of_Hrep(Integer(1)) A 3-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 24 vertices >>> P.meet_of_Hrep(Integer(4)) A 3-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 12 vertices >>> P.meet_of_Hrep(Integer(1),Integer(3),Integer(7)) A 1-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 2 vertices >>> P.meet_of_Hrep(Integer(1),Integer(3),Integer(7)).ambient_H_indices() (0, 1, 3, 7)
P = polytopes.permutahedron(5) P.meet_of_Hrep() P.meet_of_Hrep(1) P.meet_of_Hrep(4) P.meet_of_Hrep(1,3,7) P.meet_of_Hrep(1,3,7).ambient_H_indices()
The indices are the indices of the
Hrepresentation()
.0
corresponds to an equation and is ignored:sage: P.meet_of_Hrep(0) A 4-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 120 vertices
>>> from sage.all import * >>> P.meet_of_Hrep(Integer(0)) A 4-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 120 vertices
P.meet_of_Hrep(0)
The input is flexible:
sage: P.meet_of_Hrep(P.facets()[-1], P.inequalities()[2], 7) A 1-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 2 vertices
>>> from sage.all import * >>> P.meet_of_Hrep(P.facets()[-Integer(1)], P.inequalities()[Integer(2)], Integer(7)) A 1-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 2 vertices
P.meet_of_Hrep(P.facets()[-1], P.inequalities()[2], 7)
The
Hrepresentatives
must belong toself
:sage: P = polytopes.cube(backend='ppl') sage: Q = polytopes.cube(backend='field') sage: f = P.facets()[0] sage: P.meet_of_Hrep(f) A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices sage: Q.meet_of_Hrep(f) Traceback (most recent call last): ... ValueError: not a facet of ``self`` sage: f = P.inequalities()[0] sage: P.meet_of_Hrep(f) A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices sage: Q.meet_of_Hrep(f) Traceback (most recent call last): ... ValueError: not a facet of ``self``
>>> from sage.all import * >>> P = polytopes.cube(backend='ppl') >>> Q = polytopes.cube(backend='field') >>> f = P.facets()[Integer(0)] >>> P.meet_of_Hrep(f) A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices >>> Q.meet_of_Hrep(f) Traceback (most recent call last): ... ValueError: not a facet of ``self`` >>> f = P.inequalities()[Integer(0)] >>> P.meet_of_Hrep(f) A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices >>> Q.meet_of_Hrep(f) Traceback (most recent call last): ... ValueError: not a facet of ``self``
P = polytopes.cube(backend='ppl') Q = polytopes.cube(backend='field') f = P.facets()[0] P.meet_of_Hrep(f) Q.meet_of_Hrep(f) f = P.inequalities()[0] P.meet_of_Hrep(f) Q.meet_of_Hrep(f)
- incidence_matrix()[source]¶
Return the incidence matrix.
Note
The columns correspond to inequalities/equations in the order
Hrepresentation()
, the rows correspond to vertices/rays/lines in the orderVrepresentation()
.See also
EXAMPLES:
sage: p = polytopes.cuboctahedron() sage: p.incidence_matrix() [0 0 1 1 0 1 0 0 0 0 1 0 0 0] [0 0 0 1 0 0 1 0 1 0 1 0 0 0] [0 0 1 1 1 0 0 1 0 0 0 0 0 0] [1 0 0 1 1 0 1 0 0 0 0 0 0 0] [0 0 0 0 0 1 0 0 1 1 1 0 0 0] [0 0 1 0 0 1 0 1 0 0 0 1 0 0] [1 0 0 0 0 0 1 0 1 0 0 0 1 0] [1 0 0 0 1 0 0 1 0 0 0 0 0 1] [0 1 0 0 0 1 0 0 0 1 0 1 0 0] [0 1 0 0 0 0 0 0 1 1 0 0 1 0] [0 1 0 0 0 0 0 1 0 0 0 1 0 1] [1 1 0 0 0 0 0 0 0 0 0 0 1 1] sage: v = p.Vrepresentation(0) sage: v A vertex at (-1, -1, 0) sage: h = p.Hrepresentation(2) sage: h An inequality (1, 1, -1) x + 2 >= 0 sage: h.eval(v) # evaluation (1, 1, -1) * (-1/2, -1/2, 0) + 1 0 sage: h*v # same as h.eval(v) 0 sage: p.incidence_matrix() [0,2] # this entry is (v,h) 1 sage: h.contains(v) True sage: p.incidence_matrix() [2,0] # note: not symmetric 0
>>> from sage.all import * >>> p = polytopes.cuboctahedron() >>> p.incidence_matrix() [0 0 1 1 0 1 0 0 0 0 1 0 0 0] [0 0 0 1 0 0 1 0 1 0 1 0 0 0] [0 0 1 1 1 0 0 1 0 0 0 0 0 0] [1 0 0 1 1 0 1 0 0 0 0 0 0 0] [0 0 0 0 0 1 0 0 1 1 1 0 0 0] [0 0 1 0 0 1 0 1 0 0 0 1 0 0] [1 0 0 0 0 0 1 0 1 0 0 0 1 0] [1 0 0 0 1 0 0 1 0 0 0 0 0 1] [0 1 0 0 0 1 0 0 0 1 0 1 0 0] [0 1 0 0 0 0 0 0 1 1 0 0 1 0] [0 1 0 0 0 0 0 1 0 0 0 1 0 1] [1 1 0 0 0 0 0 0 0 0 0 0 1 1] >>> v = p.Vrepresentation(Integer(0)) >>> v A vertex at (-1, -1, 0) >>> h = p.Hrepresentation(Integer(2)) >>> h An inequality (1, 1, -1) x + 2 >= 0 >>> h.eval(v) # evaluation (1, 1, -1) * (-1/2, -1/2, 0) + 1 0 >>> h*v # same as h.eval(v) 0 >>> p.incidence_matrix() [Integer(0),Integer(2)] # this entry is (v,h) 1 >>> h.contains(v) True >>> p.incidence_matrix() [Integer(2),Integer(0)] # note: not symmetric 0
p = polytopes.cuboctahedron() p.incidence_matrix() v = p.Vrepresentation(0) v h = p.Hrepresentation(2) h h.eval(v) # evaluation (1, 1, -1) * (-1/2, -1/2, 0) + 1 h*v # same as h.eval(v) p.incidence_matrix() [0,2] # this entry is (v,h) h.contains(v) p.incidence_matrix() [2,0] # note: not symmetric
The incidence matrix depends on the ambient dimension:
sage: simplex = polytopes.simplex(); simplex A 3-dimensional polyhedron in ZZ^4 defined as the convex hull of 4 vertices sage: simplex.incidence_matrix() [1 1 1 1 0] [1 1 1 0 1] [1 1 0 1 1] [1 0 1 1 1] sage: simplex = simplex.affine_hull_projection(); simplex A 3-dimensional polyhedron in ZZ^3 defined as the convex hull of 4 vertices sage: simplex.incidence_matrix() [1 1 1 0] [1 1 0 1] [1 0 1 1] [0 1 1 1]
>>> from sage.all import * >>> simplex = polytopes.simplex(); simplex A 3-dimensional polyhedron in ZZ^4 defined as the convex hull of 4 vertices >>> simplex.incidence_matrix() [1 1 1 1 0] [1 1 1 0 1] [1 1 0 1 1] [1 0 1 1 1] >>> simplex = simplex.affine_hull_projection(); simplex A 3-dimensional polyhedron in ZZ^3 defined as the convex hull of 4 vertices >>> simplex.incidence_matrix() [1 1 1 0] [1 1 0 1] [1 0 1 1] [0 1 1 1]
simplex = polytopes.simplex(); simplex simplex.incidence_matrix() simplex = simplex.affine_hull_projection(); simplex simplex.incidence_matrix()
An incidence matrix does not determine a unique polyhedron:
sage: P = Polyhedron(vertices=[[0,1],[1,1],[1,0]]) sage: P.incidence_matrix() [1 1 0] [1 0 1] [0 1 1] sage: Q = Polyhedron(vertices=[[0,1], [1,0]], rays=[[1,1]]) sage: Q.incidence_matrix() [1 1 0] [1 0 1] [0 1 1]
>>> from sage.all import * >>> P = Polyhedron(vertices=[[Integer(0),Integer(1)],[Integer(1),Integer(1)],[Integer(1),Integer(0)]]) >>> P.incidence_matrix() [1 1 0] [1 0 1] [0 1 1] >>> Q = Polyhedron(vertices=[[Integer(0),Integer(1)], [Integer(1),Integer(0)]], rays=[[Integer(1),Integer(1)]]) >>> Q.incidence_matrix() [1 1 0] [1 0 1] [0 1 1]
P = Polyhedron(vertices=[[0,1],[1,1],[1,0]]) P.incidence_matrix() Q = Polyhedron(vertices=[[0,1], [1,0]], rays=[[1,1]]) Q.incidence_matrix()
An example of two polyhedra with isomorphic face lattices but different incidence matrices:
sage: Q.incidence_matrix() [1 1 0] [1 0 1] [0 1 1] sage: R = Polyhedron(vertices=[[0,1], [1,0]], rays=[[1,3/2], [3/2,1]]) sage: R.incidence_matrix() [1 1 0] [1 0 1] [0 1 0] [0 0 1]
>>> from sage.all import * >>> Q.incidence_matrix() [1 1 0] [1 0 1] [0 1 1] >>> R = Polyhedron(vertices=[[Integer(0),Integer(1)], [Integer(1),Integer(0)]], rays=[[Integer(1),Integer(3)/Integer(2)], [Integer(3)/Integer(2),Integer(1)]]) >>> R.incidence_matrix() [1 1 0] [1 0 1] [0 1 0] [0 0 1]
Q.incidence_matrix() R = Polyhedron(vertices=[[0,1], [1,0]], rays=[[1,3/2], [3/2,1]]) R.incidence_matrix()
The incidence matrix has base ring integers. This way one can express various counting questions:
sage: P = polytopes.twenty_four_cell() sage: M = P.incidence_matrix() sage: sum(sum(x) for x in M) == P.flag_f_vector(0, 3) # needs sage.combinat True
>>> from sage.all import * >>> P = polytopes.twenty_four_cell() >>> M = P.incidence_matrix() >>> sum(sum(x) for x in M) == P.flag_f_vector(Integer(0), Integer(3)) # needs sage.combinat True
P = polytopes.twenty_four_cell() M = P.incidence_matrix() sum(sum(x) for x in M) == P.flag_f_vector(0, 3) # needs sage.combinat
- is_bipyramid(certificate=False)[source]¶
Test whether the polytope is combinatorially equivalent to a bipyramid over some polytope.
INPUT:
certificate
– boolean (default:False
); specifies whether to return two vertices of the polytope which are the apices of a bipyramid, if found
OUTPUT: if
certificate
isTrue
, returns a tuple containing:Boolean.
None
or a tuple containing:The first apex.
The second apex.
If
certificate
isFalse
returns a boolean.EXAMPLES:
sage: P = polytopes.octahedron() sage: P.is_bipyramid() True sage: P.is_bipyramid(certificate=True) (True, [A vertex at (1, 0, 0), A vertex at (-1, 0, 0)]) sage: Q = polytopes.cyclic_polytope(3,7) sage: Q.is_bipyramid() False sage: R = Q.bipyramid() sage: R.is_bipyramid(certificate=True) (True, [A vertex at (1, 3, 13, 63), A vertex at (-1, 3, 13, 63)])
>>> from sage.all import * >>> P = polytopes.octahedron() >>> P.is_bipyramid() True >>> P.is_bipyramid(certificate=True) (True, [A vertex at (1, 0, 0), A vertex at (-1, 0, 0)]) >>> Q = polytopes.cyclic_polytope(Integer(3),Integer(7)) >>> Q.is_bipyramid() False >>> R = Q.bipyramid() >>> R.is_bipyramid(certificate=True) (True, [A vertex at (1, 3, 13, 63), A vertex at (-1, 3, 13, 63)])
P = polytopes.octahedron() P.is_bipyramid() P.is_bipyramid(certificate=True) Q = polytopes.cyclic_polytope(3,7) Q.is_bipyramid() R = Q.bipyramid() R.is_bipyramid(certificate=True)
- is_lawrence_polytope()[source]¶
Return
True
ifself
is a Lawrence polytope.A polytope is called a Lawrence polytope if it has a centrally symmetric (normalized) Gale diagram.
EXAMPLES:
sage: P = polytopes.hypersimplex(5,2) sage: L = P.lawrence_polytope() sage: L.is_lattice_polytope() True sage: egyptian_pyramid = polytopes.regular_polygon(4).pyramid() # needs sage.number_field sage: egyptian_pyramid.is_lawrence_polytope() # needs sage.number_field True sage: polytopes.octahedron().is_lawrence_polytope() False
>>> from sage.all import * >>> P = polytopes.hypersimplex(Integer(5),Integer(2)) >>> L = P.lawrence_polytope() >>> L.is_lattice_polytope() True >>> egyptian_pyramid = polytopes.regular_polygon(Integer(4)).pyramid() # needs sage.number_field >>> egyptian_pyramid.is_lawrence_polytope() # needs sage.number_field True >>> polytopes.octahedron().is_lawrence_polytope() False
P = polytopes.hypersimplex(5,2) L = P.lawrence_polytope() L.is_lattice_polytope() egyptian_pyramid = polytopes.regular_polygon(4).pyramid() # needs sage.number_field egyptian_pyramid.is_lawrence_polytope() # needs sage.number_field polytopes.octahedron().is_lawrence_polytope()
REFERENCES:
For more information, see [BaSt1990].
- is_neighborly(k=None)[source]¶
Return whether the polyhedron is neighborly.
If the input
k
is provided, then return whether the polyhedron isk
-neighborlyA polyhedron is neighborly if every set of \(n\) vertices forms a face for \(n\) up to floor of half the dimension of the polyhedron. It is \(k\)-neighborly if this is true for \(n\) up to \(k\).
INPUT:
k
– the dimension up to which to check if every set ofk
vertices forms a face. If nok
is provided, check up to floor of half the dimension of the polyhedron.
OUTPUT:
True
if every set of up tok
vertices forms a face,False
otherwise
See also
EXAMPLES:
sage: cube = polytopes.hypercube(3) sage: cube.is_neighborly() True sage: cube = polytopes.hypercube(4) sage: cube.is_neighborly() False
>>> from sage.all import * >>> cube = polytopes.hypercube(Integer(3)) >>> cube.is_neighborly() True >>> cube = polytopes.hypercube(Integer(4)) >>> cube.is_neighborly() False
cube = polytopes.hypercube(3) cube.is_neighborly() cube = polytopes.hypercube(4) cube.is_neighborly()
Cyclic polytopes are neighborly:
sage: all(polytopes.cyclic_polytope(i, i + 1 + j).is_neighborly() for i in range(5) for j in range(3)) True
>>> from sage.all import * >>> all(polytopes.cyclic_polytope(i, i + Integer(1) + j).is_neighborly() for i in range(Integer(5)) for j in range(Integer(3))) True
all(polytopes.cyclic_polytope(i, i + 1 + j).is_neighborly() for i in range(5) for j in range(3))
The neighborliness of a polyhedron equals floor of dimension half (or larger in case of a simplex) if and only if the polyhedron is neighborly:
sage: testpolys = [polytopes.cube(), polytopes.cyclic_polytope(6, 9), polytopes.simplex(6)] sage: [(P.neighborliness() >= P.dim() // 2) == P.is_neighborly() for P in testpolys] [True, True, True]
>>> from sage.all import * >>> testpolys = [polytopes.cube(), polytopes.cyclic_polytope(Integer(6), Integer(9)), polytopes.simplex(Integer(6))] >>> [(P.neighborliness() >= P.dim() // Integer(2)) == P.is_neighborly() for P in testpolys] [True, True, True]
testpolys = [polytopes.cube(), polytopes.cyclic_polytope(6, 9), polytopes.simplex(6)] [(P.neighborliness() >= P.dim() // 2) == P.is_neighborly() for P in testpolys]
- is_prism(certificate=False)[source]¶
Test whether the polytope is combinatorially equivalent to a prism of some polytope.
INPUT:
certificate
– boolean (default:False
); specifies whether to return two facets of the polytope which are the bases of a prism, if found
OUTPUT: if
certificate
isTrue
, returns a tuple containing:Boolean.
None
or a tuple containing:List of the vertices of the first base facet.
List of the vertices of the second base facet.
If
certificate
isFalse
returns a boolean.EXAMPLES:
sage: P = polytopes.cube() sage: P.is_prism() True sage: P.is_prism(certificate=True) (True, [(A vertex at (1, -1, -1), A vertex at (1, -1, 1), A vertex at (-1, -1, 1), A vertex at (-1, -1, -1)), (A vertex at (1, 1, -1), A vertex at (1, 1, 1), A vertex at (-1, 1, -1), A vertex at (-1, 1, 1))]) sage: Q = polytopes.cyclic_polytope(3,8) sage: Q.is_prism() False sage: R = Q.prism() sage: R.is_prism(certificate=True) (True, [(A vertex at (0, 3, 9, 27), A vertex at (0, 6, 36, 216), A vertex at (0, 0, 0, 0), A vertex at (0, 7, 49, 343), A vertex at (0, 5, 25, 125), A vertex at (0, 1, 1, 1), A vertex at (0, 2, 4, 8), A vertex at (0, 4, 16, 64)), (A vertex at (1, 6, 36, 216), A vertex at (1, 0, 0, 0), A vertex at (1, 7, 49, 343), A vertex at (1, 5, 25, 125), A vertex at (1, 1, 1, 1), A vertex at (1, 2, 4, 8), A vertex at (1, 4, 16, 64), A vertex at (1, 3, 9, 27))])
>>> from sage.all import * >>> P = polytopes.cube() >>> P.is_prism() True >>> P.is_prism(certificate=True) (True, [(A vertex at (1, -1, -1), A vertex at (1, -1, 1), A vertex at (-1, -1, 1), A vertex at (-1, -1, -1)), (A vertex at (1, 1, -1), A vertex at (1, 1, 1), A vertex at (-1, 1, -1), A vertex at (-1, 1, 1))]) >>> Q = polytopes.cyclic_polytope(Integer(3),Integer(8)) >>> Q.is_prism() False >>> R = Q.prism() >>> R.is_prism(certificate=True) (True, [(A vertex at (0, 3, 9, 27), A vertex at (0, 6, 36, 216), A vertex at (0, 0, 0, 0), A vertex at (0, 7, 49, 343), A vertex at (0, 5, 25, 125), A vertex at (0, 1, 1, 1), A vertex at (0, 2, 4, 8), A vertex at (0, 4, 16, 64)), (A vertex at (1, 6, 36, 216), A vertex at (1, 0, 0, 0), A vertex at (1, 7, 49, 343), A vertex at (1, 5, 25, 125), A vertex at (1, 1, 1, 1), A vertex at (1, 2, 4, 8), A vertex at (1, 4, 16, 64), A vertex at (1, 3, 9, 27))])
P = polytopes.cube() P.is_prism() P.is_prism(certificate=True) Q = polytopes.cyclic_polytope(3,8) Q.is_prism() R = Q.prism() R.is_prism(certificate=True)
- is_pyramid(certificate=False)[source]¶
Test whether the polytope is a pyramid over one of its facets.
INPUT:
certificate
– boolean (default:False
); specifies whether to return a vertex of the polytope which is the apex of a pyramid, if found
OUTPUT: if
certificate
isTrue
, returns a tuple containing:Boolean.
The apex of the pyramid or
None
.
If
certificate
isFalse
returns a boolean.EXAMPLES:
sage: P = polytopes.simplex(3) sage: P.is_pyramid() True sage: P.is_pyramid(certificate=True) (True, A vertex at (1, 0, 0, 0)) sage: egyptian_pyramid = polytopes.regular_polygon(4).pyramid() # needs sage.rings.number_field sage: egyptian_pyramid.is_pyramid() # needs sage.rings.number_field True sage: Q = polytopes.octahedron() sage: Q.is_pyramid() False
>>> from sage.all import * >>> P = polytopes.simplex(Integer(3)) >>> P.is_pyramid() True >>> P.is_pyramid(certificate=True) (True, A vertex at (1, 0, 0, 0)) >>> egyptian_pyramid = polytopes.regular_polygon(Integer(4)).pyramid() # needs sage.rings.number_field >>> egyptian_pyramid.is_pyramid() # needs sage.rings.number_field True >>> Q = polytopes.octahedron() >>> Q.is_pyramid() False
P = polytopes.simplex(3) P.is_pyramid() P.is_pyramid(certificate=True) egyptian_pyramid = polytopes.regular_polygon(4).pyramid() # needs sage.rings.number_field egyptian_pyramid.is_pyramid() # needs sage.rings.number_field Q = polytopes.octahedron() Q.is_pyramid()
For the \(0\)-dimensional polyhedron, the output is
True
, but it cannot be constructed as a pyramid over the empty polyhedron:sage: P = Polyhedron([[0]]) sage: P.is_pyramid() True sage: Polyhedron().pyramid() Traceback (most recent call last): ... ZeroDivisionError: rational division by zero
>>> from sage.all import * >>> P = Polyhedron([[Integer(0)]]) >>> P.is_pyramid() True >>> Polyhedron().pyramid() Traceback (most recent call last): ... ZeroDivisionError: rational division by zero
P = Polyhedron([[0]]) P.is_pyramid() Polyhedron().pyramid()
- is_simple()[source]¶
Test for simplicity of a polytope.
See Wikipedia article Simple_polytope
EXAMPLES:
sage: p = Polyhedron([[0,0,0],[1,0,0],[0,1,0],[0,0,1]]) sage: p.is_simple() True sage: p = Polyhedron([[0,0,0],[4,4,0],[4,0,0],[0,4,0],[2,2,2]]) sage: p.is_simple() False
>>> from sage.all import * >>> p = Polyhedron([[Integer(0),Integer(0),Integer(0)],[Integer(1),Integer(0),Integer(0)],[Integer(0),Integer(1),Integer(0)],[Integer(0),Integer(0),Integer(1)]]) >>> p.is_simple() True >>> p = Polyhedron([[Integer(0),Integer(0),Integer(0)],[Integer(4),Integer(4),Integer(0)],[Integer(4),Integer(0),Integer(0)],[Integer(0),Integer(4),Integer(0)],[Integer(2),Integer(2),Integer(2)]]) >>> p.is_simple() False
p = Polyhedron([[0,0,0],[1,0,0],[0,1,0],[0,0,1]]) p.is_simple() p = Polyhedron([[0,0,0],[4,4,0],[4,0,0],[0,4,0],[2,2,2]]) p.is_simple()
- is_simplex()[source]¶
Return whether the polyhedron is a simplex.
A simplex is a bounded polyhedron with \(d+1\) vertices, where \(d\) is the dimension.
EXAMPLES:
sage: Polyhedron([(0,0,0), (1,0,0), (0,1,0)]).is_simplex() True sage: polytopes.simplex(3).is_simplex() True sage: polytopes.hypercube(3).is_simplex() False
>>> from sage.all import * >>> Polyhedron([(Integer(0),Integer(0),Integer(0)), (Integer(1),Integer(0),Integer(0)), (Integer(0),Integer(1),Integer(0))]).is_simplex() True >>> polytopes.simplex(Integer(3)).is_simplex() True >>> polytopes.hypercube(Integer(3)).is_simplex() False
Polyhedron([(0,0,0), (1,0,0), (0,1,0)]).is_simplex() polytopes.simplex(3).is_simplex() polytopes.hypercube(3).is_simplex()
- is_simplicial()[source]¶
Test if the polytope is simplicial.
A polytope is simplicial if every facet is a simplex.
See Wikipedia article Simplicial_polytope
EXAMPLES:
sage: p = polytopes.hypercube(3) sage: p.is_simplicial() False sage: q = polytopes.simplex(5, project=True) sage: q.is_simplicial() True sage: p = Polyhedron([[0,0,0],[1,0,0],[0,1,0],[0,0,1]]) sage: p.is_simplicial() True sage: q = Polyhedron([[1,1,1],[-1,1,1],[1,-1,1],[-1,-1,1],[1,1,-1]]) sage: q.is_simplicial() False sage: P = polytopes.simplex(); P A 3-dimensional polyhedron in ZZ^4 defined as the convex hull of 4 vertices sage: P.is_simplicial() True
>>> from sage.all import * >>> p = polytopes.hypercube(Integer(3)) >>> p.is_simplicial() False >>> q = polytopes.simplex(Integer(5), project=True) >>> q.is_simplicial() True >>> p = Polyhedron([[Integer(0),Integer(0),Integer(0)],[Integer(1),Integer(0),Integer(0)],[Integer(0),Integer(1),Integer(0)],[Integer(0),Integer(0),Integer(1)]]) >>> p.is_simplicial() True >>> q = Polyhedron([[Integer(1),Integer(1),Integer(1)],[-Integer(1),Integer(1),Integer(1)],[Integer(1),-Integer(1),Integer(1)],[-Integer(1),-Integer(1),Integer(1)],[Integer(1),Integer(1),-Integer(1)]]) >>> q.is_simplicial() False >>> P = polytopes.simplex(); P A 3-dimensional polyhedron in ZZ^4 defined as the convex hull of 4 vertices >>> P.is_simplicial() True
p = polytopes.hypercube(3) p.is_simplicial() q = polytopes.simplex(5, project=True) q.is_simplicial() p = Polyhedron([[0,0,0],[1,0,0],[0,1,0],[0,0,1]]) p.is_simplicial() q = Polyhedron([[1,1,1],[-1,1,1],[1,-1,1],[-1,-1,1],[1,1,-1]]) q.is_simplicial() P = polytopes.simplex(); P P.is_simplicial()
The method is not implemented for unbounded polyhedra:
sage: p = Polyhedron(vertices=[(0,0)],rays=[(1,0),(0,1)]) sage: p.is_simplicial() Traceback (most recent call last): ... NotImplementedError: this function is implemented for polytopes only
>>> from sage.all import * >>> p = Polyhedron(vertices=[(Integer(0),Integer(0))],rays=[(Integer(1),Integer(0)),(Integer(0),Integer(1))]) >>> p.is_simplicial() Traceback (most recent call last): ... NotImplementedError: this function is implemented for polytopes only
p = Polyhedron(vertices=[(0,0)],rays=[(1,0),(0,1)]) p.is_simplicial()
- join_of_Vrep(*Vrepresentatives)[source]¶
Return the smallest face that contains
Vrepresentatives
.INPUT:
Vrepresentatives
– vertices/rays/lines ofself
or indices of such
OUTPUT: a
PolyhedronFace
Note
In the case of unbounded polyhedra, the join of rays etc. may not be well-defined.
EXAMPLES:
sage: P = polytopes.permutahedron(5) sage: P.join_of_Vrep(1) A 0-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 1 vertex sage: P.join_of_Vrep() A -1-dimensional face of a Polyhedron in ZZ^5 sage: P.join_of_Vrep(0,12,13).ambient_V_indices() (0, 12, 13, 68)
>>> from sage.all import * >>> P = polytopes.permutahedron(Integer(5)) >>> P.join_of_Vrep(Integer(1)) A 0-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 1 vertex >>> P.join_of_Vrep() A -1-dimensional face of a Polyhedron in ZZ^5 >>> P.join_of_Vrep(Integer(0),Integer(12),Integer(13)).ambient_V_indices() (0, 12, 13, 68)
P = polytopes.permutahedron(5) P.join_of_Vrep(1) P.join_of_Vrep() P.join_of_Vrep(0,12,13).ambient_V_indices()
The input is flexible:
sage: P.join_of_Vrep(2, P.vertices()[3], P.Vrepresentation(4)) A 2-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 6 vertices
>>> from sage.all import * >>> P.join_of_Vrep(Integer(2), P.vertices()[Integer(3)], P.Vrepresentation(Integer(4))) A 2-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 6 vertices
P.join_of_Vrep(2, P.vertices()[3], P.Vrepresentation(4))
sage: P = polytopes.cube() sage: a, b = P.faces(0)[:2] sage: P.join_of_Vrep(a, b) A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices
>>> from sage.all import * >>> P = polytopes.cube() >>> a, b = P.faces(Integer(0))[:Integer(2)] >>> P.join_of_Vrep(a, b) A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices
P = polytopes.cube() a, b = P.faces(0)[:2] P.join_of_Vrep(a, b)
In the case of an unbounded polyhedron, the join may not be well-defined:
sage: P = Polyhedron(vertices=[[1,0], [0,1]], rays=[[1,1]]) sage: P.join_of_Vrep(0) A 0-dimensional face of a Polyhedron in QQ^2 defined as the convex hull of 1 vertex sage: P.join_of_Vrep(0,1) A 1-dimensional face of a Polyhedron in QQ^2 defined as the convex hull of 2 vertices sage: P.join_of_Vrep(0,2) A 1-dimensional face of a Polyhedron in QQ^2 defined as the convex hull of 1 vertex and 1 ray sage: P.join_of_Vrep(1,2) A 1-dimensional face of a Polyhedron in QQ^2 defined as the convex hull of 1 vertex and 1 ray sage: P.join_of_Vrep(2) Traceback (most recent call last): ... ValueError: the join is not well-defined
>>> from sage.all import * >>> P = Polyhedron(vertices=[[Integer(1),Integer(0)], [Integer(0),Integer(1)]], rays=[[Integer(1),Integer(1)]]) >>> P.join_of_Vrep(Integer(0)) A 0-dimensional face of a Polyhedron in QQ^2 defined as the convex hull of 1 vertex >>> P.join_of_Vrep(Integer(0),Integer(1)) A 1-dimensional face of a Polyhedron in QQ^2 defined as the convex hull of 2 vertices >>> P.join_of_Vrep(Integer(0),Integer(2)) A 1-dimensional face of a Polyhedron in QQ^2 defined as the convex hull of 1 vertex and 1 ray >>> P.join_of_Vrep(Integer(1),Integer(2)) A 1-dimensional face of a Polyhedron in QQ^2 defined as the convex hull of 1 vertex and 1 ray >>> P.join_of_Vrep(Integer(2)) Traceback (most recent call last): ... ValueError: the join is not well-defined
P = Polyhedron(vertices=[[1,0], [0,1]], rays=[[1,1]]) P.join_of_Vrep(0) P.join_of_Vrep(0,1) P.join_of_Vrep(0,2) P.join_of_Vrep(1,2) P.join_of_Vrep(2)
The
Vrepresentatives
must be ofself
:sage: P = polytopes.cube(backend='ppl') sage: Q = polytopes.cube(backend='field') sage: v = P.vertices()[0] sage: P.join_of_Vrep(v) A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex sage: Q.join_of_Vrep(v) Traceback (most recent call last): ... ValueError: not a Vrepresentative of ``self`` sage: f = P.faces(0)[0] sage: P.join_of_Vrep(v) A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex sage: Q.join_of_Vrep(v) Traceback (most recent call last): ... ValueError: not a Vrepresentative of ``self``
>>> from sage.all import * >>> P = polytopes.cube(backend='ppl') >>> Q = polytopes.cube(backend='field') >>> v = P.vertices()[Integer(0)] >>> P.join_of_Vrep(v) A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex >>> Q.join_of_Vrep(v) Traceback (most recent call last): ... ValueError: not a Vrepresentative of ``self`` >>> f = P.faces(Integer(0))[Integer(0)] >>> P.join_of_Vrep(v) A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex >>> Q.join_of_Vrep(v) Traceback (most recent call last): ... ValueError: not a Vrepresentative of ``self``
P = polytopes.cube(backend='ppl') Q = polytopes.cube(backend='field') v = P.vertices()[0] P.join_of_Vrep(v) Q.join_of_Vrep(v) f = P.faces(0)[0] P.join_of_Vrep(v) Q.join_of_Vrep(v)
- least_common_superface_of_Vrep(*Vrepresentatives)[source]¶
Return the smallest face that contains
Vrepresentatives
.INPUT:
Vrepresentatives
– vertices/rays/lines ofself
or indices of such
OUTPUT: a
PolyhedronFace
Note
In the case of unbounded polyhedra, the join of rays etc. may not be well-defined.
EXAMPLES:
sage: P = polytopes.permutahedron(5) sage: P.join_of_Vrep(1) A 0-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 1 vertex sage: P.join_of_Vrep() A -1-dimensional face of a Polyhedron in ZZ^5 sage: P.join_of_Vrep(0,12,13).ambient_V_indices() (0, 12, 13, 68)
>>> from sage.all import * >>> P = polytopes.permutahedron(Integer(5)) >>> P.join_of_Vrep(Integer(1)) A 0-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 1 vertex >>> P.join_of_Vrep() A -1-dimensional face of a Polyhedron in ZZ^5 >>> P.join_of_Vrep(Integer(0),Integer(12),Integer(13)).ambient_V_indices() (0, 12, 13, 68)
P = polytopes.permutahedron(5) P.join_of_Vrep(1) P.join_of_Vrep() P.join_of_Vrep(0,12,13).ambient_V_indices()
The input is flexible:
sage: P.join_of_Vrep(2, P.vertices()[3], P.Vrepresentation(4)) A 2-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 6 vertices
>>> from sage.all import * >>> P.join_of_Vrep(Integer(2), P.vertices()[Integer(3)], P.Vrepresentation(Integer(4))) A 2-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 6 vertices
P.join_of_Vrep(2, P.vertices()[3], P.Vrepresentation(4))
sage: P = polytopes.cube() sage: a, b = P.faces(0)[:2] sage: P.join_of_Vrep(a, b) A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices
>>> from sage.all import * >>> P = polytopes.cube() >>> a, b = P.faces(Integer(0))[:Integer(2)] >>> P.join_of_Vrep(a, b) A 1-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 2 vertices
P = polytopes.cube() a, b = P.faces(0)[:2] P.join_of_Vrep(a, b)
In the case of an unbounded polyhedron, the join may not be well-defined:
sage: P = Polyhedron(vertices=[[1,0], [0,1]], rays=[[1,1]]) sage: P.join_of_Vrep(0) A 0-dimensional face of a Polyhedron in QQ^2 defined as the convex hull of 1 vertex sage: P.join_of_Vrep(0,1) A 1-dimensional face of a Polyhedron in QQ^2 defined as the convex hull of 2 vertices sage: P.join_of_Vrep(0,2) A 1-dimensional face of a Polyhedron in QQ^2 defined as the convex hull of 1 vertex and 1 ray sage: P.join_of_Vrep(1,2) A 1-dimensional face of a Polyhedron in QQ^2 defined as the convex hull of 1 vertex and 1 ray sage: P.join_of_Vrep(2) Traceback (most recent call last): ... ValueError: the join is not well-defined
>>> from sage.all import * >>> P = Polyhedron(vertices=[[Integer(1),Integer(0)], [Integer(0),Integer(1)]], rays=[[Integer(1),Integer(1)]]) >>> P.join_of_Vrep(Integer(0)) A 0-dimensional face of a Polyhedron in QQ^2 defined as the convex hull of 1 vertex >>> P.join_of_Vrep(Integer(0),Integer(1)) A 1-dimensional face of a Polyhedron in QQ^2 defined as the convex hull of 2 vertices >>> P.join_of_Vrep(Integer(0),Integer(2)) A 1-dimensional face of a Polyhedron in QQ^2 defined as the convex hull of 1 vertex and 1 ray >>> P.join_of_Vrep(Integer(1),Integer(2)) A 1-dimensional face of a Polyhedron in QQ^2 defined as the convex hull of 1 vertex and 1 ray >>> P.join_of_Vrep(Integer(2)) Traceback (most recent call last): ... ValueError: the join is not well-defined
P = Polyhedron(vertices=[[1,0], [0,1]], rays=[[1,1]]) P.join_of_Vrep(0) P.join_of_Vrep(0,1) P.join_of_Vrep(0,2) P.join_of_Vrep(1,2) P.join_of_Vrep(2)
The
Vrepresentatives
must be ofself
:sage: P = polytopes.cube(backend='ppl') sage: Q = polytopes.cube(backend='field') sage: v = P.vertices()[0] sage: P.join_of_Vrep(v) A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex sage: Q.join_of_Vrep(v) Traceback (most recent call last): ... ValueError: not a Vrepresentative of ``self`` sage: f = P.faces(0)[0] sage: P.join_of_Vrep(v) A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex sage: Q.join_of_Vrep(v) Traceback (most recent call last): ... ValueError: not a Vrepresentative of ``self``
>>> from sage.all import * >>> P = polytopes.cube(backend='ppl') >>> Q = polytopes.cube(backend='field') >>> v = P.vertices()[Integer(0)] >>> P.join_of_Vrep(v) A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex >>> Q.join_of_Vrep(v) Traceback (most recent call last): ... ValueError: not a Vrepresentative of ``self`` >>> f = P.faces(Integer(0))[Integer(0)] >>> P.join_of_Vrep(v) A 0-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 1 vertex >>> Q.join_of_Vrep(v) Traceback (most recent call last): ... ValueError: not a Vrepresentative of ``self``
P = polytopes.cube(backend='ppl') Q = polytopes.cube(backend='field') v = P.vertices()[0] P.join_of_Vrep(v) Q.join_of_Vrep(v) f = P.faces(0)[0] P.join_of_Vrep(v) Q.join_of_Vrep(v)
- meet_of_Hrep(*Hrepresentatives)[source]¶
Return the largest face that is contained in
Hrepresentatives
.INPUT:
Hrepresentatives
– facets or indices of Hrepresentatives; the indices are assumed to be the indices of theHrepresentation()
OUTPUT: a
PolyhedronFace
EXAMPLES:
sage: P = polytopes.permutahedron(5) sage: P.meet_of_Hrep() A 4-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 120 vertices sage: P.meet_of_Hrep(1) A 3-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 24 vertices sage: P.meet_of_Hrep(4) A 3-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 12 vertices sage: P.meet_of_Hrep(1,3,7) A 1-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 2 vertices sage: P.meet_of_Hrep(1,3,7).ambient_H_indices() (0, 1, 3, 7)
>>> from sage.all import * >>> P = polytopes.permutahedron(Integer(5)) >>> P.meet_of_Hrep() A 4-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 120 vertices >>> P.meet_of_Hrep(Integer(1)) A 3-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 24 vertices >>> P.meet_of_Hrep(Integer(4)) A 3-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 12 vertices >>> P.meet_of_Hrep(Integer(1),Integer(3),Integer(7)) A 1-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 2 vertices >>> P.meet_of_Hrep(Integer(1),Integer(3),Integer(7)).ambient_H_indices() (0, 1, 3, 7)
P = polytopes.permutahedron(5) P.meet_of_Hrep() P.meet_of_Hrep(1) P.meet_of_Hrep(4) P.meet_of_Hrep(1,3,7) P.meet_of_Hrep(1,3,7).ambient_H_indices()
The indices are the indices of the
Hrepresentation()
.0
corresponds to an equation and is ignored:sage: P.meet_of_Hrep(0) A 4-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 120 vertices
>>> from sage.all import * >>> P.meet_of_Hrep(Integer(0)) A 4-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 120 vertices
P.meet_of_Hrep(0)
The input is flexible:
sage: P.meet_of_Hrep(P.facets()[-1], P.inequalities()[2], 7) A 1-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 2 vertices
>>> from sage.all import * >>> P.meet_of_Hrep(P.facets()[-Integer(1)], P.inequalities()[Integer(2)], Integer(7)) A 1-dimensional face of a Polyhedron in ZZ^5 defined as the convex hull of 2 vertices
P.meet_of_Hrep(P.facets()[-1], P.inequalities()[2], 7)
The
Hrepresentatives
must belong toself
:sage: P = polytopes.cube(backend='ppl') sage: Q = polytopes.cube(backend='field') sage: f = P.facets()[0] sage: P.meet_of_Hrep(f) A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices sage: Q.meet_of_Hrep(f) Traceback (most recent call last): ... ValueError: not a facet of ``self`` sage: f = P.inequalities()[0] sage: P.meet_of_Hrep(f) A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices sage: Q.meet_of_Hrep(f) Traceback (most recent call last): ... ValueError: not a facet of ``self``
>>> from sage.all import * >>> P = polytopes.cube(backend='ppl') >>> Q = polytopes.cube(backend='field') >>> f = P.facets()[Integer(0)] >>> P.meet_of_Hrep(f) A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices >>> Q.meet_of_Hrep(f) Traceback (most recent call last): ... ValueError: not a facet of ``self`` >>> f = P.inequalities()[Integer(0)] >>> P.meet_of_Hrep(f) A 2-dimensional face of a Polyhedron in ZZ^3 defined as the convex hull of 4 vertices >>> Q.meet_of_Hrep(f) Traceback (most recent call last): ... ValueError: not a facet of ``self``
P = polytopes.cube(backend='ppl') Q = polytopes.cube(backend='field') f = P.facets()[0] P.meet_of_Hrep(f) Q.meet_of_Hrep(f) f = P.inequalities()[0] P.meet_of_Hrep(f) Q.meet_of_Hrep(f)
- neighborliness()[source]¶
Return the largest
k
, such that the polyhedron isk
-neighborly.A polyhedron is \(k\)-neighborly if every set of \(n\) vertices forms a face for \(n\) up to \(k\).
In case of the \(d\)-dimensional simplex, it returns \(d + 1\).
See also
EXAMPLES:
sage: cube = polytopes.cube() sage: cube.neighborliness() 1 sage: P = Polyhedron(); P The empty polyhedron in ZZ^0 sage: P.neighborliness() 0 sage: P = Polyhedron([[0]]); P A 0-dimensional polyhedron in ZZ^1 defined as the convex hull of 1 vertex sage: P.neighborliness() 1 sage: S = polytopes.simplex(5); S A 5-dimensional polyhedron in ZZ^6 defined as the convex hull of 6 vertices sage: S.neighborliness() 6 sage: C = polytopes.cyclic_polytope(7,10); C A 7-dimensional polyhedron in QQ^7 defined as the convex hull of 10 vertices sage: C.neighborliness() 3 sage: C = polytopes.cyclic_polytope(6,11); C A 6-dimensional polyhedron in QQ^6 defined as the convex hull of 11 vertices sage: C.neighborliness() 3 sage: [polytopes.cyclic_polytope(5,n).neighborliness() for n in range(6,10)] [6, 2, 2, 2]
>>> from sage.all import * >>> cube = polytopes.cube() >>> cube.neighborliness() 1 >>> P = Polyhedron(); P The empty polyhedron in ZZ^0 >>> P.neighborliness() 0 >>> P = Polyhedron([[Integer(0)]]); P A 0-dimensional polyhedron in ZZ^1 defined as the convex hull of 1 vertex >>> P.neighborliness() 1 >>> S = polytopes.simplex(Integer(5)); S A 5-dimensional polyhedron in ZZ^6 defined as the convex hull of 6 vertices >>> S.neighborliness() 6 >>> C = polytopes.cyclic_polytope(Integer(7),Integer(10)); C A 7-dimensional polyhedron in QQ^7 defined as the convex hull of 10 vertices >>> C.neighborliness() 3 >>> C = polytopes.cyclic_polytope(Integer(6),Integer(11)); C A 6-dimensional polyhedron in QQ^6 defined as the convex hull of 11 vertices >>> C.neighborliness() 3 >>> [polytopes.cyclic_polytope(Integer(5),n).neighborliness() for n in range(Integer(6),Integer(10))] [6, 2, 2, 2]
cube = polytopes.cube() cube.neighborliness() P = Polyhedron(); P P.neighborliness() P = Polyhedron([[0]]); P P.neighborliness() S = polytopes.simplex(5); S S.neighborliness() C = polytopes.cyclic_polytope(7,10); C C.neighborliness() C = polytopes.cyclic_polytope(6,11); C C.neighborliness() [polytopes.cyclic_polytope(5,n).neighborliness() for n in range(6,10)]
- simpliciality()[source]¶
Return the largest integer \(k\) such that the polytope is \(k\)-simplicial.
A polytope is \(k\)-simplicial, if every \(k\)-face is a simplex. If
self
is a simplex, returns its dimension.EXAMPLES:
sage: polytopes.cyclic_polytope(10,4).simpliciality() 3 sage: polytopes.hypersimplex(5,2).simpliciality() 2 sage: polytopes.cross_polytope(4).simpliciality() 3 sage: polytopes.simplex(3).simpliciality() 3 sage: polytopes.simplex(1).simpliciality() 1
>>> from sage.all import * >>> polytopes.cyclic_polytope(Integer(10),Integer(4)).simpliciality() 3 >>> polytopes.hypersimplex(Integer(5),Integer(2)).simpliciality() 2 >>> polytopes.cross_polytope(Integer(4)).simpliciality() 3 >>> polytopes.simplex(Integer(3)).simpliciality() 3 >>> polytopes.simplex(Integer(1)).simpliciality() 1
polytopes.cyclic_polytope(10,4).simpliciality() polytopes.hypersimplex(5,2).simpliciality() polytopes.cross_polytope(4).simpliciality() polytopes.simplex(3).simpliciality() polytopes.simplex(1).simpliciality()
The method is not implemented for unbounded polyhedra:
sage: p = Polyhedron(vertices=[(0,0)],rays=[(1,0),(0,1)]) sage: p.simpliciality() Traceback (most recent call last): ... NotImplementedError: this function is implemented for polytopes only
>>> from sage.all import * >>> p = Polyhedron(vertices=[(Integer(0),Integer(0))],rays=[(Integer(1),Integer(0)),(Integer(0),Integer(1))]) >>> p.simpliciality() Traceback (most recent call last): ... NotImplementedError: this function is implemented for polytopes only
p = Polyhedron(vertices=[(0,0)],rays=[(1,0),(0,1)]) p.simpliciality()
- simplicity()[source]¶
Return the largest integer \(k\) such that the polytope is \(k\)-simple.
A polytope \(P\) is \(k\)-simple, if every \((d-1-k)\)-face is contained in exactly \(k+1\) facets of \(P\) for \(1 \leq k \leq d-1\). Equivalently it is \(k\)-simple if the polar/dual polytope is \(k\)-simplicial. If
self
is a simplex, it returns its dimension.EXAMPLES:
sage: polytopes.hypersimplex(4,2).simplicity() 1 sage: polytopes.hypersimplex(5,2).simplicity() 2 sage: polytopes.hypersimplex(6,2).simplicity() 3 sage: polytopes.simplex(3).simplicity() 3 sage: polytopes.simplex(1).simplicity() 1
>>> from sage.all import * >>> polytopes.hypersimplex(Integer(4),Integer(2)).simplicity() 1 >>> polytopes.hypersimplex(Integer(5),Integer(2)).simplicity() 2 >>> polytopes.hypersimplex(Integer(6),Integer(2)).simplicity() 3 >>> polytopes.simplex(Integer(3)).simplicity() 3 >>> polytopes.simplex(Integer(1)).simplicity() 1
polytopes.hypersimplex(4,2).simplicity() polytopes.hypersimplex(5,2).simplicity() polytopes.hypersimplex(6,2).simplicity() polytopes.simplex(3).simplicity() polytopes.simplex(1).simplicity()
The method is not implemented for unbounded polyhedra:
sage: p = Polyhedron(vertices=[(0,0)],rays=[(1,0),(0,1)]) sage: p.simplicity() Traceback (most recent call last): ... NotImplementedError: this function is implemented for polytopes only
>>> from sage.all import * >>> p = Polyhedron(vertices=[(Integer(0),Integer(0))],rays=[(Integer(1),Integer(0)),(Integer(0),Integer(1))]) >>> p.simplicity() Traceback (most recent call last): ... NotImplementedError: this function is implemented for polytopes only
p = Polyhedron(vertices=[(0,0)],rays=[(1,0),(0,1)]) p.simplicity()
- slack_matrix()[source]¶
Return the slack matrix.
The entries correspond to the evaluation of the Hrepresentation elements on the Vrepresentation elements.
Note
The columns correspond to inequalities/equations in the order
Hrepresentation()
, the rows correspond to vertices/rays/lines in the orderVrepresentation()
.See also
EXAMPLES:
sage: P = polytopes.cube() sage: P.slack_matrix() [0 2 2 2 0 0] [0 0 2 2 0 2] [0 0 0 2 2 2] [0 2 0 2 2 0] [2 2 0 0 2 0] [2 2 2 0 0 0] [2 0 2 0 0 2] [2 0 0 0 2 2] sage: P = polytopes.cube(intervals='zero_one') sage: P.slack_matrix() [0 1 1 1 0 0] [0 0 1 1 0 1] [0 0 0 1 1 1] [0 1 0 1 1 0] [1 1 0 0 1 0] [1 1 1 0 0 0] [1 0 1 0 0 1] [1 0 0 0 1 1] sage: # needs sage.rings.number_field sage: P = polytopes.dodecahedron().faces(2)[0].as_polyhedron() sage: P.slack_matrix() [1/2*sqrt5 - 1/2 0 0 1 1/2*sqrt5 - 1/2 0] [ 0 0 1/2*sqrt5 - 1/2 1/2*sqrt5 - 1/2 1 0] [ 0 1/2*sqrt5 - 1/2 1 0 1/2*sqrt5 - 1/2 0] [ 1 1/2*sqrt5 - 1/2 0 1/2*sqrt5 - 1/2 0 0] [1/2*sqrt5 - 1/2 1 1/2*sqrt5 - 1/2 0 0 0] sage: P = Polyhedron(rays=[[1, 0], [0, 1]]) sage: P.slack_matrix() [0 0] [0 1] [1 0]
>>> from sage.all import * >>> P = polytopes.cube() >>> P.slack_matrix() [0 2 2 2 0 0] [0 0 2 2 0 2] [0 0 0 2 2 2] [0 2 0 2 2 0] [2 2 0 0 2 0] [2 2 2 0 0 0] [2 0 2 0 0 2] [2 0 0 0 2 2] >>> P = polytopes.cube(intervals='zero_one') >>> P.slack_matrix() [0 1 1 1 0 0] [0 0 1 1 0 1] [0 0 0 1 1 1] [0 1 0 1 1 0] [1 1 0 0 1 0] [1 1 1 0 0 0] [1 0 1 0 0 1] [1 0 0 0 1 1] >>> # needs sage.rings.number_field >>> P = polytopes.dodecahedron().faces(Integer(2))[Integer(0)].as_polyhedron() >>> P.slack_matrix() [1/2*sqrt5 - 1/2 0 0 1 1/2*sqrt5 - 1/2 0] [ 0 0 1/2*sqrt5 - 1/2 1/2*sqrt5 - 1/2 1 0] [ 0 1/2*sqrt5 - 1/2 1 0 1/2*sqrt5 - 1/2 0] [ 1 1/2*sqrt5 - 1/2 0 1/2*sqrt5 - 1/2 0 0] [1/2*sqrt5 - 1/2 1 1/2*sqrt5 - 1/2 0 0 0] >>> P = Polyhedron(rays=[[Integer(1), Integer(0)], [Integer(0), Integer(1)]]) >>> P.slack_matrix() [0 0] [0 1] [1 0]
P = polytopes.cube() P.slack_matrix() P = polytopes.cube(intervals='zero_one') P.slack_matrix() # needs sage.rings.number_field P = polytopes.dodecahedron().faces(2)[0].as_polyhedron() P.slack_matrix() P = Polyhedron(rays=[[1, 0], [0, 1]]) P.slack_matrix()
- vertex_adjacency_matrix(algorithm=None)[source]¶
Return the binary matrix of vertex adjacencies.
INPUT:
algorithm
– string (optional); specify whether the face generator starts with facets or vertices:'primal'
– start with the facets'dual'
– start with the verticesNone
– choose automatically
EXAMPLES:
sage: polytopes.simplex(4).vertex_adjacency_matrix() [0 1 1 1 1] [1 0 1 1 1] [1 1 0 1 1] [1 1 1 0 1] [1 1 1 1 0]
>>> from sage.all import * >>> polytopes.simplex(Integer(4)).vertex_adjacency_matrix() [0 1 1 1 1] [1 0 1 1 1] [1 1 0 1 1] [1 1 1 0 1] [1 1 1 1 0]
polytopes.simplex(4).vertex_adjacency_matrix()
The rows and columns of the vertex adjacency matrix correspond to the
Vrepresentation()
objects: vertices, rays, and lines. The \((i,j)\) matrix entry equals \(1\) if the \(i\)-th and \(j\)-th V-representation object are adjacent.Two vertices are adjacent if they are the endpoints of an edge, that is, a one-dimensional face. For unbounded polyhedra this clearly needs to be generalized and we define two V-representation objects (see
sage.geometry.polyhedron.constructor
) to be adjacent if they together generate a one-face. There are three possible combinations:Two vertices can bound a finite-length edge.
A vertex and a ray can generate a half-infinite edge starting at the vertex and with the direction given by the ray.
A vertex and a line can generate an infinite edge. The position of the vertex on the line is arbitrary in this case, only its transverse position matters. The direction of the edge is given by the line generator.
For example, take the half-plane:
sage: half_plane = Polyhedron(ieqs=[(0,1,0)]) sage: half_plane.Hrepresentation() (An inequality (1, 0) x + 0 >= 0,)
>>> from sage.all import * >>> half_plane = Polyhedron(ieqs=[(Integer(0),Integer(1),Integer(0))]) >>> half_plane.Hrepresentation() (An inequality (1, 0) x + 0 >= 0,)
half_plane = Polyhedron(ieqs=[(0,1,0)]) half_plane.Hrepresentation()
Its (non-unique) V-representation consists of a vertex, a ray, and a line. The only edge is spanned by the vertex and the line generator, so they are adjacent:
sage: half_plane.Vrepresentation() (A line in the direction (0, 1), A ray in the direction (1, 0), A vertex at (0, 0)) sage: half_plane.vertex_adjacency_matrix() [0 0 1] [0 0 0] [1 0 0]
>>> from sage.all import * >>> half_plane.Vrepresentation() (A line in the direction (0, 1), A ray in the direction (1, 0), A vertex at (0, 0)) >>> half_plane.vertex_adjacency_matrix() [0 0 1] [0 0 0] [1 0 0]
half_plane.Vrepresentation() half_plane.vertex_adjacency_matrix()
In one dimension higher, that is for a half-space in 3 dimensions, there is no one-dimensional face. Hence nothing is adjacent:
sage: Polyhedron(ieqs=[(0,1,0,0)]).vertex_adjacency_matrix() [0 0 0 0] [0 0 0 0] [0 0 0 0] [0 0 0 0]
>>> from sage.all import * >>> Polyhedron(ieqs=[(Integer(0),Integer(1),Integer(0),Integer(0))]).vertex_adjacency_matrix() [0 0 0 0] [0 0 0 0] [0 0 0 0] [0 0 0 0]
Polyhedron(ieqs=[(0,1,0,0)]).vertex_adjacency_matrix()
EXAMPLES:
In a bounded polygon, every vertex has precisely two adjacent ones:
sage: P = Polyhedron(vertices=[(0, 1), (1, 0), (3, 0), (4, 1)]) sage: for v in P.Vrep_generator(): ....: print("{} {}".format(P.adjacency_matrix().row(v.index()), v)) (0, 1, 0, 1) A vertex at (0, 1) (1, 0, 1, 0) A vertex at (1, 0) (0, 1, 0, 1) A vertex at (3, 0) (1, 0, 1, 0) A vertex at (4, 1)
>>> from sage.all import * >>> P = Polyhedron(vertices=[(Integer(0), Integer(1)), (Integer(1), Integer(0)), (Integer(3), Integer(0)), (Integer(4), Integer(1))]) >>> for v in P.Vrep_generator(): ... print("{} {}".format(P.adjacency_matrix().row(v.index()), v)) (0, 1, 0, 1) A vertex at (0, 1) (1, 0, 1, 0) A vertex at (1, 0) (0, 1, 0, 1) A vertex at (3, 0) (1, 0, 1, 0) A vertex at (4, 1)
P = Polyhedron(vertices=[(0, 1), (1, 0), (3, 0), (4, 1)]) for v in P.Vrep_generator(): print("{} {}".format(P.adjacency_matrix().row(v.index()), v))
If the V-representation of the polygon contains vertices and one ray, then each V-representation object is adjacent to two V-representation objects:
sage: P = Polyhedron(vertices=[(0, 1), (1, 0), (3, 0), (4, 1)], ....: rays=[(0,1)]) sage: for v in P.Vrep_generator(): ....: print("{} {}".format(P.adjacency_matrix().row(v.index()), v)) (0, 1, 0, 0, 1) A ray in the direction (0, 1) (1, 0, 1, 0, 0) A vertex at (0, 1) (0, 1, 0, 1, 0) A vertex at (1, 0) (0, 0, 1, 0, 1) A vertex at (3, 0) (1, 0, 0, 1, 0) A vertex at (4, 1)
>>> from sage.all import * >>> P = Polyhedron(vertices=[(Integer(0), Integer(1)), (Integer(1), Integer(0)), (Integer(3), Integer(0)), (Integer(4), Integer(1))], ... rays=[(Integer(0),Integer(1))]) >>> for v in P.Vrep_generator(): ... print("{} {}".format(P.adjacency_matrix().row(v.index()), v)) (0, 1, 0, 0, 1) A ray in the direction (0, 1) (1, 0, 1, 0, 0) A vertex at (0, 1) (0, 1, 0, 1, 0) A vertex at (1, 0) (0, 0, 1, 0, 1) A vertex at (3, 0) (1, 0, 0, 1, 0) A vertex at (4, 1)
P = Polyhedron(vertices=[(0, 1), (1, 0), (3, 0), (4, 1)], rays=[(0,1)]) for v in P.Vrep_generator(): print("{} {}".format(P.adjacency_matrix().row(v.index()), v))
If the V-representation of the polygon contains vertices and two distinct rays, then each vertex is adjacent to two V-representation objects (which can now be vertices or rays). The two rays are not adjacent to each other:
sage: P = Polyhedron(vertices=[(0, 1), (1, 0), (3, 0), (4, 1)], ....: rays=[(0,1), (1,1)]) sage: for v in P.Vrep_generator(): ....: print("{} {}".format(P.adjacency_matrix().row(v.index()), v)) (0, 1, 0, 0, 0) A ray in the direction (0, 1) (1, 0, 1, 0, 0) A vertex at (0, 1) (0, 1, 0, 0, 1) A vertex at (1, 0) (0, 0, 0, 0, 1) A ray in the direction (1, 1) (0, 0, 1, 1, 0) A vertex at (3, 0)
>>> from sage.all import * >>> P = Polyhedron(vertices=[(Integer(0), Integer(1)), (Integer(1), Integer(0)), (Integer(3), Integer(0)), (Integer(4), Integer(1))], ... rays=[(Integer(0),Integer(1)), (Integer(1),Integer(1))]) >>> for v in P.Vrep_generator(): ... print("{} {}".format(P.adjacency_matrix().row(v.index()), v)) (0, 1, 0, 0, 0) A ray in the direction (0, 1) (1, 0, 1, 0, 0) A vertex at (0, 1) (0, 1, 0, 0, 1) A vertex at (1, 0) (0, 0, 0, 0, 1) A ray in the direction (1, 1) (0, 0, 1, 1, 0) A vertex at (3, 0)
P = Polyhedron(vertices=[(0, 1), (1, 0), (3, 0), (4, 1)], rays=[(0,1), (1,1)]) for v in P.Vrep_generator(): print("{} {}".format(P.adjacency_matrix().row(v.index()), v))
The vertex adjacency matrix has base ring integers. This way one can express various counting questions:
sage: P = polytopes.cube() sage: Q = P.stack(P.faces(2)[0]) sage: M = Q.vertex_adjacency_matrix() sage: sum(M) (4, 4, 3, 3, 4, 4, 4, 3, 3) sage: G = Q.vertex_graph() # needs sage.graphs sage: G.degree() # needs sage.graphs [4, 4, 3, 3, 4, 4, 4, 3, 3]
>>> from sage.all import * >>> P = polytopes.cube() >>> Q = P.stack(P.faces(Integer(2))[Integer(0)]) >>> M = Q.vertex_adjacency_matrix() >>> sum(M) (4, 4, 3, 3, 4, 4, 4, 3, 3) >>> G = Q.vertex_graph() # needs sage.graphs >>> G.degree() # needs sage.graphs [4, 4, 3, 3, 4, 4, 4, 3, 3]
P = polytopes.cube() Q = P.stack(P.faces(2)[0]) M = Q.vertex_adjacency_matrix() sum(M) G = Q.vertex_graph() # needs sage.graphs G.degree() # needs sage.graphs