What is a cache function
The so-called cache function is to cache the result of the function operation, which is a typical way to exchange memory for performance. It is often used to cache data calculation results and cache objects. A cache is simply a temporary data store that holds data so that future requests for that data can be processed more quickly.
Memory function usage scenarios
In the client front page, some data (such as some data that does not change often later) can be taken and stored in the JS object at the time of the first request, so that you do not have to request the server every time you need it. For pages that make heavy use of similar drop-down boxes, this approach can greatly reduce access to the server. Can provide convenience, reduce the number of queries and consumption of time.
The implementation principle of function caching
- Closures (see this article for an introduction)
- Higher-order functions
Higher-order functions
JavaScript functions actually refer to a variable. Since variables can point to functions and arguments to functions can accept variables, a function can accept arguments to another function, which is called a higher-order function.
We can use the ideas of the higher-order functions to achieve a simple cache, inside the function with an object storage input parameters, if the next time the input of the same parameters, then compare the attributes of the object, the value from the object out, don’t need to continue to run, thus greatly save the waiting time for the client.
function serialize(obj) {
if(obj == null) {
return ' '
} else if(Array.isArray(obj)) {
const items = obj.map((item) = > serialize(item) )
return ` [${items.join(', ')}] `
} else if(isPlainObject(obj)) {
const keys = Object.keys(obj)
keys.sort()
const items = keys.map((key) = > `${key}=${serialize(obj[key])}`)
return ` {${items.join('&')}} `
} else if(typeof obj === 'object' && typeof obj.valueOf === 'function') {
return serialize(obj.valueOf())
} else {
return obj + ' '}}function isPlainObject(obj) {
let prototype = Object.getPrototypeOf(obj)
return obj && typeof obj === 'object' && (prototype === null || prototype === Object.prototype)
}
function memorize(fn) {
const results = {}
return (. args) = > {
const key = serialize(args)
if(! (keyin results)) {
console.log('New cache')
results[key] = fn.apply(this, args)
}
console.log('Cache results', results)
return results[key]
}
}
const add = (. args) = > {
return args.reduce((accValue, curValue, index) = > accValue + curValue, 0)}const adder = memorize(add)
adder(1.2.3) {[1,2,3]: 6}
adder(1.2.3) {[1,2,3]: 6}
adder(1.2.3.4.5) {[1,2,3]: 6, [1,2,3,4,5]: 15}
Copy the code
We included a function in memorize and returned a closure function. According to the serialization, we compared whether the parameter of the function was changed and determined whether results was evaluated in the closure scope or re-executed the input function. As you can see, as long as the parameters are the same, each value is taken from results in the closure scope cache.