書籍「すごいHaskellたのしく学ぼう!」を現在学習中です。
(+) <$> (*2) <*> (+10)
第14章に上記のコードが出てきましたが、最終的にどんな動作をする関数になるのか分かりません。
実行してみると、引数を3倍した値?に10を足したような結果が出てきます。
1引数関数同士をfmapすると関数合成と同じになるので(+) <$> (*2)
が(+) ((*2) x)
になるのは理解できますが、
これに対して<*> (+10)
を適用するとどんな風になるのでしょうか。
<*>
は以下のように「ファンクターに包まれた関数を取り出して、ファンクター値に適用する」という風に理解をしてしまっています。
(+) <$> Just 10 <*> Just 20
以下のような質問にはグッドを送りましょう
- 質問内容が明確
- 自分も答えを知りたい
- 質問者以外のユーザにも役立つ
グッドが多くついた質問は、TOPページの「注目」タブのフィードに表示されやすくなります。
気になる質問をクリップする
クリップした質問は、後からいつでもマイページで確認できます。
またクリップした質問に回答があった際、通知やメールを受け取ることができます。
下記のような質問は推奨されていません。
- 間違っている
- 質問になっていない投稿
- スパムや攻撃的な表現を用いた投稿
適切な質問に修正を依頼しましょう。
回答2件
2
ベストアンサー
結論からいうと次のようになります。
haskell
1(+) <$> (* 2) <*> (+ 10) ≡ \x -> (x * 2) + (x + 10)
ここからはなぜそうなるのかを見ていきます。
まず (<$>)
と (<*>)
の定義を見ましょう。
haskell
1-- https://hackage.haskell.org/package/base-4.14.0.0/docs/src/GHC.Base.html#line-973 2instance Applicative ((->) r) where 3 … 4 (<*>) f g x = f x (g x) 5 … 6infixl 4 <*> 7 8-- https://hackage.haskell.org/package/base-4.14.0.0/docs/src/Data.Functor.html#%3C%24%3E 9(<$>) :: Functor f => (a -> b) -> f a -> f b 10(<$>) = fmap 11infixl 4 $> 12 13-- https://hackage.haskell.org/package/base-4.14.0.0/docs/src/GHC.Base.html#line-973 14instance Functor ((->) r) where 15 fmap = (.)
この定義に沿って式変形をしていきます。
haskell
1 (+) <$> (* 2) <*> (+ 10) 2≡〈優先順位の括弧を明記〉-- (1) 3 ((+) <$> (* 2)) <*> (+ 10) 4≡〈(<*>) の定義を代入〉 -- (2) 5 \x -> ((+) <$> (* 2)) x ((+ 10) x) 6≡〈中置記法に変換〉 -- (3) 7 \x -> ((+) <$> (* 2)) x (x + 10) 8≡〈(<$>) の定義を代入〉 9 \x -> ((+) . (* 2)) x (x + 10) 10≡〈(.) の定義を代入〉 11 \x -> (\y -> (+) ((* 2) y)) x (x + 10) 12≡〈中置記法に変換〉 13 \x -> (\y -> (+) (y * 2)) x (x + 10) 14≡〈関数適用〉 15 \x -> ((+) (x * 2)) (x + 10) 16≡〈中置記法に変換〉 17 \x -> (x * 2) + (x + 10)
(<$>)
と(<*>)
はともに同じ優先順位の左結合です。(<*>) f g x = f x (g x)
は(<*>) = \f -> \g -> \x -> f x (g x)
と同じです。(+ 10) x
がx + 10
になります。
定義に沿ってひとつずつ変形してくと理解が深まると思います。
投稿2020/10/25 09:03
総合スコア285
1
@kakkun61 さんの回答に補足します。
要約すると関数->
における<*>
は、同じ引数を共有しながら合成する演算子となっています。
例えば<*>
の定義
instance Applicative ((->) r) where … (<*>) f g x = f x (g x)
においても、x
をf
とg
両方に渡しながら(共有させながら)f
とg
を合成していますよね?
そしてラムダ式に変換した、
(+) <$> (* 2) <*> (+ 10) ≡ \x -> (x * 2) + (x + 10)
でも、やはりx
を(* 2)
と(+ 10)
という二つの関数で共有しています。
投稿2020/10/25 10:47
編集2020/11/11 09:48総合スコア423
良いと思った回答にはグッドを送りましょう。
グッドが多くついた回答ほどページの上位に表示されるので、他の人が素晴らしい回答を見つけやすくなります。
下記のような回答は推奨されていません。
- 間違っている回答
- 質問の回答になっていない投稿
- スパムや攻撃的な表現を用いた投稿
このような回答には修正を依頼しましょう。
回答へのコメント
関連した質問
Q&A
解決済
TypeError: a bytes-like object is required, not 'str'
回答1
クリップ0
更新
2018/06/18
Q&A
解決済
selenium データスクレイピングエラー
回答1
クリップ0
更新
2022/11/16
Q&A
解決済
math domain errorとエラーが出てくる
回答1
クリップ0
更新
2019/12/16
Q&A
解決済
データベースに接続できていない?SQLSTATE[HY000] [2002] No such file or directory
回答1
クリップ0
更新
2021/07/05
Q&A
解決済
eclipseでGradleプロジェクト作成後にbuildするとエラーが出る
回答1
クリップ2
更新
2014/11/18
Q&A
解決済
pythonコードでif文がエラーになる理由を教えて下さい
回答2
クリップ0
更新
2017/01/16
Q&A
解決済
Haskell の "(>>=) Just" の型の意味を教えてください
回答1
クリップ0
更新
2023/01/28
Q&A
解決済
HTMLタグにwidthが設定されている
回答1
クリップ0
更新
2023/02/02
同じタグがついた質問を見る
Haskellは高い機能性をもった関数型プログラミング言語で、他の手続き型プログラミング言語では難しいとされている関数でも容易に行うことができます。強い静的型付け、遅延評価などに対応しています。
良いと思った回答にはグッドを送りましょう。
グッドが多くついた回答ほどページの上位に表示されるので、他の人が素晴らしい回答を見つけやすくなります。
下記のような回答は推奨されていません。
このような回答には修正を依頼しましょう。
2020/10/26 04:30