As a system grows large, it is important to consider some method of decoupling for subsequent maintenance. MVC pattern: Introduces the concept of layers, but there is significant coupling between layers.

MVC

The MVC pattern divides the program into three parts: Model, View and Controller.

  1. Model layer: Model layer: business data processing and storage, data update after update;
  2. View Layer: View layer: human-computer interaction interface, generally shown to the user interface;
  3. Controller layer: the Controller layer is responsible for connecting the Model layer and the View layer, receiving and processing the events triggered by the View layer, and updating the View layer when the data state of the Model layer changes;

The purpose of the MVC pattern is to separate the Model layer from the View layer by introducing the Controller layer, which improves the system’s maintainability and readability. The MVC pattern is just an idea that can be expressed in many ways, but the underlying idea remains the same.

Typical MVC ideas

The typical idea isViewLayers are notified by eventsControllerLayer,ControllerLayer after the event processing to complete the relevant business logic, requirementsModelLayer changes data state,ModelLayer then updates the new data toViewLayer.



Pseudo code

// Pseudo-code examples
Model = {
    data: {data or information that the program needs to manipulate},create: {add data},delete: {delete data},update(data) { 
       Object.assign(m.data, data) // Replace old data with new data
       eventBus.trigger('m:upate') // eventBus triggers the 'M :update' message, notifying the View to refresh
    },
    get:{get data}}// Pseudo-code examples
View = {
    el: The element to refresh,html: `

M V C

.... The content displayed on the page
init(){v.l: Element to refresh},render(){refresh page}}// Pseudo-code examples Controller = { init(){ v.init() // View initialization v.render() // First render c.autoBindEvents() // Automatic event binding eventBus.on('m:update'.() = > { v.render() }) // View refreshes when eventBus triggers 'm:update' }, events:{events are hashed},method(){data = new data after change m.update(data)},autoBindEvents(){automatically bind events}}EventBusCopy the code

EventBus

Function: Communication between objects listens for events at the Controller layer and triggers events at the Model layer

api

  • On (Listening event)
  • This trigger triggers events.
  • Off (Cancel listening)
eventBus.trigger('m:updated') // Trigger the event to shout 'M has been updated'
eventBus.on('m:updated'.() = > { console.log('here')})'Listen for events, execute functions when you hear them'
Copy the code

Table driven programming

Table driven programming is described in Code:

Table-driven methods are methods that allow you to look up information in a table without having to use logical statements (if or case) to find it. In fact, any information can be selected by the table. In simple cases, logical statements tend to be simpler and more straightforward. But as logical chains become more complex, tables become more attractive.

Abstract a class of operations, and then if… else… The contents of the judgments are put into the hash table, so that no matter how many judgments there are, they can be added to the hash table without changing the logic code.

// Automatic binding events for the Controller layer above
events: {// Hash table of event set
   'click #app1': 'a' operation.'click #app2': 'b' operation.'click #app3': 'c' operation.'click #app4': 'd operation',}autoBindEvents() { // Bind events automatically by hashing
    for ( let key in c.events) {
        if ( c.events.hasOwnProperty(key) ) {
            const spaceIndex = key.indexOf(' ')   // Find the array subscript for the 'click #app1' space
            const part1 = key.slice(0, spaceIndex) // Split 'click' and '#app1' with spaceIndex
            const part2 = key.slice(spaceIndex + 1)
            const value = c[c.events[key]]
            v.el.on(part1, part2, value)  // Automatically bind multiple events to an element}}}Copy the code

modular

Decouple, decouple, decouple! Reduce duplicate code, improve code reuse, and make the project structure clearer and easier to maintain.

export

The export statement is used to export real-time bound functions, objects, or raw values from a module so that other programs can use them through the import statement. There are two export methods:

  1. Named exports (any number per module)
  2. Default export (one per module)
// Export a single feature
export letName1 name2,... , nameN;// also var, const
export letName1 =... , name2 =... ,... , nameN;// also var, const
export function FunctionName(){... }export class ClassName {... }// Export the list
export { name1, name2, …, nameN };

// Rename the export
export { variable1 as name1, variable2 asName2,... , nameN };// Deconstruct export and rename
export const { name1, name2: bar } = o;

// Export by default
export default expression;
export default function (...) {... }// also class, function*
export default function name1(...) {... }// also class, function*
export { name1 as default,... };// Export a collection of modules
export * from... ;// does not set the default export
export * as name1 from... ;/ / Draft ECMAScript ® 2 o21
export { name1, name2, …, nameN } from... ;export { import1 as name1, import2 asName2,... , nameN }from... ;export { default } from... ;Copy the code

You can define more than one named export per module, but only one default export is allowed.