zoukankan      html  css  js  c++  java
  • [Compose] 20. Principled type conversions with Natural Transformations

    Natural Transformations, let's explain it by using a coding example, for example, we have a 'Either' holding a value 'a' and we want to transform a' Task' that holding an 'a'.

    // Either(a) -> Task(a)

    Let's start coding:

    const Either = require('data.either');
    const {Right, Left, fromNullable} = Either;
    const Task = require('data.task');
    
    const eitherToTask = e =>
        e.fold(Task.rejected, Task.of);
    
    eitherToTask(Right('Good')).fork(
        e => console.error('err', e),
        x => console.log('res', a)
    ) // 'res' Good

    Let's go thought:

    const eitherToTask = e => 
        e.fold(Task.rejected, Task.of);

    'Either' has 'fold' method (left, right), as we know 'fold' will unbox the contianer just return the value, so that it will unbox the 'Either' type, and we wrap the return value with 'Task'. So now, 'eitherToTask' return a 'Task'

    eitherToTask(Right('Good')).fork(...)

    The result we using 'Right' instead of 'Left' we will explain later, but here, we know we have a 'Task', therefore we can call 'fork' to trigger the side effect.

    Law:

    natural_transform(Functor).map(function) === natural_transform(Functor.map(function))

    On the left side of equals, we have natural transform function wraps a Functor, then map to a function.

    On the right side, we have a natural transform function wrap functor that map to a funciton.

    Let's see an example:

    const Box = x => ({
        map: f => Box(f(x)),
        fold: f => f(x)
    })
    
    const boxToEither = b => 
        b.fold(Right);
    
    const res1 = boxToEither(Box(100)).map(x => x * 2);
    console.log(res1);  // Either (200)
    const res2 = boxToEither(Box(100).map(x => x * 2));
    console.log(res2);  // Either (200)

    What if we using 'Left' instead of 'Right':

    const Box = x => ({
        map: f => Box(f(x)),
        fold: f => f(x)
    })
    
    const boxToEither = b => 
        b.fold(Left);
    
    const res1 = boxToEither(Box(100)).map(x => x * 2);
    console.log(res1);  // Either(100)
    const res2 = boxToEither(Box(100).map(x => x * 2));
    console.log(res2);  // Either(200)

    As we can see 'res1' is no longer equals to 'res2', because Left will ingore mapping function. 

    Another example: List -> Either:

    // first :: [] -> Either
    const first = xs => fromNullable(xs[0]);
    const res3 = first([1,2,3]).map(x => x +1);
    const res4 = first([1,2,3].map(x => x +1));
     console.log(res3);  // Either(2)
     console.log(res4); // Either(2)

    These two shall be equal and they are. Any function that satisfies this equation is a natural transformation. Let's look at this on the board here.

     

    If we have some F(a) and some functor holding an a and we map(f) over it, it transforms that a to a b. we're just mapping a function from the type a to some type b here all inside our functor f. Then we run a natural transformation we'll have a G(b).

    If we take the other path moving downward we'll first naturally transform our functor holding an a into the G(a) here, and then we map(f) over that to get a G(b). We end up with the same result. This can be quite useful.

  • 相关阅读:
    南阳理工ACM(题目56)
    南阳理工ACM(题目56)
    南阳理工ACM(题目56)
    csuoj1009
    素数槽csuoj
    简单动态规划问题分析
    sort函数使用的基本知识
    2014年7月19日——比赛题取石头问题1
    CODEVS——T 1269 匈牙利游戏 2012年CCC加拿大高中生信息学奥赛
    洛谷—— P1640 [SCOI2010]连续攻击游戏
  • 原文地址:https://www.cnblogs.com/Answer1215/p/6215482.html
Copyright © 2011-2022 走看看