Derangements#
AUTHORS:
Alasdair McAndrew (2010-05): Initial version
Travis Scrimshaw (2013-03-30): Put derangements into category framework
- class sage.combinat.derangements.Derangement(parent, *args, **kwds)[source]#
Bases:
CombinatorialElement
A derangement.
A derangement on a set \(S\) is a permutation \(\sigma\) such that \(\sigma(x) \neq x\) for all \(x \in S\), i.e. \(\sigma\) is a permutation of \(S\) with no fixed points.
EXAMPLES:
sage: D = Derangements(4) sage: elt = D([4,3,2,1]) sage: TestSuite(elt).run()
>>> from sage.all import * >>> D = Derangements(Integer(4)) >>> elt = D([Integer(4),Integer(3),Integer(2),Integer(1)]) >>> TestSuite(elt).run()
- to_permutation()[source]#
Return the permutation corresponding to
self
.EXAMPLES:
sage: D = Derangements(4) sage: p = D([4,3,2,1]).to_permutation(); p [4, 3, 2, 1] sage: type(p) <class 'sage.combinat.permutation.StandardPermutations_all_with_category.element_class'> sage: D = Derangements([1, 3, 3, 4]) sage: D[0].to_permutation() Traceback (most recent call last): ... ValueError: can only convert to a permutation for derangements of [1, 2, ..., n]
>>> from sage.all import * >>> D = Derangements(Integer(4)) >>> p = D([Integer(4),Integer(3),Integer(2),Integer(1)]).to_permutation(); p [4, 3, 2, 1] >>> type(p) <class 'sage.combinat.permutation.StandardPermutations_all_with_category.element_class'> >>> D = Derangements([Integer(1), Integer(3), Integer(3), Integer(4)]) >>> D[Integer(0)].to_permutation() Traceback (most recent call last): ... ValueError: can only convert to a permutation for derangements of [1, 2, ..., n]
- class sage.combinat.derangements.Derangements(x)[source]#
Bases:
UniqueRepresentation
,Parent
The class of all derangements of a set or multiset.
A derangement on a set \(S\) is a permutation \(\sigma\) such that \(\sigma(x) \neq x\) for all \(x \in S\), i.e. \(\sigma\) is a permutation of \(S\) with no fixed points.
For an integer, or a list or string with all elements distinct, the derangements are obtained by a standard result described in [BV2004]. For a list or string with repeated elements, the derangements are formed by computing all permutations of the input and discarding all non-derangements.
INPUT:
x
– Can be an integer which corresponds to derangements of \(\{1, 2, 3, \ldots, x\}\), a list, or a string
REFERENCES:
EXAMPLES:
sage: D1 = Derangements([2,3,4,5]) sage: D1.list() [[3, 4, 5, 2], [5, 4, 2, 3], [3, 5, 2, 4], [4, 5, 3, 2], [4, 2, 5, 3], [5, 2, 3, 4], [5, 4, 3, 2], [4, 5, 2, 3], [3, 2, 5, 4]] sage: D1.cardinality() 9 sage: D1.random_element() # random [4, 2, 5, 3] sage: D2 = Derangements([1,2,3,1,2,3]) sage: D2.cardinality() 10 sage: D2.list() [[2, 1, 1, 3, 3, 2], [2, 1, 2, 3, 3, 1], [2, 3, 1, 2, 3, 1], [2, 3, 1, 3, 1, 2], [2, 3, 2, 3, 1, 1], [3, 1, 1, 2, 3, 2], [3, 1, 2, 2, 3, 1], [3, 1, 2, 3, 1, 2], [3, 3, 1, 2, 1, 2], [3, 3, 2, 2, 1, 1]] sage: D2.random_element() # random [2, 3, 1, 3, 1, 2]
>>> from sage.all import * >>> D1 = Derangements([Integer(2),Integer(3),Integer(4),Integer(5)]) >>> D1.list() [[3, 4, 5, 2], [5, 4, 2, 3], [3, 5, 2, 4], [4, 5, 3, 2], [4, 2, 5, 3], [5, 2, 3, 4], [5, 4, 3, 2], [4, 5, 2, 3], [3, 2, 5, 4]] >>> D1.cardinality() 9 >>> D1.random_element() # random [4, 2, 5, 3] >>> D2 = Derangements([Integer(1),Integer(2),Integer(3),Integer(1),Integer(2),Integer(3)]) >>> D2.cardinality() 10 >>> D2.list() [[2, 1, 1, 3, 3, 2], [2, 1, 2, 3, 3, 1], [2, 3, 1, 2, 3, 1], [2, 3, 1, 3, 1, 2], [2, 3, 2, 3, 1, 1], [3, 1, 1, 2, 3, 2], [3, 1, 2, 2, 3, 1], [3, 1, 2, 3, 1, 2], [3, 3, 1, 2, 1, 2], [3, 3, 2, 2, 1, 1]] >>> D2.random_element() # random [2, 3, 1, 3, 1, 2]
- Element[source]#
alias of
Derangement
- cardinality()[source]#
Counts the number of derangements of a positive integer, a list, or a string. The list or string may contain repeated elements. If an integer \(n\) is given, the value returned is the number of derangements of \([1, 2, 3, \ldots, n]\).
For an integer, or a list or string with all elements distinct, the value is obtained by the standard result \(D_2 = 1, D_3 = 2, D_n = (n-1) (D_{n-1} + D_{n-2})\).
For a list or string with repeated elements, the number of derangements is computed by Macmahon’s theorem. If the numbers of repeated elements are \(a_1, a_2, \ldots, a_k\) then the number of derangements is given by the coefficient of \(x_1 x_2 \cdots x_k\) in the expansion of \(\prod_{i=0}^k (S - s_i)^{a_i}\) where \(S = x_1 + x_2 + \cdots + x_k\).
EXAMPLES:
sage: D = Derangements(5) sage: D.cardinality() 44 sage: D = Derangements([1,44,918,67,254]) sage: D.cardinality() 44 sage: D = Derangements(['A','AT','CAT','CATS','CARTS']) sage: D.cardinality() 44 sage: D = Derangements('UNCOPYRIGHTABLE') sage: D.cardinality() 481066515734 sage: D = Derangements([1,1,2,2,3,3]) sage: D.cardinality() 10 sage: D = Derangements('SATTAS') sage: D.cardinality() 10 sage: D = Derangements([1,1,2,2,2]) sage: D.cardinality() 0
>>> from sage.all import * >>> D = Derangements(Integer(5)) >>> D.cardinality() 44 >>> D = Derangements([Integer(1),Integer(44),Integer(918),Integer(67),Integer(254)]) >>> D.cardinality() 44 >>> D = Derangements(['A','AT','CAT','CATS','CARTS']) >>> D.cardinality() 44 >>> D = Derangements('UNCOPYRIGHTABLE') >>> D.cardinality() 481066515734 >>> D = Derangements([Integer(1),Integer(1),Integer(2),Integer(2),Integer(3),Integer(3)]) >>> D.cardinality() 10 >>> D = Derangements('SATTAS') >>> D.cardinality() 10 >>> D = Derangements([Integer(1),Integer(1),Integer(2),Integer(2),Integer(2)]) >>> D.cardinality() 0
- random_element()[source]#
Produce all derangements of a positive integer, a list, or a string. The list or string may contain repeated elements. If an integer \(n\) is given, then a random derangements of \([1, 2, 3, \ldots, n]\) is returned
For an integer, or a list or string with all elements distinct, the value is obtained by an algorithm described in [MPP2008]. For a list or string with repeated elements the derangement is formed by choosing an element at random from the list of all possible derangements.
OUTPUT:
A single list or string containing a derangement, or an empty list if there are no derangements.
EXAMPLES:
sage: D = Derangements(4) sage: D.random_element() # random [2, 3, 4, 1] sage: D = Derangements(['A','AT','CAT','CATS','CARTS','CARETS']) sage: D.random_element() # random ['AT', 'CARTS', 'A', 'CAT', 'CARETS', 'CATS'] sage: D = Derangements('UNCOPYRIGHTABLE') sage: D.random_element() # random ['C', 'U', 'I', 'H', 'O', 'G', 'N', 'B', 'E', 'L', 'A', 'R', 'P', 'Y', 'T'] sage: D = Derangements([1,1,1,1,2,2,2,2,3,3,3,3]) sage: D.random_element() # random [3, 2, 2, 3, 1, 3, 1, 3, 2, 1, 1, 2] sage: D = Derangements('ESSENCES') sage: D.random_element() # random ['N', 'E', 'E', 'C', 'S', 'S', 'S', 'E'] sage: D = Derangements([1,1,2,2,2]) sage: D.random_element() []
>>> from sage.all import * >>> D = Derangements(Integer(4)) >>> D.random_element() # random [2, 3, 4, 1] >>> D = Derangements(['A','AT','CAT','CATS','CARTS','CARETS']) >>> D.random_element() # random ['AT', 'CARTS', 'A', 'CAT', 'CARETS', 'CATS'] >>> D = Derangements('UNCOPYRIGHTABLE') >>> D.random_element() # random ['C', 'U', 'I', 'H', 'O', 'G', 'N', 'B', 'E', 'L', 'A', 'R', 'P', 'Y', 'T'] >>> D = Derangements([Integer(1),Integer(1),Integer(1),Integer(1),Integer(2),Integer(2),Integer(2),Integer(2),Integer(3),Integer(3),Integer(3),Integer(3)]) >>> D.random_element() # random [3, 2, 2, 3, 1, 3, 1, 3, 2, 1, 1, 2] >>> D = Derangements('ESSENCES') >>> D.random_element() # random ['N', 'E', 'E', 'C', 'S', 'S', 'S', 'E'] >>> D = Derangements([Integer(1),Integer(1),Integer(2),Integer(2),Integer(2)]) >>> D.random_element() []