LawfulFunctor
LawfulFunctor
は、Functor
型クラスに関手則を満たすという条件を加えたものです。
関手則とは、関手 F : Type u → Type u
が満たしているべきルールで、以下のようなものです。
Functor.map
は恒等関数を保存する。つまりid <$> x = x
が成り立つ。Functor.map
は関数合成を保存する。つまり(f ∘ g) <$> x = f <$> (g <$> x)
が成り立つ。
LawfulFunctor
クラスは、これをほぼそのままコードに落とし込んだものとして、おおむね次のように定義されています。
open Function
universe u v
variable {α β γ : Type u}
class LawfulFunctor (f : Type u → Type v) [Functor f] : Prop where
/-- `Functor.mapConst` が仕様を満たす -/
map_const : (Functor.mapConst : α → f β → f α) = Functor.map ∘ const β
/-- 恒等関数を保つ -/
id_map (x : f α) : id <$> x = x
/-- 合成を保つ -/
comp_map (g : α → β) (h : β → γ) (x : f α) : (h ∘ g) <$> x = h <$> g <$> x
関手則の帰結
関手則の意味について理解していただくために、関手則の帰結をひとつ紹介します。
まず、型 A, B
は全単射 f : A → B
とその逆射 g : B → A
が存在するとき 同値(equivalent) であるといい、これを (· ≃ ·)
で表します。つまり A ≃ B
であるとは、f : A → B
と g : B → A
が存在して f ∘ g = id
かつ g ∘ f = id
が成り立つことを意味します。
関手則が守られているとき、関手 F
は合成を保ち、かつ id
を id
に写すので、関手は同値性を保つことになります。
section
variable {A B : Type} {F : Type → Type}
example [Functor F] [LawfulFunctor F] (h : A ≃ B) : F A ≃ F B := by
obtain ⟨f, g, hf, hg⟩ := h
-- 関手 `F` による像で同値になる
refine ⟨Functor.map f, Functor.map g, ?hFf, ?hFg⟩
-- infoviewを見やすくする
all_goals
dsimp [Function.RightInverse] at *
dsimp [Function.LeftInverse] at *
case hFf =>
have gfid : g ∘ f = id := by
ext x
simp_all
intro x
have : g <$> f <$> x = x := calc
_ = (g ∘ f) <$> x := by rw [LawfulFunctor.comp_map]
_ = id <$> x := by rw [gfid]
_ = x := by rw [LawfulFunctor.id_map]
assumption
case hFg =>
have fgid : f ∘ g = id := by
ext x
simp_all
intro x
have : f <$> g <$> x = x := calc
_ = (f ∘ g) <$> x := by rw [LawfulFunctor.comp_map]
_ = id <$> x := by rw [fgid]
_ = x := by rw [LawfulFunctor.id_map]
assumption
end
関手の例
いくつか LawfulFunctor
クラスのインスタンスを作ってみます。
Id
Id
は関手則を満たします。
def MyId (α : Type) := α
def MyId.map {α β : Type} (f : α → β) (x : MyId α) : MyId β := f x
instance : Functor MyId where
map := MyId.map
instance : LawfulFunctor MyId where
map_const := by aesop
id_map := by aesop
comp_map := by aesop
List
List
は関手則を満たします。
/-- 自前で定義したリスト -/
inductive MyList (α : Type) where
| nil
| cons (head : α) (tail : MyList α)
notation:80 "[]" => MyList.nil
infixr:80 "::" => MyList.cons
/-- リストの中身に関数をそれぞれ適用する -/
def MyList.map {α β : Type} (f : α → β) (xs : MyList α) : MyList β :=
match xs with
| [] => []
| x :: xs => f x :: MyList.map f xs
instance : Functor MyList where
map := MyList.map
instance : LawfulFunctor MyList where
map_const := by aesop
id_map := by
intro α xs
dsimp [(· <$> ·)]
induction xs with
| nil => rfl
| cons x xs ih =>
dsimp [MyList.map]
rw [ih]
comp_map := by
intro α β γ g h xs
induction xs with
| nil => rfl
| cons x xs ih =>
dsimp [(· <$> ·), MyList.map] at ih ⊢
rw [ih]
Option
Option
は関手則を満たします。
@[aesop unsafe 70% cases]
inductive MyOption (α : Type) where
| none
| some (x : α)
def MyOption.map {α β : Type} (f : α → β) (x : MyOption α) : MyOption β :=
match x with
| none => none
| some x => some (f x)
instance : Functor MyOption where
map := MyOption.map
instance : LawfulFunctor MyOption where
map_const := by aesop
id_map := by aesop
comp_map := by aesop
(A → ·)
fun X => (A → X)
という対応 Type → Type
は関手則を満たします。
/-- 型から、その型への関数型を返す -/
abbrev Hom (A : Type) (X : Type) := A → X
instance {A : Type} : Functor (Hom A) where
map f g := f ∘ g
instance {A : Type} : LawfulFunctor (Hom A) where
map_const := by aesop
id_map := by aesop
comp_map := by aesop