Что такое каррирование, как оно работает и зачем нужно? Что такое мемоизация?
Каррирование - это приём в функциональном программировании, позволяющий преобразовать функцию, заменив её несколько первых аргументов константными значениями, тем самым создав новую функцию с меньшим количеством аргументов на основе старой. Этот будет удобно применять в случае, когда первые несколько аргументов функции заранее известны, и указывать их при каждом вызове нет необходимости.
function sum(a, b) {
return a + b;
}
function sayTwoWords(word1, word2) {
console.log(word1 + " " + word2);
}
function curry(func, a) {
return function (b) {
return func(a, b);
};
}
var sayGoodbyeTo = curry(sayTwoWords, "Goodbye");
sayGoodbyeTo("Mary"); // "Goodbye Mary"
var minusThree = curry(sum, -3);
console.log(minusThree(13)); // 10
В JS функция при вызове может принимать произвольное количество аргументов, вне зависимости от того, сколько их было указанно при объявлении. Пользуясь этой возможностью, объявим куррирующую функцию с одним параметром func, в который будет передаваться каррируемая функция. Остальные же параметры будут значениями, на которые будут заменяться первые аргументы каррируемой функции. Эти значения мы будем извлекать из объекта arguments, в котором содержатся все значения аргументов функции, и записывать в массив. В возвращаемой функции нам необходимо вызвать каррируемую функцию, подставив в качестве первых аргументов значения из массива curryArgs, а в качестве остальных — значения, переданные в качестве параметров возвращаемой функции. Для этого воспользуемся методом apply, который позволяет вызвать функцию со значениями аргументов, которые переданы ему в массиве. Перед тем, как вызвать метод apply от каррируемой функции, преобразуем в массив объект arguments возвращаемой функции и склеим, при помощи метода concat, массив curryArgs c полученным массивом. Результат этой операции и передадим в качестве второго аргумента методу apply. В итоге функция curry примет вид:
function curry(func) {
var args = arguments, curryArgs = [];
for (var i = 1; i < args.length; i++) {
curryArgs[i - 1] = args[i];
}
return function () {
var argsArr = Array.prototype.slice.call(arguments, 0);
curryArgs = curryArgs.concat(argsArr);
return func.apply(this, curryArgs);
}
}
В ES2015 функцию каррирования можно записать короче:
function curry( func, ...curryArgs )
{
return ( ...rest ) =>
func( ...curryArgs, ...rest );
}
Примеры:
function multiply( x, y )
{
return x * y;
}
const double = curry( multiply, 2 );
double( 2 ); // 4
const double4 = ( x ) => multiply( x, 2 );
const double3 = ( y ) => multiply( 2, y );
Иногда наши рекурсивные или часто вызываемые функции очень сильно «тормозят» выполнение программы, потому что результаты промежуточных вычислений нигде не сохраняются, а вычисляются заново. Мемоизация — сохранение результатов выполнения функций для предотвращения повторных вычислений. Перед вызовом функции проверяется, вызывалась ли функция ранее: если не вызывалась, функция вызывается и результат её выполнения сохраняется; если вызывалась, используется сохранённый результат. Классическим примером для иллюстрации является функция, вычисляющая последовательность чисел Фибоначчи:
function fibonacci(n) {
if (n === 0 || n === 1)
return n;
return fibonacci(n - 1) + fibonacci(n - 2);
}
Для больших чисел n количество вызовов функции растёт очень быстро. Уже для n=50 это порядка 40 миллиардов вызовов. Используя идею мемоизации, то есть кеширования промежуточного результата, можно добиться уменьшения количества вызовов для n=50 до всего лишь 99:
var fibonacci = (function() {
var memo = {};
function f(n) {
var value;
if (n in memo) {
value = memo[n];
}
else {
if (n === 0 || n === 1)
value = n;
else
value = f(n - 1) + f(n - 2);
memo[n] = value;
}
return value;
}
return f;
})();
Работа с объектом типа Date.
Для работы с датой и временем в JavaScript используются объекты Date.
Создание
Для создания нового объекта типа Date используется один из синтаксисов:
new Date()
Создает объект Date с текущей датой и временем:
var now = new Date();
new Date(milliseconds)
Создает объект Date, значение которого равно количеству миллисекунд (1/1000 секунды), прошедших с 1 января 1970 года GMT+0.
var Jan02_1970 = new Date(3600 * 24 * 1000);
new Date(datestring)
Если единственный аргумент – строка, используется вызов Date.parse для чтения даты из неё. Метод Date.parse(str) разбирает строку str в формате YYYY-MM-DDTHH:mm:ss.sssZ и возвращает соответствующее ей количество миллисекунд. Если это невозможно, Date.parse возвращает NaN. Например:
var msUTC = Date.parse('2012-01-26T13:51:50.417Z'); // зона UTC
alert( msUTC ); // 1327571510417 (число миллисекунд)
С таймзоной -07:00 GMT:
var ms = Date.parse('2012-01-26T13:51:50.417-07:00');
alert( ms ); // 1327611110417 (число миллисекунд)
new Date(year, month, date, hours, minutes, seconds, ms)
Дату можно создать, используя компоненты в местной временной зоне. Для этого формата обязательны только первые два аргумента. Отсутствующие параметры, начиная с hours считаются равными нулю, а date – единице. Год year должен быть из 4 цифр. Отсчет месяцев month начинается с нуля 0.
new Date(2011, 0, 1); // 1 января 2011, 00:00:00
Получение компонентов даты
getFullYear() - получить год (из 4 цифр)
getMonth() - получить месяц, от 0 до 11.
getDate() - получить число месяца, от 1 до 31.
getHours(), getMinutes(), getSeconds(), getMilliseconds() - получить соответствующие компоненты.
getDay() - получить номер дня в неделе. Неделя в JavaScript начинается с воскресенья, так что результат будет числом от 0(воскресенье) до 6(суббота).
Все методы, указанные выше, возвращают результат для местной временной зоны. Существуют также UTC-варианты этих методов, возвращающие день, месяц, год и т.п. для зоны GMT+0 (UTC): getUTCFullYear(), getUTCMonth(), getUTCDay().
Кроме описанных выше, существуют два специальных метода без UTC-варианта:
getTime() - возвращает число миллисекунд, прошедших с 1 января 1970 года GMT+0.
getTimezoneOffset() - возвращает разницу между местным и UTC-временем, в минутах.
Установка компонентов даты
Следующие методы позволяют устанавливать компоненты даты и времени:
ü setFullYear(year[, month, date])
ü setMonth(month[, date])
ü setDate(date)
ü setHours(hour[, min, sec, ms])
ü setMinutes(min[, sec, ms])
ü setSeconds(sec[, ms])
ü setMilliseconds(ms)
ü setTime(milliseconds) (устанавливает всю дату по миллисекундам с 01.01.1970 UTC)
Все они, кроме setTime(), обладают также UTC-вариантом, например: setUTCHours().
Автоисправление даты
Автоисправление – очень удобное свойство объектов Date. Оно заключается в том, что можно устанавливать заведомо некорректные компоненты (например 32 января), а объект сам себя поправит.
var d = new Date(2013, 0, 32); // 32 января 2013 ?!?
alert(d); // это 1 февраля 2013!