Encodable types #
This file defines encodable (constructively countable) types as a typeclass.
This is used to provide explicit encode/decode functions from and to ℕ
, with the information that
those functions are inverses of each other.
The difference with Denumerable
is that finite types are encodable. For infinite types,
Encodable
and Denumerable
agree.
Main declarations #
Encodable α
: States that there exists an explicit encoding functionencode : α → ℕ
with a partial inversedecode : ℕ → Option α
.decode₂
: Version ofdecode
that is equal tonone
outside of the range ofencode
. Useful as we do not require this in the definition ofdecode
.ULower α
: Any encodable type has an equivalent type living in the lowest universe, namely a subtype ofℕ
.ULower α
finds it.
Implementation notes #
The point of asking for an explicit partial inverse decode : ℕ → Option α
to encode : α → ℕ
is
to make the range of encode
decidable even when the finiteness of α
is not.
Constructively countable type. Made from an explicit injection encode : α → ℕ
and a partial
inverse decode : ℕ → Option α
. Note that finite types are countable. See Denumerable
if you
wish to enforce infiniteness.
- encode : α → ℕ
Encoding from Type α to ℕ
Decoding from ℕ to Option α
Invariant relationship between encoding and decoding
Instances
An encodable type has decidable equality. Not set as an instance because this is usually not the best way to infer decidability.
Equations
Instances For
If α
is encodable and there is an injection f : β → α
, then β
is encodable as well.
Equations
- Encodable.ofLeftInjection f finv linv = { encode := fun (b : β) => Encodable.encode (f b), decode := fun (n : ℕ) => (Encodable.decode n).bind finv, encodek := ⋯ }
Instances For
If α
is encodable and f : β → α
is invertible, then β
is encodable as well.
Instances For
Encodability is preserved by equivalence.
Instances For
Equations
- Nat.encodable = { encode := id, decode := some, encodek := Nat.encodable.proof_1 }
Equations
- IsEmpty.toEncodable = { encode := fun (a : α) => isEmptyElim a, decode := fun (x : ℕ) => none, encodek := ⋯ }
Equations
- PUnit.encodable = { encode := fun (x : PUnit.{?u.7 + 1}) => 0, decode := fun (n : ℕ) => Nat.casesOn n (some PUnit.unit) fun (x : ℕ) => none, encodek := PUnit.encodable.proof_1 }
The encoding function has decidable range.
Equations
Instances For
A type with unique element is encodable. This is not an instance to avoid diamonds.
Equations
- Unique.encodable = { encode := fun (x : α) => 0, decode := fun (x : ℕ) => some default, encodek := ⋯ }
Instances For
Explicit encoding function for the sum of two encodable types.
Equations
Instances For
If α
and β
are encodable, then so is their sum.
Equations
- Sum.encodable = { encode := Encodable.encodeSum, decode := Encodable.decodeSum, encodek := ⋯ }
Explicit decoding function for Sigma γ
Equations
- Encodable.decodeSigma n = match Nat.unpair n with | (n₁, n₂) => (Encodable.decode n₁).bind fun (a : α) => Option.map (Sigma.mk a) (Encodable.decode n₂)
Instances For
Equations
- Sigma.encodable = { encode := Encodable.encodeSigma, decode := Encodable.decodeSigma, encodek := ⋯ }
If α
and β
are encodable, then so is their product.
Equations
- Encodable.Prod.encodable = Encodable.ofEquiv ((_ : α) × β) (Equiv.sigmaEquivProd α β).symm
Explicit encoding function for a decidable subtype of an encodable type
Instances For
Explicit decoding function for a decidable subtype of an encodable type
Equations
- Encodable.decodeSubtype v = (Encodable.decode v).bind fun (a : α) => if h : P a then some ⟨a, h⟩ else none
Instances For
A decidable subtype of an encodable type is encodable.
Equations
- Subtype.encodable = { encode := Encodable.encodeSubtype, decode := Encodable.decodeSubtype, encodek := ⋯ }
Equations
Equations
The lift of an encodable type is encodable
Equations
The lift of an encodable type is encodable.
Equations
If β
is encodable and there is an injection f : α → β
, then α
is encodable as well.
Instances For
If α
is countable, then it has a (non-canonical) Encodable
structure.
Instances For
Equations
The equivalence between the encodable type α
and ULower α : Type
.
Instances For
Lowers an a : α
into ULower α
.
Instances For
Equations
- ULower.instInhabited = { default := ULower.down default }
Constructive choice function for a decidable subtype of an encodable type.
Equations
- Encodable.chooseX h = match Encodable.decode (Nat.find ⋯), ⋯ with | some a, h => ⟨a, h⟩
Instances For
Constructive choice function for a decidable predicate over an encodable type.
Instances For
A constructive version of Classical.axiom_of_choice
for Encodable
types.
A constructive version of Classical.skolem
for Encodable
types.
Given a Directed r
function f : α → β
defined on an encodable inhabited type,
construct a noncomputable sequence such that r (f (x n)) (f (x (n + 1)))
and r (f a) (f (x (encode a + 1))
.
Equations
Instances For
Representative of an equivalence class. This is a computable version of Quot.out
for a setoid
on an encodable type.
Equations
Instances For
The quotient of an encodable space by a decidable equivalence relation is encodable.
Equations
- encodableQuotient = { encode := fun (q : Quotient s) => Encodable.encode q.rep, decode := fun (n : ℕ) => Quotient.mk'' <$> Encodable.decode n, encodek := ⋯ }