[Functional Programming] Randomly Pull an Item from an Array with the State ADT (Pair)
Functor
composition is a powerful concept that arises when we have one Functor
nested in another Functor
. It becomes even more powerful when both of those are Chain
s, allowing us to apply each Functor
’s special properties and effects for a given computation.
We witness this power first hand by combining our beloved State
with a Pair
type mixing the ability to randomly pick an index from an Array
using State
and use it to draw an element from the Array
. We will build up to an easy to use interface that allows us to pull as many elements as we need, by chaining on single State transaction.
- // getDeck :: () -> State AppState Deck
- const getDeck = () => generateCards()
- .map(xs => Pair([], xs));
- /**
- Pair(
- [ ],
- [ { id: "orange-square", color: "orange", shape: "square" }, { id: "orange-triangle", color: "orange", shape: "triangle" }, { id: "orange-circle", color: "orange", shape:
- "circle" }, { id: "green-square", color: "green", shape: "square" }, { id: "green-triangle", color: "green", shape: "triangle" }, { id: "green-circle", color: "green", shape: "circle" }, { id: "blue-square", color: "blue", shape: "square" }, { id: "blue-triangle", color: "blue", shape: "triangle" }, { id: "blue-circle", color: "blue", shape: "circle" }, { id: "yellow-square", color: "yellow", shape: "square" }, { id: "yellow-triangle", color: "yellow", shape: "triangle" }, { id: "yellow-circle", color: "yellow", shape: "circle" } ]
- )
- */
- // draw :: Integer -> Deck -> Deck
- const draw = compose(
- chain,
- drawCardAt
- );
- // const draw = index => deck => deck.chain(drawCardAt(index))
- // drawRandom :: Deck -> State AppState Deck
- // From the right side pair, get a random index, then draw the card by index
- const drawRandom = converge(
- liftA2(draw),
- compose(
- randomIndex,
- snd
- ),
- liftState(identity)
- )
- console.log(
- getDeck()
- .chain(drawRandom)
- .chain(drawRandom)
- .chain(drawRandom)
- .evalWith(state).fst()
- )
- /**
- * [ { id: 'orange-square', color: 'orange', shape: 'square' },
- { id: 'blue-triangle', color: 'blue', shape: 'triangle' },
- { id: 'blue-square', color: 'blue', shape: 'square' } ]
- */
- console.log(
- getDeck()
- .chain(drawRandom)
- .chain(drawRandom)
- .chain(drawRandom)
- .evalWith(state).snd()
- )
- /**
- [ { id: 'orange-triangle', color: 'orange', shape: 'triangle' },
- { id: 'orange-circle', color: 'orange', shape: 'circle' },
- { id: 'green-square', color: 'green', shape: 'square' },
- { id: 'green-triangle', color: 'green', shape: 'triangle' },
- { id: 'green-circle', color: 'green', shape: 'circle' },
- { id: 'blue-circle', color: 'blue', shape: 'circle' },
- { id: 'yellow-square', color: 'yellow', shape: 'square' },
- { id: 'yellow-triangle', color: 'yellow', shape: 'triangle' },
- { id: 'yellow-circle', color: 'yellow', shape: 'circle' } ]
- */
------
- const {prop,assoc, Pair, pick, bimap, State, snd, identity, omit, curry, filter, fanout, converge,map, composeK, liftA2, equals, constant,option, chain, mapProps, find, propEq, isNumber, compose, safe} = require('crocks');
- const {get, modify, of} = State;
- const state = {
- colors: [ 'orange', 'green', 'blue', 'yellow' ],
- shapes: [ 'square', 'triangle', 'circle' ],
- seed: Date.now()
- };
- const liftState = (fn) => compose(
- of, fn
- );
- const getState = key => get(prop(key))
- // #region random
- // nextSeed :: Integer -> Integer
- const nextSeed = seed =>
- (seed * + ) & 0x7fffffff
- // value :: Integer -> Number
- const value = seed =>
- (seed >>> ) / 0x7fff
- // normalize :: (Integer, Integer) -> Number -> Integer
- const normalize = (min, max) =>
- x => Math.floor(x * (max - min)) + min
- // getNextSeed :: () -> State AppState Integer
- const getNextSeed = () =>
- get(({ seed }) => nextSeed(seed))
- // updateSeed :: Integer -> State AppState ()
- const updateSeed = seed =>
- modify(assoc('seed', seed))
- // nextValue :: Integer -> State AppState Number
- const nextValue = converge(
- liftA2(constant),
- liftState(value),
- updateSeed
- )
- // random :: () -> State AppState Number
- const random =
- composeK(nextValue, getNextSeed)
- // between :: (Integer, Integer) -> State AppState Integer
- const between = (min, max) =>
- random()
- .map(normalize(min, max));
- const randomIndex = xs => between(, xs.length);
- // #endregion
- // #region generate
- const getColors = () => getState('colors').map(option([]));
- const getShapes = () => getState('shapes').map(option([]));
- const buildCard = curry((color, shape) => ({
- id: `${color}-${shape}`,
- color,
- shape
- }));
- const buildCards = liftA2(buildCard);
- const generateCards = converge(
- liftA2(buildCards),
- getColors,
- getShapes
- );
- // #endregion
- // #region draw
- const getAt = index => array => Array.of(array[index]);
- const unsetAt = index => array => [
- ...array.slice(, index),
- ...array.slice(index + )
- ];
- const drawCardAt = index => fanout(getAt(index), unsetAt(index));
- // #endregion
- // getDeck :: () -> State AppState Deck
- const getDeck = () => generateCards()
- .map(xs => Pair([], xs));
- // draw :: Integer -> Deck -> Deck
- const draw = compose(
- chain,
- drawCardAt
- );
- // const draw = index => deck => deck.chain(drawCardAt(index))
- // drawRandom :: Deck -> State AppState Deck
- const drawRandom = converge(
- liftA2(draw),
- compose(
- randomIndex,
- snd
- ),
- liftState(identity)
- )
- console.log(
- getDeck()
- .chain(drawRandom)
- .chain(drawRandom)
- .chain(drawRandom)
- .chain(drawRandom)
- .evalWith(state).fst()
- )
[Functional Programming] Randomly Pull an Item from an Array with the State ADT (Pair)的更多相关文章
- Beginning Scala study note(4) Functional Programming in Scala
1. Functional programming treats computation as the evaluation of mathematical and avoids state and ...
- a primary example for Functional programming in javascript
background In pursuit of a real-world application, let’s say we need an e-commerce web applicationfo ...
- Functional programming
In computer science, functional programming is a programming paradigm, a style of building the struc ...
- Functional Programming without Lambda - Part 2 Lifting, Functor, Monad
Lifting Now, let's review map from another perspective. map :: (T -> R) -> [T] -> [R] accep ...
- Functional Programming without Lambda - Part 1 Functional Composition
Functions in Java Prior to the introduction of Lambda Expressions feature in version 8, Java had lon ...
- Java 中的函数式编程(Functional Programming):Lambda 初识
Java 8 发布带来的一个主要特性就是对函数式编程的支持. 而 Lambda 表达式就是一个新的并且很重要的一个概念. 它提供了一个简单并且很简洁的编码方式. 首先从几个简单的 Lambda 表达式 ...
- Functional programming idiom
A functional programming function is like a mathematical function, which produces an output that typ ...
- 关于函数式编程(Functional Programming)
初学函数式编程,相信很多程序员兄弟们对于这个名字熟悉又陌生.函数,对于程序员来说并不陌生,编程对于程序员来说也并不陌生,但是函数式编程语言(Functional Programming languag ...
- Functional Programming 资料收集
书籍: Functional Programming for Java Developers SICP(Structure and Interpretation of Computer Program ...
随机推荐
- SpringMVC - 个人对@ModelAttribute的见解 和 一些注入参数、返回数据的见解
2016-8-23修正. 因为对modelattribute这个注解不了解,所以在网上搜寻一些答案,感觉还是似懂非懂的,所以便自己测试,同时还结合网上别人的答案:最后得出我自己的见解和结果,不知道正确 ...
- 【Android开发日记】之入门篇(十三)——Android的控件解析
Android的控件都派生自android.view.View类,在android.widget包中定义了大量的系统控件供开发者使用,开发者也可以从View类及其子类中,派生出自定义的控件. 一.An ...
- javascript三种嵌入方式
什么是JavaScript? JavaScript是运行在浏览器端的脚步语言,JavaScript主要解决的是前端与用户交互的问题,包括使用交互与数据交互,JavaScript是浏览器解释执行的. J ...
- 前段基础HTML
HTML介绍 Web服务本质 import socket sk = socket.socket() sk.bind(("127.0.0.1", 8080)) sk.listen(5 ...
- 转:Laravel 安装指南
Git 介绍 之所以要说 Git,就是因为 Composre 有时需要用到 Git,还是安装上比较好,Composer 暂且不表,先来了解一下 Git 吧(已经安装的童鞋跳过这里,直接看 Compos ...
- HDU 1754.I Hate It-结构体版线段树(单点更新+区间查询最值)
I Hate It Time Limit: 9000/3000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total S ...
- android9.0请求异常
抛出异常:CLEARTEXT communication to not permitted by network security policy android 9开始默认不能使用http 1.在re ...
- 一个排好序的数组,找出两数之和为x的所有组合【双指针】
#include <bits/stdc++.h> using namespace std; const int N = 1e6,INF = 0x3f3f3f3f; int a[N]; in ...
- 简单DP【p3399】丝绸之路
Background 张骞于公元前138年曾历尽艰险出使过西域.加强了汉朝与西域各国的友好往来.从那以后,一队队骆驼商队在这漫长的商贸大道上行进,他们越过崇山峻岭,将中国的先进技术带向中亚.西亚和欧洲 ...
- mysql 列转行,合并字段的方法
数据表(表名:xsk) +----+------+-----------+-------+ | id | name| course | score | +----+------+----------- ...