Monads are a design pattern in functional programming. Monads became more known due to their use in Haskell; however, there is nothing Haskell-specific in monads. I think that monads are simply a trick in lambda-calculus, a trick that turned out to be a useful abstraction. I would say that this is perhaps useful in any functional language (it is not so clear to me whether the lazy-evaluation policy of Haskell has any significance for monadic programming).
There is a proliferation of \C In the monadic programming, this functor is Called "lift" rather than "map". Let us denote the type "lists of type to" by the notation "M (a)", emphasizing That We Could Easily Be just as talking about trees with nodes of type a, or Any Other structure made out of the type "a" in Some Way.
The type of "lift" is
Then lift: (a-> b) -> (M (a) -> M (b))
functions ie it maps from a-> b to functions M (a) -> M (b). If
we want to implement "lift" for lists, Could we just write an iterative Implementation, going-through the list. In monadic programming, we do Not do this iteration in "lift", pero INSTEAD we define "lift" through two OTHER FUNCTIONS: "bind" and "unit." When we define Them, we will have defined the \t instead \nd dog we define the "monadic composition" of f1 and f2 as
(bind f2). f1: a-> M (c)
However, we can Also consider "bind" That as an operation contains a separate "flattening" step. Then it is Useful to look at the "flattening" operation separately. The "flattening" operation is Called "join" in the parlance monadic:
join: M (M (x)) -> M (x)
and is, in my view, the core idea of the monad. This idea Can Be Formulated as: there is no reason to enclose x Twice in the "M" layer. A nested "M" container is equivalent to a single "M" container. An object of type "M (M (a))" Can Be Reduced, But standard in a nontrivial way, to an object of type "M (a)." Reduction This is what "join" does. Now suppose we
or"Lift (const x)" is Not Necessarily a constant function! So we not could define
unit x = lift (\\ i -> x) --- Wrong! So
the Functionality of "unit" is not a subset of the Functionality of "lift". We always Need to define "unit" separately.
The lifting of functions is defined Not Necessarily Through Merely lifting of values. So "lift" not can be defined only through "unit." If we know how to make a one-element list out of "x" Not yet we do know how to iterate through to Longer list. The iteration algorithm is hidden in the "bind" function. For this reason, "lift" is definable through "unit" and "bind."
We Can not define "bind" or "lift" through "join" and "unit" alone. "Join" flattens lists, "unit" makes a single-element list [x] from "x", BUT Neither "unit" nor "join" can make a multi-element list out of "x".
Notes:
1. In Our example, "M (a)" Was a list of values of type "a". However, the Same Rules Apply to Other algebraic structures. For example, "M (a)" Could be a tree containing values of type "a" in STI nodes. Or "M (a)" Could be a dictionary with values of type "a".
Or, more simply, "M (a)" Could be a pair [a, x] WHERE X is always a number. Let us look at this last example in more detail.
We define the type "M (a)" as a two-element list [a, x] containing a value of type "a" and a number x. The "unit" operation Needs to extend a value of type "a" into a pair [a, x]. Let us define "unit" by
ad denote this function by the symbol "monadic_composition.) Then We Can compute the composition of These functions: We Have
first f1 a = [sqrt (a), 1] Then we compute
(monadic_composition f1 f2) a = (bind f2) [sqrt (a), 1]
= [cos (sqrt (a)), 1 +2]
= [cos (sqrt (a)), 3]. Thus, the "monadic composition" of f1 and f2 will add the counters. The "total cost" is computed Correctly.
We Could use this kind of monad for Representing functions Whose evaluation "costs" Something. Then the counter on the final value will Be The total cost. Merely We Need to define the functions and Their costs. The monadic code with Computations cleanly Separates the counters, the initial values orf the counters, and the computation of functions themselves.
In this example, we can see in what way the \f type \ould be * equal * to the result of lifting "x" once.
join. unit. unit == unit - On Any functions as type "to" In the example
Above, we Could not Have defined "unit" by
unit a = [a, 1] - Wrong! Because
then (join. Unit. Unit) x = [x, 2] while unit x = [x, 1].
b) If We Have An object of type M (M (a)), and we flatten it, Then we get an object of type M (a), if we now lift it with "unit" Should we get again The Same object as before.
unit. join. unit == unit - as functions on type M (a)
c) There Are, However, Also some "monadic law" concerning "lift" / "bind." This law Seems To Be The expression of the fact-Lifted That the composition of functions is equal to thelift of the composition of the functions. (How can this fail to hold? Or maybe I'm Mistaken here somewhere?)
(lift f). (Lift g) = Lift (f. G)
Now, "lift" can Be defined through "bind" and "unit." Then We Can substitute this definition Above Into the formula and get some identity That contains only "bind" and "unit." If "bind" does not Satisfy That identity, Then the "lift" defined Through It Will Be nonsensical.
I can not find a simple way of understanding That identity, or of writing it in a reasonable way.
3. It is important to note That There is no function for extracting the value of type "a" from a value of type "M (a)." Once Lifted Some value is to the monad, There is no way back. Values Lifted Can Be Used as argumentLifted s to functions or to functions of type a-> M (a) through "monadic composition" (ie "bind). The result is always a value of type M (a).
Conclusions:
- Monadic programming is a trick in lambda-calculus That Consists of Replacing the "lift" functor by "monadic composition".
- Monadic programming has nothing to do with input / output or imperative programming. It is just a Different Way of Structuring code independent Into parts.
- In order to define a monad, one Needs to define "unit" and "bind" Alternatively or "unit", "join" and "lift" That Need to Satisfy Certain Laws.
- Once " unit "and" bind "are defined, we can define" join "and" lift "through Them. Once" join "and" lift "are defined, we can
0 comments:
Post a Comment