


Implémentation de la série React de 0 à 1 : implémentation du cycle de vie et des diff
Cette série d'articles rationalise le contenu principal du framework React (JSX/DOM virtuel/composant/cycle de vie/algorithme diff/...) tout en implémentant un (x)react. 🎜>
- Implémentation de la série React de 0 à 1 - JSX et Virtual DOM
- Implémentation de la série React de 0 à 1 - composants et états|accessoires
// 原来的 _render 函数,为了将职责拆分得更细,将 virtual dom 转为 real dom 的函数单独抽离出来 function vdomToDom(vdom) { if (_.isFunction(vdom.nodeName)) { // 为了更加方便地书写生命周期逻辑,将解析自定义组件逻辑和一般 html 标签的逻辑分离开 const component = createComponent(vdom) // 构造组件 setProps(component) // 更改组件 props renderComponent(component) // 渲染组件,将 dom 节点赋值到 component return component.base // 返回真实 dom } ... }Nous pouvons ajouter ,
méthode, la fonction setProps est la suivante : componentWillMount
componentWillReceiveProps
function setProps(component) { if (component && component.componentWillMount) { component.componentWillMount() } else if (component.base && component.componentWillReceiveProps) { component.componentWillReceiveProps(component.props) // 后面待实现 } },
, componentDidMount
, shouldComponentUpdate
méthode componentWillUpdate
componentDidUpdate
Test du cycle de vie
function renderComponent(component) { if (component.base && component.shouldComponentUpdate) { const bool = component.shouldComponentUpdate(component.props, component.state) if (!bool && bool !== undefined) { return false // shouldComponentUpdate() 返回 false,则生命周期终止 } } if (component.base && component.componentWillUpdate) { component.componentWillUpdate() } const rendered = component.render() const base = vdomToDom(rendered) if (component.base && component.componentDidUpdate) { component.componentDidUpdate() } else if (component && component.componentDidMount) { component.componentDidMount() } if (component.base && component.base.parentNode) { // setState 进入此逻辑 component.base.parentNode.replaceChild(base, component.base) } component.base = base // 标志符 }Testez les cas d'utilisation suivants :
Le résultat de sortie lorsque la page est chargée est le suivant :
class A extends Component { componentWillReceiveProps(props) { console.log('componentWillReceiveProps') } render() { return ( <p>{this.props.count}</p> ) } } class B extends Component { constructor(props) { super(props) this.state = { count: 1 } } componentWillMount() { console.log('componentWillMount') } componentDidMount() { console.log('componentDidMount') } shouldComponentUpdate(nextProps, nextState) { console.log('shouldComponentUpdate', nextProps, nextState) return true } componentWillUpdate() { console.log('componentWillUpdate') } componentDidUpdate() { console.log('componentDidUpdate') } click() { this.setState({ count: ++this.state.count }) } render() { console.log('render') return ( <p> <button onClick={this.click.bind(this)}>Click Me!</button> <A count={this.state.count} /> </p> ) } } ReactDOM.render( <B />, document.getElementById('root') )
Lorsque vous cliquez sur le bouton, le résultat de sortie est le suivant :
componentWillMount render componentDidMount
Implémentation de diff
shouldComponentUpdate componentWillUpdate render componentDidUpdateEn réaction, l'idée deimplémenter diff est de comparer l'ancien et le nouveau dom virtuel et afficher le patch comparé sur la page, obtenant ainsi un rafraîchissement partiel ; Cet article s'appuie sur l'implémentation diff en preact et simple-react. L'idée générale est de comparer l'ancien nœud dom avec le nouveau nœud dom virtuel et d'appeler. la logique correspondante selon différents types de comparaison (nœuds de texte, nœuds non-texte, composants personnalisés) pour obtenir un rendu partiel de la page. La structure globale du code est la suivante :
Ce qui suit implémente la logique correspondante selon différents types de comparaison.
/** * 比较旧的 dom 节点和新的 virtual dom 节点: * @param {*} oldDom 旧的 dom 节点 * @param {*} newVdom 新的 virtual dom 节点 */ function diff(oldDom, newVdom) { ... if (_.isString(newVdom)) { return diffTextDom(oldDom, newVdom) // 对比文本 dom 节点 } if (oldDom.nodeName.toLowerCase() !== newVdom.nodeName) { diffNotTextDom(oldDom, newVdom) // 对比非文本 dom 节点 } if (_.isFunction(newVdom.nodeName)) { return diffComponent(oldDom, newVdom) // 对比自定义组件 } diffAttribute(oldDom, newVdom) // 对比属性 if (newVdom.children.length > 0) { diffChild(oldDom, newVdom) // 遍历对比子节点 } return oldDom }Comparez les nœuds de texteComparez d'abord les nœuds de texte simples, le code est le suivant :
Comparez les nœuds non-textes
// 对比文本节点 function diffTextDom(oldDom, newVdom) { let dom = oldDom if (oldDom && oldDom.nodeType === 3) { // 如果老节点是文本节点 if (oldDom.textContent !== newVdom) { // 这里一个细节:textContent/innerHTML/innerText 的区别 oldDom.textContent = newVdom } } else { // 如果旧 dom 元素不为文本节点 dom = document.createTextNode(newVdom) if (oldDom && oldDom.parentNode) { oldDom.parentNode.replaceChild(dom, oldDom) } } return dom }Comparez les nœuds non-textes nœud de texte, l'idée est de remplacer les anciens nœuds du même niveau par de nouveaux nœuds, le code est le suivant :
Comparer les composants personnalisés
// 对比非文本节点 function diffNotTextDom(oldDom, newVdom) { const newDom = document.createElement(newVdom.nodeName); [...oldDom.childNodes].map(newDom.appendChild) // 将旧节点下的元素添加到新节点下 if (oldDom && oldDom.parentNode) { oldDom.parentNode.replaceChild(oldDom, newDom) } }L'idée de comparer les composants personnalisés est : si les nouveaux et les anciens composants sont différents, remplacez directement l'ancien composant par le nouveau composant ; si l'ancien et le nouveau composant sont identiques, attribuez les accessoires du nouveau composant à l'ancien composant, puis effectuez une comparaison différentielle de les anciens composants avant et après l'obtention des nouveaux accessoires. Le code est le suivant :
Parcourir et comparer les nœuds enfants
// 对比自定义组件 function diffComponent(oldDom, newVdom) { if (oldDom._component && (oldDom._component.constructor !== newVdom.nodeName)) { // 如果新老组件不同,则直接将新组件替换老组件 const newDom = vdomToDom(newVdom) oldDom._component.parentNode.insertBefore(newDom, oldDom._component) oldDom._component.parentNode.removeChild(oldDom._component) } else { setProps(oldDom._component, newVdom.attributes) // 如果新老组件相同,则将新组件的 props 赋到老组件上 renderComponent(oldDom._component) // 对获得新 props 前后的老组件做 diff 比较(renderComponent 中调用了 diff) } }Il existe deux stratégies pour parcourir et comparer les nœuds enfants : l'une consiste à comparer uniquement les nœuds du même niveau, et la une autre consiste à ajouter des attributs clés aux nœuds. Leur objectif est de réduire la complexité de l'espace. Le code est le suivant :
Test
// 对比子节点 function diffChild(oldDom, newVdom) { const keyed = {} const children = [] const oldChildNodes = oldDom.childNodes for (let i = 0; i < oldChildNodes.length; i++) { if (oldChildNodes[i].key) { // 将含有 key 的节点存进对象 keyed keyed[oldChildNodes[i].key] = oldChildNodes[i] } else { // 将不含有 key 的节点存进数组 children children.push(oldChildNodes[i]) } } const newChildNodes = newVdom.children let child for (let i = 0; i < newChildNodes.length; i++) { if (keyed[newChildNodes[i].key]) { // 对应上面存在 key 的情形 child = keyed[newChildNodes[i].key] keyed[newChildNodes[i].key] = undefined } else { // 对应上面不存在 key 的情形 for (let j = 0; j < children.length; j++) { if (isSameNodeType(children[i], newChildNodes[i])) { // 如果不存在 key,则优先找到节点类型相同的元素 child = children[i] children[i] = undefined break } } } diff(child, newChildNodes[i]) // 递归比较 } }Dans la section cycle de vie, la méthode componentWillReceiveProps n'est pas encore exécutée. Il suffit de modifier légèrement la fonction setProps :
Testons le dernier cas de test de la section cycle de vie :
/** * 更改属性,componentWillMount 和 componentWillReceiveProps 方法 */ function setProps(component, attributes) { if (attributes) { component.props = attributes // 这段逻辑对应上文自定义组件比较中新老组件相同时 setProps 的逻辑 } if (component && component.base && component.componentWillReceiveProps) { component.componentWillReceiveProps(component.props) } else if (component && component.componentWillMount) { component.componentWillMount() } }Test du cycle de vie
Adresse du projet, sur la façon de pr
Analyse des instances du cycle de vie des composants React
Explication détaillée du cycle de vie des composants React
Tutoriel vidéo du framework Virtual dom-React
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

JavaScript s'exécute dans les navigateurs et les environnements Node.js et s'appuie sur le moteur JavaScript pour analyser et exécuter du code. 1) Générer une arborescence de syntaxe abstraite (AST) au stade d'analyse; 2) Convertir AST en bytecode ou code machine à l'étape de compilation; 3) Exécutez le code compilé à l'étape d'exécution.

Les tendances futures de Python et JavaScript incluent: 1. Python consolidera sa position dans les domaines de l'informatique scientifique et de l'IA, 2. JavaScript favorisera le développement de la technologie Web, 3. Le développement de plate-forme multiplié deviendra un sujet brûlant, et 4. L'optimisation des performances sera le focus. Les deux continueront d'étendre les scénarios d'application dans leurs champs respectifs et de faire plus de percées dans les performances.

Les choix de Python et JavaScript dans les environnements de développement sont importants. 1) L'environnement de développement de Python comprend Pycharm, Jupyternotebook et Anaconda, qui conviennent à la science des données et au prototypage rapide. 2) L'environnement de développement de JavaScript comprend Node.js, VScode et WebPack, qui conviennent au développement frontal et back-end. Le choix des bons outils en fonction des besoins du projet peut améliorer l'efficacité du développement et le taux de réussite du projet.

Oui, le noyau du moteur de JavaScript est écrit en C. 1) Le langage C fournit des performances efficaces et un contrôle sous-jacent, qui convient au développement du moteur JavaScript. 2) Prendre le moteur V8 comme exemple, son noyau est écrit en C, combinant l'efficacité et les caractéristiques orientées objet de C. 3) Le principe de travail du moteur JavaScript comprend l'analyse, la compilation et l'exécution, et le langage C joue un rôle clé dans ces processus.

JavaScript est au cœur des sites Web modernes car il améliore l'interactivité et la dynamicité des pages Web. 1) Il permet de modifier le contenu sans rafraîchir la page, 2) manipuler les pages Web via Domapi, 3) prendre en charge les effets interactifs complexes tels que l'animation et le glisser-déposer, 4) Optimiser les performances et les meilleures pratiques pour améliorer l'expérience utilisateur.

C et JavaScript réalisent l'interopérabilité via WebAssembly. 1) Le code C est compilé dans le module WebAssembly et introduit dans un environnement JavaScript pour améliorer la puissance de calcul. 2) Dans le développement de jeux, C gère les moteurs de physique et le rendu graphique, et JavaScript est responsable de la logique du jeu et de l'interface utilisateur.

JavaScript est largement utilisé dans les sites Web, les applications mobiles, les applications de bureau et la programmation côté serveur. 1) Dans le développement de sites Web, JavaScript exploite DOM avec HTML et CSS pour réaliser des effets dynamiques et prend en charge des cadres tels que JQuery et React. 2) Grâce à la réactnative et ionique, JavaScript est utilisé pour développer des applications mobiles multiplateformes. 3) Le cadre électronique permet à JavaScript de créer des applications de bureau. 4) Node.js permet à JavaScript d'exécuter le côté du serveur et prend en charge les demandes simultanées élevées.

Python est plus adapté à la science et à l'automatisation des données, tandis que JavaScript est plus adapté au développement frontal et complet. 1. Python fonctionne bien dans la science des données et l'apprentissage automatique, en utilisant des bibliothèques telles que Numpy et Pandas pour le traitement et la modélisation des données. 2. Python est concis et efficace dans l'automatisation et les scripts. 3. JavaScript est indispensable dans le développement frontal et est utilisé pour créer des pages Web dynamiques et des applications à une seule page. 4. JavaScript joue un rôle dans le développement back-end via Node.js et prend en charge le développement complet de la pile.


Outils d'IA chauds

Undresser.AI Undress
Application basée sur l'IA pour créer des photos de nu réalistes

AI Clothes Remover
Outil d'IA en ligne pour supprimer les vêtements des photos.

Undress AI Tool
Images de déshabillage gratuites

Clothoff.io
Dissolvant de vêtements AI

Video Face Swap
Échangez les visages dans n'importe quelle vidéo sans effort grâce à notre outil d'échange de visage AI entièrement gratuit !

Article chaud

Outils chauds

Version crackée d'EditPlus en chinois
Petite taille, coloration syntaxique, ne prend pas en charge la fonction d'invite de code

Bloc-notes++7.3.1
Éditeur de code facile à utiliser et gratuit

Envoyer Studio 13.0.1
Puissant environnement de développement intégré PHP

SublimeText3 version Mac
Logiciel d'édition de code au niveau de Dieu (SublimeText3)

Télécharger la version Mac de l'éditeur Atom
L'éditeur open source le plus populaire
