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.

  • 相关阅读:
    线程的资源释放(一)
    iOS开发完整项目
    iOS开发多线程技术方案
    Windows 7 Beta泄漏版存在安全问题 狼人:
    工信部:黑客入侵等是网络安全防护工作的重点 狼人:
    微软下周2将发布13个补丁 修复26个安全漏洞 狼人:
    调查显示互联网14%SSL认证不安全 狼人:
    09年恶意软件放缓 2010年共享最危险 狼人:
    IE曝新安全漏洞 千万网民隐私遭受威胁 狼人:
    安全关注:2009年信息安全八大预测 狼人:
  • 原文地址:https://www.cnblogs.com/Answer1215/p/6215482.html
Copyright © 2011-2022 走看看