と のいくつかのプロパティを証明していましたが、このプロパティに出くわすfilter
までmap
、すべてがうまくいきました: filter p (map f xs) ≡ map f (filter (p ∘ f) xs)
。関連するコードの一部を次に示します。
open import Relation.Binary.PropositionalEquality
open import Data.Bool
open import Data.List hiding (filter)
import Level
filter : ∀ {a} {A : Set a} → (A → Bool) → List A → List A
filter _ [] = []
filter p (x ∷ xs) with p x
... | true = x ∷ filter p xs
... | false = filter p xs
≡-Reasoning
さて、モジュールを使って証明を書くのが大好きなので、最初に試したことは次のとおりです。
open ≡-Reasoning
open import Function
filter-map : ∀ {a b} {A : Set a} {B : Set b}
(xs : List A) (f : A → B) (p : B → Bool) →
filter p (map f xs) ≡ map f (filter (p ∘ f) xs)
filter-map [] _ _ = refl
filter-map (x ∷ xs) f p with p (f x)
... | true = begin
filter p (map f (x ∷ xs))
≡⟨ refl ⟩
f x ∷ filter p (map f xs)
-- ...
しかし、残念ながら、それはうまくいきませんでした。1時間試した後、最終的にあきらめて、次のように証明しました。
filter-map (x ∷ xs) f p with p (f x)
... | true = cong (λ a → f x ∷ a) (filter-map xs f p)
... | false = filter-map xs f p
≡-Reasoning
なぜ通り抜けることがうまくいかなかったのかまだ興味があるので、私は非常に些細なことを試しました:
filter-map-def : ∀ {a b} {A : Set a} {B : Set b}
(x : A) xs (f : A → B) (p : B → Bool) → T (p (f x)) →
filter p (map f (x ∷ xs)) ≡ f x ∷ filter p (map f xs)
filter-map-def x xs f p _ with p (f x)
filter-map-def x xs f p () | false
filter-map-def x xs f p _ | true = -- not writing refl on purpose
begin
filter p (map f (x ∷ xs))
≡⟨ refl ⟩
f x ∷ filter p (map f xs)
∎
しかし、タイプチェッカーは私に同意しません。現在の目標は残っているように見えます.でfilter p (f x ∷ map f xs) | p (f x)
パターンマッチしたとしても.p (f x)
filter
f x ∷ filter p (map f xs)
これを機能させる方法はあり≡-Reasoning
ますか?
ありがとう!