Performance is an essential topic when developing Web applications. For single-page applications packaged with WebPack, there are many ways to optimize performance, such as tree-shaking, lazy loading of modules, and speeding up routine optimizations with Extrens Network CDN. Even in vue-CLI projects we can optimize the program by using the –modern directive to generate both old and new browser code.

In fact, caching is definitely one of the most effective ways to improve web applications, especially if users are limited by Internet speed. Improve system responsiveness and reduce network consumption. Of course, the closer the content is to the user, the faster and more effective the cache will be.

On the client side, there are many ways to cache data and resources, such as standard browser caching and the currently popular Service worker. However, they are better suited for caching static content. Such as HTML, JS, CSS and image files. For caching system data, I use a different approach.

So LET me take a look at the various API request caching schemes THAT I applied to the project, from simple to complex.

Plan 1: Data caching

Simple data caching that gets the data on the first request and then uses the data without requesting the back-end API. The code is as follows:

const dataCache = new Map()

async getWares() {
    let key = 'wares'// Retrieve data from the data cachelet data = dataCache.get(key)
    if(! Data) {// No data request server const res = await request.get('/getWares') // Other operations... data = ... DataCache. Set (key, data)}return data
} 
Copy the code

The first line of code uses maps above ES6. If you do not understand maps, refer to the ECMAScript 6 Introduction to Set and Maps or Exploring ES6 about maps and sets, which can be understood as a key-value pair storage structure.

Then the code uses async functions to make asynchronous operations more convenient. You can use the ECMAScript 6 Introduction Async function to learn or consolidate your knowledge.

The code itself is easy to understand, caching data using a Map object and then calling to fetch data from the Map object. For extremely simple business scenarios, just use this code.

Call method:

getWares().then( ... ) // The second call gets the previous data getWares().then(...)Copy the code

Scheme 2: Promise caching

Option one is insufficient by itself. If more than two calls to this API are considered simultaneously, a second request to the API will be made because the request is not returned. Of course, if you add a single data source framework like Vuex or Redux to your system, this is less of a problem, but it can happen when you want to call apis for complex components without having to communicate data to them.

const promiseCache = new Map()

getWares() {
    const key = 'wares'
    letpromise = promiseCache.get(key); // The promise does not exist in the current promise cacheif(! promise) { promise = request.get('/getWares'). Then (res => {// perform operations on res... }). Catch (error => {// After the request returns, if there is a problem, remove the promise from the cache to avoid a second request. S promiseCache.returnPromise.reject(error)}) promiseCache. Set (key, Promise)} // Return a Promisereturn promise
}
Copy the code

This code avoids the problem of multiple requests at the same time in scenario 1. At the same time, the promise is also deleted in the case of backend error, so that there will not be the problem that the cache error promise will always be wrong.

Call method:

getWares().then( ... ) // The second call gets the previous promise getWares().then(...)Copy the code

Scheme 3 multiple promise cache

The scheme is to return data simultaneously if more than one API request is required, and if an error occurs in one API. None of them return correct data.

const querys ={
    wares: 'getWares',
    skus: 'getSku'} const promiseCache = new Map() async queryAll(queryApiName) {// Check whether incoming data is an array const queryIsArray = Array.isArray(queryApiName) const apis = queryIsArray? QueryApiName: [queryApiName] // Get all request services const promiseApi = [] apis. ForEach (API => {// Use promiselet promise = promiseCache.get(api)

        if(promise) {// If there is one in the cache, just push promise.push(promise)}else{promise = request.get(querys[API]). Then (res => { }). Catch (error => {// After the request returns, if there is a problem, the promise is removed from the cache promiseCache. Delete (API)return Promise.reject(error)
            })
            promiseCache.set(api, promise)
            promiseCache.push(promise)
        }
    })
    returnPromise.all(promiseApi).then(res => {// Return data depending on whether a string or an array is passed in, because it is an array operation // If a string is passed in, then fetch is requiredreturn queryIsArray ? res : res[0]
    })
}

Copy the code

This scheme is a way to obtain data from multiple servers simultaneously. Multiple data can be obtained at the same time to operate without errors due to the failure of a single data.

Call way

queryAll('wares').then( ... ) // the 2nd call does not fetch wares, only goes to skus queryAll(['wares'.'skus']).then( ... )
Copy the code

Option four adds time-dependent caching

Caching is often harmful. If we know that the data has been modified, we can directly delete the cache, and then we can call the method to request the server. This way we circumvent the need for the front end to display old data. However, we may not operate on the data for a period of time, so the old data will always exist, so we had better set a time to remove the data. In this scheme, class persistent data is used to do data caching, and expiration time data and parameterization are added. The code looks like this: First define a persistent class, which can store promises or data

class ItemCacheConstruct () {construct(data, timeout) {this.data = data // Set timeout to how many seconds this.timeout = timeout This.cachetime = (new Date()).getTime()}}Copy the code

We then define the data cache. We use basically the same API for Map

Class ExpriesCache {// Define a static data map as the cache pool static cacheMap = new map () // Whether the data times out static isOverTime(name) {const data = ExpriesCache. CacheMap. Get (name) / / no data must be timed outif(! data)return trueConst currentTime = (new Date()).getTime() const overTime = (currentTime -) Data.cachetime) / 1000 // If the number of seconds in the past is greater than the current timeout, null is also returned to fetch data from the serverif(math.abs (comp) > data.timeout) {// This code can not be used, there will be no problem, but if you have this code, you can enter the method again to reduce the judge. ExpriesCache.cacheMap.delete(name)return true} // No timeoutreturn false} static has(name) {return! Expriescache. isOverTime(name)} // Delete data static delete(name) {returnExpriesCache. CacheMap. Delete (name)} / / get the static get (name) {const isDataOverTiem = ExpriesCache. IsOverTime (name) / / if Data timeout, returns NULL, but no timeout, returns data, not ItemCache objectreturnisDataOverTiem ? Null: ExpriesCache. CacheMap. Get (name). The static data} / / the default storage 20 minutesset(name, data, timeout = 1200) {// Set itemCache const itemCache = mew itemCache (data, The timeout) / / cache ExpriesCache. CacheMap. Set (name, itemCache)}}Copy the code

At this point, the data class and the action class are defined, and we can do this at the API layer

// generateKeyError const generateKeyError = new Error("Can't generate key from name and argument") // Generate the key valuefunctionGenerateKey (name, argument) {const params = array.from (argument).join()', ') try{// Returns a string, function name + function argumentreturn `${name}:${params}'}catch(_) {// Returns a key generation errorreturnGenerateKeyError}} async getWare(params1, params2) {// generateKey const key = generateKey('getWare', [params1, params2]) // Get datalet data = ExpriesCache.get(key)
    if(! data) { const res = await request('/getWares'Expriescache. set(key, res, 10)} {params1, params2})return data
}
Copy the code

This scheme uses the method of caching with different expiration time and API parameters. Most business scenarios can be met.

Call way

GetWares (1, 2). Then (...). // the second call gets the previous promise getWares(1,2).then(...) GetWares (1,3).then(...)Copy the code

Scheme 5 scheme 4 based on modifiers

Same solution as scheme 4, but based on modifiers. The code is as follows:

// generateKeyError const generateKeyError = new Error("Can't generate key from name and argument") // Generate the key valuefunctionGenerateKey (name, argument) {const params = array.from (argument).join()', ') try{// Returns a stringreturn `${name}:${params}`
    }catch(_) {
        return generateKeyError
    }
}

function// Check whether the last data descriptor is descriptor. If it is descriptor, use // for examplelogModifiers like thisif (isDescriptor(entryArgs[entryArgs.length - 1])) {
        returnhandleDescription(... entryArgs, []) }else{// If not // modifiers such as add(1) plus(20)return function() {
            returnhandleDescription(... Array.protptype.slice.call(arguments), entryArgs) } } }functionhandleApiCache(target, name, descriptor, ... Config) {// Get the function body and save const fn = description. value // change the function descriptorfunction() {const key = generateKey(name, arguments) // Key cannot be generatedif(key === generateKeyError) {// Make the request with the function body just savedreturn fn.apply(null, arguments)
        }
        let promise = ExpriesCache.get(key)
        if(! Promise = fn.apply(null, arguments). Catch (error => {// After the request comes back, if there is a problem, Expriescache.delete (key) // Returns an errorreturnSet (key, Promise, config[0])} expriesCache.set (key, Promise, config[0])return promise 
    }
    returndescriptor; } // Specify the decoratorfunctionApiCache(... args) {return decorate(handleApiCache, args)
}
Copy the code

At this point we use classes to cache the API

Class Api {// cache 10s@apicache (10) // Do not use the default value at this time, because the current modifier can not get getWare(params1, params2) {return request.get('/getWares')}}Copy the code

Since function promotion exists, there is no way to use functions to make modifiers such as:

var counter = 0;

var add = function () {
  counter++;
};

@add
function foo() {}Copy the code

This code is supposed to go counter equals 1, but it actually turns out that counter equals 0. Because the function is promoted, the actual code executed looks like this

@add
function foo() {
}

var counter;
var add;

counter = 0;

add = function () {
  counter++;
};
Copy the code

So there’s no way to use modifiers on functions. See ECMAScript 6 Getting Started Decorator for details. This approach is simple to write and doesn’t have much impact on the business layer. You cannot change the cache time dynamically

Call way

GetWares (1, 2). Then (...). // the second call gets the previous promise getWares(1,2).then(...) GetWares (1,3).then(...)Copy the code

conclusion

API cache mechanism and scene here is also basically introduced, basically can complete the vast majority of data business cache, here I also want to ask to teach you, there is no better solution, or this blog has any wrong place, welcome to correct, thank you here. There is also a lot of unfinished work that will be improved in a later blog.