zoukankan      html  css  js  c++  java
  • [Functional Programming] Combine Multiple State ADT Instances with the Same Input (converge(liftA2(constant)))

    When combining multiple State ADT instances that depend on the same input, using chain can become quite burdensome. We end up having to play leapfrog with values inside of nested chain statements. Very reminiscent of the old callback nastiness we had to deal with before Promisesgraced our existence.

    But fear not, but taking advantage of the Applicative Functor portion of the State ADT in combination with the converge combinator, we can apply these types of transitions in unison, passing the value to both virtually simultaneously.

    For example:

    // () -> b
    const validateAnswer = converge(
        liftA2(equals),
        cardToHint,
        getHint
    )

    'validateAnswer' return a boolean value.

    We want to take this value, and pass to two functions 'setIsCorrect' & 'updateRank'

    // b -> ()
    const setIsCorrect = b => over('isCorrect', constant(b));
    
    // b -> ()
    const updateRank = b => over('rank', adjustRank(b));

    One functional way to do this is using 'converge':

    const applyFeedback = converge(
        liftA2(constant),
        setIsCorrect,
        updateRank
    )

    ----------------

    const {prop, State, omit, curry, converge,map, composeK, liftA2, equals, constant,option, chain, mapProps, find, propEq, isNumber, compose, safe} = require('crocks');
    const  {get, modify, of} = State; 
    
    const state = {
        cards: [
            {id: 'green-square', color: 'green', shape: 'square'},
            {id: 'orange-square', color: 'orange', shape: 'square'},
            {id: 'blue-triangle', color: 'blue', shape: 'triangle'}
        ],
        hint: {
            color: 'green',
            shape: 'square'
        },
        isCorrect: null,
        rank: 2
    }
    
    const inc = x => x + 1;
    const dec = x => x - 1;
    const incOrDec = b => b ? dec :  inc;
    const clamp = (min, max) => x => Math.min(Math.max(min, x), max);
    const clampAfter = curry((min, max, fn) => compose(clamp(min, max), fn));
    const limitRank = clampAfter(0, 4);
    const over = (key, fn) => modify(mapProps({[key]: fn}))
    const getState = key => get(prop(key));
    const liftState = fn => compose(
        of,
        fn
    )
    const getCard = id => getState('cards')
        .map(chain(find(propEq('id', id))))
        .map(option({}))
    const getHint = () => getState('hint')
        .map(option({}))
    const cardToHint = composeK(
        liftState(omit(['id'])),
        getCard
    )
    // () -> b
    const validateAnswer = converge(
        liftA2(equals),
        cardToHint,
        getHint
    )
    // b -> ()
    const setIsCorrect = b => over('isCorrect', constant(b));
    
    const adjustRank = compose(limitRank, incOrDec);
    // b -> ()
    const updateRank = b => over('rank', adjustRank(b));
    
    const applyFeedback = converge(
        liftA2(constant),
        setIsCorrect,
        updateRank
    )
    
    const feedback = composeK(
        applyFeedback,
        validateAnswer
    )
    
    console.log(
        feedback('green-square')
            .execWith(state)
    )
  • 相关阅读:
    mysql常用时间列表的查询——七天内、本月、本周、某天
    js取数组获取最大值的四种方式
    Mysql按周,按月,按日,按小时分组统计数据
    Mysql中的sum函数为null时的解决办法
    Maven 中 dependencyManagement 标签使用
    嵌入式printf函数运用基于ti 16位MCU
    md5实现对用户名与密码的保护
    c# md5 算法实现
    define a struct
    file.open异常处理
  • 原文地址:https://www.cnblogs.com/Answer1215/p/10268647.html
Copyright © 2011-2022 走看看