preface

This article is only to record their own learning process, there are other understanding students can leave a message. Note that I’ve been working on a 28 strategy where 20 percent of the code implements 80 percent of the functionality, so in this series we’re just going to focus on the core logic and implementation of the functionality


The body of the

The last article introduced how Vue’s updated render is achieved. This article introduces how the updated render is compared to the updated render, which is not the first render

Patch core rendering method rewrite

function patch(oldVnode, vnode) { const isRealElement = oldVnode.nodeType; If (isRealElement) {// oldVnode is a real DOM element that represents the first rendering} else {// oldVnode is a virtual DOM that uses the diff algorithm if (oldvNode.tag! = = vnode. Tag) {/ / if the old and the new tag Replace the old with the new oldVnode. El represents the real dom node -- at the same level to compare oldVnode. El. ParentNode. ReplaceChild (createElm (vnode), oldVnode.el); } // If the old node is a text node if (! oldVnode.tag) { if (oldVnode.text ! == vnode.text) { oldVnode.el.textContent = vnode.text; }} const el = (vnode.el = oldvNode.el); // Const el = (vnode.el = oldvNode.el); updateProperties(vnode, oldVnode.data); / / update the attribute const oldCh = oldVnode. Children | | []; / / old son const newCh = vnode. Children | | []; // New child if (oldch.length > 0 && newch.length > 0) {// New child if (oldch.length > 0 && newch.length > 0) {// New child updateChildren(el, oldCh, newCh); } else if (oldch.length) {// The old has a son and the new has no el.innerhtml = ""; } else if (newch.length) {// new for (let I = 0; i < newCh.length; i++) { const child = newCh[i]; el.appendChild(createElm(child)); }}}}Copy the code

UpdateProperties Updates the property

function updateProperties(vnode, oldProps = {}) { const newProps = vnode.data || {}; // New vnode const el = vnode.el; For (const k in oldProps) {if (! newProps[k]) { el.removeAttribute(k); }} / / for style style to do special processing If there is no need to put the new and old style value set is empty const newStyle = newProps. Style | | {}; const oldStyle = oldProps.style || {}; for (const key in oldStyle) { if (! newStyle[key]) { el.style[key] = ""; For (const key in newProps) {if (key === "style") {for (const styleName in newProps) { el.style[styleName] = newProps.style[styleName]; } } else if (key === "class") { el.className = newProps.class; El.setattribute (key, newProps[key]); }}}Copy the code

UpdateChildren Updates child nodes -diff core method

Function isSameVnode(oldVnode, newVnode) { return oldVnode.tag === newVnode.tag && oldVnode.key === newVnode.key; } function updateChildren(parent, oldCh, newCh) {let oldStartIndex = 0; // let oldStartVnode = oldCh[0]; // let oldEndIndex = oldch.length - 1; // let oldEndVnode = oldCh[oldEndIndex]; // let newStartIndex = 0; // Let newStartVnode = newCh[0]; let newEndIndex = newCh.length - 1; let newEndVnode = newCh[newEndIndex]; Function makeIndexByKey(children) {makeIndexByKey(children) {makeIndexByKey(children) {makeIndexByKey(children) {makeIndexByKey(children) map = {}; children.forEach((item, index) => { map[item.key] = index; }); return map; } let map = makeIndexByKey(oldCh); While (oldStartIndex <= oldEndIndex && newStartIndex <= newEndIndex) {if (oldStartIndex <= oldEndIndex && newStartIndex <= newEndIndex) { // Set the moving vnode to undefined because of the violence comparison process. oldStartVnode) { oldStartVnode = oldCh[++oldStartIndex]; } else if (! oldEndVnode) { oldEndVnode = oldCh[--oldEndIndex]; } else if (isSameVnode(oldStartVnode, newStartVnode)) {// Add patch(oldStartVnode, newStartVnode); OldStartVnode = oldCh[++oldStartIndex]; newStartVnode = newCh[++newStartIndex]; } else if (isSameVnode(oldEndVnode, newEndVnode)) {if (oldEndVnode, newEndVnode); oldEndVnode = oldCh[--oldEndIndex]; newEndVnode = newCh[--newEndIndex]; } else if (isSameVnode(oldStartVnode, newEndVnode)) {// The old header is the same as the new tail patch(oldStartVnode, newEndVnode); parent.insertBefore(oldStartVnode.el, oldEndVnode.el.nextSibling); //insertBefore can move or insert the real dom oldStartVnode = oldCh[++oldStartIndex]; newEndVnode = newCh[--newEndIndex]; } else if (isSameVnode(oldEndVnode, newStartVnode)) {patch(oldEndVnode, newStartVnode); parent.insertBefore(oldEndVnode.el, oldStartVnode.el); oldEndVnode = oldCh[--oldEndIndex]; newStartVnode = newCh[++newStartIndex]; } else {// If none of the above conditions is met, then we need to compare the index with the key of the old child node. // If we can find the new child node, we need to move map[newStartVnode.key]; if (! Parent.insertbefore (createElm(newStartVnode), oldStartvNode.el); } else { let moveVnode = oldCh[moveIndex]; OldCh [moveIndex] = undefined; Parent.insertbefore (movevNode. el, oldStartVNode.el); // Move the found node to the front patch(moveVnode, newStartVnode); If (newStartIndex <= newEndIndex) {for (let I = newStartIndex; i <= newEndIndex; InsertBefore = null const ele = newCh[newEndIndex + 1] == null? null : newCh[newEndIndex + 1].el; parent.insertBefore(createElm(newCh[i]), ele); If (oldStartIndex <= oldEndIndex) {for (let I = oldStartIndex; i <= oldEndIndex; i++) { let child = oldCh[i]; if (child ! = undefined) { parent.removeChild(child.el); }}}}Copy the code

The above code is very long, the logic is: using the method of double pointer comparison of the old and new vNode son node, each need to find a new start index, end index; Compare old start index, end index;

IsSameVnode is used to determine whether the head of the new parent node is the same node. If so, move the pointer (head, head, tail) or move the DOM node (head, tail, tail, head) accordingly. After it is found, patch is executed on the old child node, which will not only update the node, but also play the role of recursive update

If all are not equal do brute force comparison if you find a map using key and index to move the old child to the front if you can’t find it just insert it

And finally, if there are too many old children, delete them and if there are too many new children, add them to the corresponding position

Mind mapping

If you found this article helpful, please like, bookmark or comment. Thank you very much!