JS Polyfills - Part 6 Curry, Memoise and Generator Functions
Github Code: JS Polyfills
16. Curry
Currying: Transformation of function with multiple parameters to a sequence of single parameters is Currying. Currying just transforms the function, it will not call the function. Partial functions can be easily generated using currying.
Function:
Curry(fn)
Description: A functional programming technique for transforming multiple parameter function to multiple single parameter functions.
Code Sandbox: Curry
//A functional programming technique for transforming multiple parameter function to a multiple single parameter functions.
//In this example, the curry function takes a function fn as an argument and returns a new curried function curried. The curried function takes a variable number of arguments, stored in the args array. If the number of arguments passed to curried is equal to or greater than the number of arguments expected by fn, fn is called with the apply method, passing in the args array. If the number of arguments passed to curried is less than the number of arguments expected by fn, the curried function returns a new function that takes the remaining arguments and concatenates them to the args array. This allows us to partially apply the arguments to the function and return a new function that takes the remaining arguments. The returned function can be called multiple times until all the arguments have been provided, at which point fn is finally executed.
function curry(fn) {
return function curried(...args) {
if (args.length >= fn.length) {
return fn.apply(this, args);
}
return function (...args2) {
return curried.apply(this, args.concat(args2));
};
};
}
// Usage example
function add(a, b, c) {
return a + b + c;
}
const curriedAdd = curry(add);
const add5 = curriedAdd(5);
console.log(add5(10)(15));
// Output: 30
17. Memoise
Memoization: A functional programming optimization technique with a top-down, depth-first approach for caching/storing previous value/computation so that the computed value is never executed again when necessary.
Function:
Memoize(fn)
Description: Memoize function takes another function fn as its argument and returns a new function that wraps fn
Code Sandbox: Memoize
//This function takes another function fn as its argument and returns a new function that wraps fn. The returned function caches the results of previous calls to fn using a simple JavaScript object cache. The cache key is generated by JSON.stringifying the arguments passed to the function. If a matching key is found in the cache, the cached result is returned. Otherwise, fn is called and its result is stored in the cache before being returned.
function memoize(fn) {
let cache = {};
return function (...args) {
let key = JSON.stringify(args);
if (cache[key]) {
return cache[key];
}
let result = fn.apply(this, args);
cache[key] = result;
return result;
};
}
// A function to calculate the factorial of a number
function factorial(n) {
if (n === 1) return 1;
return n * factorial(n - 1);
}
// Memoized version of the factorial function
let memoizedFactorial = memoize(factorial);
// Calculating factorials
console.log(memoizedFactorial(5)); // 120
console.log(memoizedFactorial(5)); // 120 (returned from cache)
console.log(memoizedFactorial(6)); // 720
18. Generator Functions
Generator Function: Generator functions can be exited and reentered with saved context. These functions can mitigate callback hell and inversion of control. When generator functions are called, a generator object will be returned.
- yield When the iterator's next() method is called, the generator function's body is executed until the first yield expression, which specifies the value to be returned from the iterator or, with yield*, delegates to another generator function.
- next The next() method returns an object with a value property containing the yielded value and a done property which indicates whether the generator has yielded its last value, as a boolean.
Function:
GeneratorFunction(fn* () { })
Description:Generator will return a function or nothing.
Code Sandbox: Generator Function
//This polyfill checks if the Symbol.iterator property is a function. If it's not, it defines the GeneratorFunction constructor, which is the constructor of a generator function, and sets its Symbol.iterator property to a function that returns an object with a next method. This next method simply returns an object with value set to undefined and done set to true, which is the expected behavior for an iterator that has completed.
(function () {
if (typeof Symbol.iterator !== 'function') {
let GeneratorFunction = (function* () { }).constructor;
GeneratorFunction.prototype[Symbol.iterator] = function () {
return {
next: function () {
return { value: void 0, done: true };
}
};
};
}
})();
Sources: MDN
Keep Learning!
Subscribe to my newsletter
Read articles from sriya directly inside your inbox. Subscribe to the newsletter, and don't miss out.
Written by
sriya
sriya
Hi, I'm a full stack UI developer working in a specialist role. Sparked by details!