functional-concepts v0.0.1
functional-concepts
Библиотека с набором функций для возможности использования некоторых приемов функционального программирования.
В состав входят следующие функции:
composecurryflipismemoizepartialpipeconcateachfilterfindfindIndexlastmapreducereduceRightreversesortuniquezip
А также их каррированные варианты.
Установка
NPM:
npm install functional-conceptsИспользование
// классические варианты функций
import { /* имена функций */ } from 'functional-concepts'
// каррированные варианты функций
import { /* имена функций */ } from 'functional-concepts/curried'Примеры использования
В дальнейших примерах будем оперировать массивом объектов описывающих машины. Каждая машина представлена объектом следующего вида:
{
name: 'Aston Martin One-77',
horsePower: 750,
maxSpeed: 220,
price: 1850000,
inStock: true
}Каррирование
Функция curry(f, arity = f.length) преобразует функцию в набор функций с единственным аргументом.
Рассмотрим вариант функции без каррирования:
// функция с двумя аргументами
const map = (f, array) => array.map(f)
// деталь
const getPrices = cars => map(car => car.price, cars) // не очень удобно создавать детали - над именами аргументов приходится подуматьПрименим каррирование:
// каррированная функция
const map = curry((f, array) => array.map(f))
// деталь
const getPrices = map(car => car.price) // уже лучшеМожем использовать бесточечный стиль (point-free style):
// вспомогательная функция
const prop = curry((name, obj) => obj[name]) // можно переиспользовать
// селектор
const selectPrice = prop('price') // можно переиспользовать
// деталь
const getPrices = map(selectPrice) // бесточечный стиль, выглядит кратко и лаконичноПример использования:
console.log(getPrices(getCars()))Несмотря на то, что в классическом понимании каррирование преобразует функцию в набор функций с единственным параметром, на практике реализации каррирования могут принимать несколько аргументов за раз:
const sum = (x, y, z) => x + y + z
const curriedSum = curry(sum)
curriedSum(8, 13)(21) // 42
curriedSum(8)(13, 21) // 42
curriedSum(8, 13, 21) // 42Частичное применение
Функция partial(f, ...args) преобразует функцию с некоторым числом аргументов в функцию с меньшим числом аргументов.
// функции с двумя аргументами
const map = (f, array) => array.map(f)
const prop = (name, obj) => obj[name]
// частичное применение
const selectPrice = partial(prop, 'price')
const getPrices = partial(map, selectPrice)
console.log(getPrices(getCars())) // пример использованияВ отличии от каррированной функции, в функцию с частичным применением нужно передать все недостающие параметры:
const sum = (x, y, z) => x + y + z
const partialSum = partial(sum, 8)
partialSum(13, 21) // 42 т.к. выполнится sum(8, 13, 21)
partialSum(13) // NaN т.к. выполнится sum(8, 13, undefined)Композиция
Функция compose(...fs) возвращает функцию, аргумент которой последовательно проходит справа налево:
Определим функции с которыми будем работать в примере:
const add = (x, y) => x + y
const map = (f, array) => array.map(f)
const reduce = (f, array) => array.reduce(f)
// вспомогательная функция, вычисляющая среднее арифметическое из значений в массиве
const average = xs => reduce(add, xs) / xs.lengthБез композиции деталь была бы такой:
// деталь
const getAveragePrice = cars => average(map(car => car.price, cars))Используем композицию:
// деталь
const getAveragePrice = compose(average, map(car => car.price)) // уже лучшеВ бесточечном стиле:
// вспомогательная функция
const prop = name => obj => obj[name] // можно переиспользовать
// селектор
const selectPrice = prop('price') // можно переиспользовать
// деталь в бесточечном стиле
const getAveragePrice = compose(average, map(selectPrice)) // еще лучшеПример использования:
console.log(getAveragePrice(getCars()))Конвейер
Функция pipe(...fs) возвращает функцию аргумент которой, последовательно проходит слева направо:
const sort = xs => [...xs].sort()
const last = xs => xs[xs.length - 1]
const selectMaxSpeed = prop('maxSpeed')
// деталь
const getFastestCar = pipe(map(selectMaxSpeed), sort, last) // конвейер
console.log(getFastestCar(getCars())) // пример использованияПорядок выполнения функций в конвейере отличается от порядка выполнения функций в композиции:
// оригинальная цепочка вызовов
three(two(one(x)))
// более естественно с точки зрения чтения
pipe(one, two, three)(x)
// более естественно с точки зрения записи
compose(three, two, one)(x)Мемоизация
Это полезный приём, при котором функция кеширует результаты своего вызова, чтобы не выполнять повторяющиеся вычисления.
Функция memoize(f, cache = new Map) возвращает функцию кеширующую результаты выполнения оригинальной функции.
Пример без мемоизации:
const f = x => x * x
f(2) // вычислить
f(2) // вычислить
f(2) // вычислитьС мемоизацией:
const f = memoize(x => x * x)
f(2) // вычислить
f(2) // взять из кеша
f(2) // взять из кешаПроверка типа
Функция is(type, x) - это предикат, проверяющий принадлежность своего аргумента x к определенному классу type.
const elements = [...document.body.children].filter(is(Element))Изменение порядка аргументов
Функция flip(f) меняет порядок аргументов. Иногда это бывает полезно.
const pipe = flip(compose)Тестирование при разработке
npm testЛицензия
MIT