Vue 2.x full link operation mechanism?

Initialize and mount

After new Vue() is executed, Vue executes an init() method that initializes lifecycle, events, props, data, methods, computed, watch, and an observer for all data.

The $mount component is executed after initialization, and if it is compiled at runtime (template, render function), it is compiled to render function.

Compile the compile ()

The compile phase is divided into parse, optimize, and Genarate phases, and finally generates the Render function.

  • Parse: Parse generates an abstract syntax tree (AST) by matching the entire template with regular expressions.
  • Optimize: Mainly for marking static nodes. A static node is simply the part of the page that is written dead (no vUE is involved). The function of marking static nodes is to compare the old and new virtual DOM with a patch process when updating the view. If the static nodes are used, they can be skipped directly. This is an optimization of performance in VUE.
  • Genarate: Converts an AST to a Render function string.

responsive

During init(), the getter and setter are set through the object.defineProperty () method, which causes the getter to be executed when the data is read and the setter to be executed when the data is modified.

When executing the Render function, Vue will point to the getter to collect the data dependency. The so-called dependency collection is to allocate a Dep for each data, and then add the Watcher that depends on the data to the corresponding Dep (generally a Vue component corresponds to a Watcher).

The setter executes the notify method through Dep, notifies the wacher to execute the update method, and updates the view.

Dom const dep = new dep () object.defineProperty (obj, key, {enumerable:) object.defineProperty (obj, key, {enumerable:) true, configurable: true, get: Function reactiveGetter () {if (dep.target) {// do the dependent collection dep.depend()} return value}, set: Function reactiveSetter (newVal) {// change the value of value val = newVal // Notify watcher to update dep.notify()}})Copy the code

Virtual DOM

When we execute the compiled render function, we generate something called a virtual DOM, which is a mapping of the real DOM. You can eventually generate the real DOM through a series of operations.

  • Apply the page changes to the virtual DOM, rather than directly to the DOM.
  • When the changes are applied to the virtual DOM, the virtual DOM does not rush to render the page, but merely adjusts the internal state of the virtual DOM, so that the cost of manipulating the virtual DOM becomes very light.
  • When the virtual DOM collects enough changes, it applies those changes to the real DOM once and for all.

Life cycle of Vue

Each Vue instance goes through a series of initialization procedures when it is created — for example, it needs to set up the data listener, compile the template, mount the instance to the DOM, and update the DOM when the data changes. It also runs functions called lifecycle hooks, which give users the opportunity to add their own code at different stages.

Refs is often used in development projects to directly access the methods of child components. Refs will be available only after dom rendering is completed. Make sure that DOM has been rendered before using refs. For example, in a lifecycle mounted(){} hook, or in this.$nextTick(()=>{}). NextTick is a delayed callback that is performed after the next DOM update loop ends. If you use nextTick after modifying the data, you can get the updated DOM in the callback.

Parameter passing between Vue components

Communication between Vue components refers to the following three types of communication: parent-child communication, intergenerational communication, and sibling communication.

  • Props/Emit works with parent and child components.

  • Ref communicates with parent and children components. Ref: if used on a normal DOM element, the reference refers to the DOM element; If used on a child component, the reference points to the component instance. Children is the direct child component of the current component, which is an unordered array. The parent component accesses the child component and passes data through the child component. $children does not guarantee the order, nor is it responsive.

  • EventBus (Emit, on) is suitable for communication between father and son, intergenerational and sibling components. EventBus is a vue instance that acts as a global eventBus. Components pass data through eventbus. on and eventbus. on and eventbus. emit registered events.

  • Attrs, Listeners are suitable for inter-generation component communication. Attrs: contains feature bindings (except class and style) that are not identified (and obtained) in the parent scope of a prop. When a component does not declare any prop, all parent bindings (except class and style) are included, and internal components can be passed through v-bind=” attrs “. Usually used in conjunction with the inheritAttrs option. Listeners: V-ON event listeners in the parent scope (without.native decorators) It can pass in internal components via V-ON =” Listeners “.

  • Provide/Inject is applicable to intergenerational component communication. The provide/inject combination allows an ancestor component to inject a dependency (property and method) into all its descendants, no matter how deep in the component hierarchy, for as long as its upstream and downstream relationships are established, thus enabling cross-level communication between parent and child components. The provide/inject combination is used when developing higher-level plug-ins/component libraries

  • Vuex is suitable for communication between father and son, intergeneration and brother components. Vuex is a state management pattern developed specifically for vue.js applications. At the heart of every Vuex app is the store. A “store” is basically a container that contains most of the states in your app. The state store of Vuex is responsive. When the Vue component reads the state from the store, if the state in the store changes, the corresponding component will be updated accordingly. The only way to change the state in a Store is to explicitly commit the mutation. This makes it easy to track each state change.

Vue routing implementation: Hash mode, History mode, abstract mode

As the business functions of front-end applications become more and more complex, users have higher and higher requirements for use experience, and SPA has become the mainstream form of front-end applications. One of the most notable features of large single-page applications is the front-end routing system that updates the page view by changing the URL without rerequesting the page.

Updating the view without re-requesting the page is one of the core principles of front-end routing. Currently, there are two ways to implement this function in the browser environment:

  • usingURLIn thehash("#");
  • usingHistory interfaceinHTML5Method added in;

Vue-router is a routing plug-in for the Vue.js framework. It controls the implementation mode of the route using the mode parameter:

const router=new VueRouter({
    mode:'history',
    routes:[...]
})
Copy the code

The string attribute mode passed in as an argument is simply a marker indicating the implementation class of the actual object attribute history that is in effect:

    modehistory:
        'history':HTML5History;
        'hash':HashHistory;
        'abstract':AbstractHistory;
Copy the code
  1. After initializing the correspondinghistoryBefore, it would be rightmodeDo some verification: if the browser does not supportHTML5HistoryBy way ofsupportsPushStateVariable judgment), thenmodeSet tohash; If not running in a browser environment, thenmodeSet toabstract;
  2. VueRouterIn the classonReady().push()Such a method is just a proxy, the actual is called concretehistoryObject in theinit()Method, is also based onhistoryDifferent operations are performed for specific types of objects.

** Hash mode: ** In the browser symbol “#”, # and the characters following # are called hash, read with window.location.hash; The hash is in the URL but is not included in the HTTP request. The hash is used to guide browser actions and is useless to the server. The hash does not reload the page.

In hash mode, only the content before the hash symbol is included in the request, such as www.xxx.com. Therefore, the backend does not return a 404 error even if the route is not fully covered. Each change in the hash value adds a record to the browser’s access history. So we can use the browser’s back and forward buttons to control the hash switching; You can use the a tag and set the href attribute. When the user clicks the tag, the HASH value of the URL will change. Or use JavaScript to assign loaction.hash and change the HASH value of the URL. You can use the HashChange event to listen for hash values to jump (render) to the page.

** History mode: ** History uses new HTML5 features; Two new methods are provided: pushState (), replaceState () to make changes to the browser history stack, and popState events to listen for state changes. Through back(),forward(),go() and other methods, we can read the information of the browser history stack and perform various jump operations.

In history mode, the URL on the front end must be the same as the URL that initiates the request on the back end, for example, www.xxx.com/items/id. The backend will return a 404 error if the routing processing for /items/id is missing. So add a candidate resource on the server that covers all cases: if the URL doesn’t match any static resources, it should return the same index.html page that your app depends on.

HTML5 provides the History API to implement URL changes. The two most important apis are history.pushState() and history.repalcestate (). These two apis allow you to manipulate the browser’s history without refreshing it. The only difference is that the former is a new history, the latter is a direct replacement of the current history.

In general requirements scenarios, the hash mode is similar to the History mode. According to MDN, calling history.pushState() has the following advantages over modifying the hash directly:

  • pushStateSet a newurlCan be with the currenturlAny of the same originurlAnd thehashCan only be modified#Therefore, you can only set the same document as the current oneurl
  • pushStateSet a newurlCan be used with the currenturlExactly the same. This will also add the record to the stack, andhashThe new value set must be different from the original to trigger the record to be added to the stack
  • pushStatethroughstateObjectCan be added to any type of data record, andhashOnly short strings can be added
  • pushStateAdditional Settings availabletitleProperty for later use

Vue-router Lazy route loading and three implementation methods

When a project build is packaged, it is usually placed in a separate JS file, but if many pages are placed in the same JS file, the page will inevitably become very large. For a single-page application like Vue, if there is no lazy application loading, the files packaged with WebPack will be unusually large. When entering the home page, too much content needs to be loaded for a long time, resulting in a long white screen. Even if loading is performed, this is not conducive to user experience. And the use of lazy loading can be divided into pages, when needed to load the page, can effectively share the loading pressure of the home page, reduce the loading time of the home page.

The main function is to package the corresponding components of the route into a js code block. The corresponding components are loaded only when the route is accessed. Otherwise, they are not loaded.

The vUE project implements three ways of route loading on demand (lazy route loading) :

  • Vue Asynchronous component

    { path: ‘/problem’, name: ‘problem’, component: resolve => require([‘../pages/home/problemList’], resolve) }

  • ES6 standard syntax import () ——— Recommended

    { path: ‘/problem’, name: ‘problem’, component: () => import(‘.. /pages/home/problemList’) }

  • The require of webpack. Ensure ()

Import vs. require

  • Import is a deconstruction process and is executed at compile time.
  • Require is an assignment process and is executed at runtime, i.e., asynchronous loading.
  • The performance of require is slightly lower than import because require introduces modules at run time and is assigned to a variable.

Understanding of v-Model

In a VUE project, the v-Model directive is used to create bidirectional data binding on elements such as form input, Textarea, select, etc. The V-Model is essentially just syntax sugar. Inside, the V-Model uses different attributes for different input elements and throw different events:

Text and Textarea elements use the value attribute and input event; Checkbox and radio use the Checked property and the change event; The select field takes value as a prop and change as an event.

<input v-model='something'>
Copy the code

The equivalent of

<input :value="something" @input="something = $event.target.value">
Copy the code

Value =”something”; value=”something”; value=”something”; value=”something”; value=”something”; Because the syntax sugar is listening on the input box with @input.

In a custom component

<my-component v-model="inputValue"></my-component>
Copy the code

The equivalent of

<my-component v-bind:value="inputValue" v-on:input="inputValue = argument[0]"></my-component>
Copy the code

In this case, the value that inputValue accepts is the first parameter of the input event callback function, so in the custom component, to implement data binding, you need to emit the input event.

this.$emit('input', value)
Copy the code

What is vuex? How does it work? Which functional scenario uses it?

Vuex is a state management pattern developed specifically for vue.js applications. It uses centralized storage to manage the state of all components of an application, and rules to ensure that the state changes in a predictable manner. In simple terms, vuex is used when an application encounters state shared by multiple components.

Vuex flow: Pages asynchronously submit events to actions via mapAction. Action synchronizes the parameters to mutation through commit, and mutation changes the corresponding values in state. And then you get the value out through the getter, and in the computed property of the page, you get the value in state dynamically through the mapGetter.

What are the properties of VUex

There are five: State, Getter, Mutation, Action, and Module (mapAction).

  • State: vuex’s basic data, used to store variables.
  • Getter: Data derived from the basic data (state), equivalent to the computed property of state.
  • Mutation: The method to submit update data, which must be synchronous (action if required asynchronously). Each mutation has an event type (type) of a string and a callback function (handler). The callback function is where we actually make the state change, and it takes state as the first argument and commits the payload as the second argument.
  • Action: and mutation have much the same functionality, except that 1. Action commits mutation instead of directly changing the state. 2. The Action can contain any asynchronous operation.
  • Modules: Modular VUEX allows each module to have its own state, mutation, action and getters, making the structure very clear and convenient to manage.

Understanding of SPA single page, what are its advantages and disadvantages?

SPA (Single-Page Application) loads the appropriate HTML, JavaScript, and CSS only when the Web page is initialized. Once the page is loaded, SPA will not reload or jump to the page because of user operations; Instead, a routing mechanism is used to transform HTML content, UI interaction with the user, and avoid page reloading.

Advantages:

  • User experience is good, fast, content changes do not need to reload the entire page, avoid unnecessary jump and repeat rendering;
  • Based on the above point, SPA is relatively less pressure on the server;
  • The responsibilities of the front and back end are separated, the architecture is clear, the front end carries out interactive logic, and the back end is responsible for data processing;

Disadvantages:

  • The initial loading is time-consuming: In order to realize the functions and display effect of a single-page Web application, JavaScript and CSS need to be loaded uniformly during the page loading, and some pages need to be loaded on demand.
  • Forward and backward routing management: Since the single-page application displays all the content in one page, the forward and backward function of the browser cannot be used, and all page switching needs to establish stack management.
  • SEO is difficult: because all content is dynamically replaced on one page, it has a natural weakness in SEO.

Understand Vue’s unidirectional data flow?

All prop causes a one-way downward binding between parent and child: updates to the parent prop flow down to the child component, but not vice versa. This prevents the state of the parent component from changing unexpectedly from the child component, which can cause the flow of application data to be difficult to understand.

Additionally, every time the parent component is updated, all prop in the child component will be refreshed to the latest value. This means that a prop should not be changed inside a child component. If you do, Vue will issue a warning in the browser’s console. When the child component wants to modify, it can only send a custom event through $Emit. After receiving the event, the parent component will modify it. There are two common situations in which you try to change a prop:

This prop is used to pass an initial value; The child component next wants to use it as a local prop data. In this case, it is best to define a local data property and use the prop as its initial value:

props: ['initialCounter'],
data: function () {
  return {
    counter: this.initialCounter
  }
}
Copy the code

The prop is passed in with a raw value and needs to be converted. In this case, it is best to use the value of the prop to define a computed property

props: ['size'],
computed: {
  normalizedSize: function () {
    return this.size.trim().toLowerCase()
  }
}
Copy the code

What are the differences between computed and Watch and the scenarios used?

Computed: The value of computed is cached. Only when the value of the dependent attribute changes, the value of computed is recalculated the next time the value of computed is obtained.

Watch: More of a “watch” function, similar to the listening callback for some data, which is executed whenever the listening data changes

Usage scenario: Use computed when you need to perform numeric computations that depend on other data because the caching feature of computed prevents you from having to recalculate the values every time you get them. Watch should be used when we need to perform an asynchronous or expensive operation when data changes. Using the Watch option allows us to perform an asynchronous operation (access an API), limits how often we perform the operation, and sets the intermediate state until we get the final result. These are things that computational properties can’t do.

Can a parent component listen to the lifecycle of a child component?

If the Parent component (Parent) and the Child component (Child) are mounted, the Parent component (Parent) hears that the mounted component is mounted. If the Parent component (Parent) hears that the Child is mounted, the Parent component (Child) hears that the mounted component is mounted.

// Parent.vue
<Child @mounted="doSomething"/>
    
// Child.vue
mounted() {
  this.$emit("mounted");
}
Copy the code

$emit triggers the parent component event manually. A simpler way is to listen when the parent component references the child component via @hook, as shown below:

// Parent. Vue <Child @hook: "></Child> doSomething() {console. '); }, // child.vue mounted(){console.log(' Child component triggers mounted hook function... '); }, // The output sequence is: // The child component triggers the Mounted hook function... // The parent listens to the Mounted hook function...Copy the code

Of course, @hook method can not only listen on mounted, other life cycle events, such as: created, updated, etc.

Why is data a function in a component?

  • Components in VUE are used for reuse. To prevent data reuse, they are defined as functions.
  • The data in the VUE component should be isolated from each other and not affect each other. Each time the component is reused, the data data should be copied once. Later, when the data data in a component that is reused is changed, the data data of other components that are reused will not be affected. You need to return the state of an object as a component through the data function.
  • When we write the data in a component as a function, the data is defined in the form of the return value of the function, so that each time the component is reused, a new copy of data is returned, which has its own scope, similar to creating a private data space for each component instance to maintain its own data.
  • When the date of a component is written as an object and the instances use the same constructor, because of the JavaScript feature, all the component instances share the same data, resulting in a result that all changes.

Using the Vue. $set ()

Since Vue performs getter/setter transformations on the property when initializing the instance, the property must exist on the data object for Vue to convert it to responsive. When a normal JavaScript Object is passed to a Vue instance as the data option, Vue will walk through all the properties of the Object and use object.defineProperty to turn them into getters/setters. Since Vue performs a getter/setter transformation on the property when initializing the instance, the property must be present on the data object for Vue to transform it in order for it to be responsive.

Due to the limitations of modern JavaScript, Vue cannot detect the addition or removal of object properties. However, Vue provides vue.set (object, propertyName, value)/vm.$set (object, propertyName, value) to implement adding responsive properties to objects.

Vue also cannot detect changes when you set an array item directly using the index, such as vm.items[indexOfItem] = newValue, and when you change the length of the array, such as vm. kitems.length = newLength. To solve the first problem, Vue provides the following operations:

Vue.set(vue.items, indexOfItem, newValue) // vue.set (vue.items, indexOfItem, newValue) $set(vm.items, indexOfItem, newValue) // array.prototype.splice vm.items.splice(indexOfItem, 1, indexOfItem) newValue)Copy the code

To solve the second problem, Vue provides the following operations:

// Array.prototype.splice
vm.items.splice(newLength)
Copy the code

Vm.$set is implemented as follows:

  • If the target is an array, trigger the response using the array’s splice method directly.
  • If the target is an object, the attribute is read for existence, the object is responsive, and finally, if the attribute is to be responsive, it is handled responsively by calling defineReactive (which means that when Vue initializes the object, Method to dynamically add getters and setters to an Object property using object.defineProperty)

What is the function of key in Vue?

When vue. js is updating a list of rendered elements with v-for, it defaults to a “reuse in place” policy. If the order of the data items is changed, Vue will not move the DOM elements to match the order of the data items, but will simply reuse each element here and make sure that it shows each element that has been rendered under a specific index.

The key in a Vue is the unique identifier of a Vnode in a Vue. Using this key, we can perform diff operations more accurately and quickly.

More accurate: Since the key is not reused in place, Vue will remember the order of the elements after having the key attribute and will insert/delete elements in the appropriate place to complete the update. So it will be more accurate. Faster: Using the uniqueness of the key to generate a map object to obtain the corresponding node, faster than the traversal method.

Understanding of vue3.0 features

Faster: Node tag + event cache + responsive proxy

  • Node tag: Distinguishes between static nodes and dynamic nodes. For static nodes, diff operation is not performed during update, which saves overhead. No matter how deep the hierarchy is nested, dynamic nodes can be directly traversed during update. No developer awareness.
  • Event Cache enabled: This function is enabled by default to prevent repeated compilation of templates.
  • Reactive proxy: A proxy is installed on top of the target object. The proxy intercepts the object instead of its properties, which greatly improves performance. Mainly through reactive function () to add a Proxy object to monitor internal attributes to implement the data to monitor.

More powerful: Powerful composition API + powerful Teleport + powerful Fragment + powerful createRenderer

  • Powerful Composition API: Positioned as an advanced feature designed to solve problems that occur primarily in large applications. The Compostion API focuses on the JavaScript (logical) parts of the code, keeping the logically related code together so that it is easy to maintain. Within the VUE2 component, code is organized using the Option API style (Data/Methods/Mounted). This will disperse the logic. For example, if we implement a counter function, we will declare variables in data and define response functions in methods. If you want to maintain a function in a component that has a large number of functions and code, you will need to repeatedly switch to the corresponding location in data/ Methods/Mounted and then change the code. Composition API, as the name implies, does not pass data, mounted and other parameters. It implements bidirectional data binding and life cycle function execution by introducing methods such as REF and onMounted. In VUe3, the setup function is used.

  • <template> <div>{{count}}</div> < button@click ="addCount"> </template> <script lang="ts"> import { defineComponent, ref, onMounted } from 'vue'; export default defineComponent({ name: 'App', setup () { const count = ref(0) const getCount = () => { count.value = Math.floor(Math.random() * 10) } const addCount =  () => { count.value++ } onMounted(() => { getCount() }) return { count, addCount } } }); </script>Copy the code
  • Powerful Teleport: Teleport is a technology that allows us to move our templates outside of the Vue app in the DOM. Template code can be written at the logical location of the component, which means we can use the data or props of the component and then render it outside the scope of the Vue application. As follows, in this case, even if Modal is rendered in a different place, it is still a child of the current component (the component that invoked Modal) and will receive prop ** from it

    <teleport to="#modal-container"> <! -- use the modal component, pass in the prop --> <modal :show="showModal" @close="showModal = false"> <template #header> <h3>custom header</h3> </template> </modal> </teleport> import { ref } from 'vue'; import Modal from './Modal.vue'; Export default {components: {Modal}, setup() {// Modal encapsulation const showModal = ref(false); return { showModal } } }**Copy the code
  • Powerful Fragment: Components do not need a unique root node.

  • Powerful createRenderer: In vuE3, in order to achieve better multi-platform applications, the renderer create function createRenderer method is separate, do not need to fork like Weex, and then to change the source code.

Smaller: Tree Shaking loads on demand

Tree Shaking is a technique for optimizing the packaging volume of a project by eliminating redundant code.

In Vue2, no matter what features we use, they end up in the production code. The main reason is that Vue instances are singletons in the project, and the bundle cannot detect which properties of the object are being used in the code.

import Vue from ‘vue’

And Vue3 source introduces tree shaking feature, the global API for partitioning. If you don’t use some of its features, they won’t be included in your base package.

import { nextTick, observable } from ‘vue’

Tree Shaking is based on the ES6 template syntax (import and exports). It takes the idea of static compilation of ES6 modules, which determines module dependencies and input and output variables at compile time.

Tree Shaking does two things:

  • ES6 Module is used during compilation to determine which modules have been loaded.
  • Determine which modules and variables are not being used or referenced, and delete the corresponding code.