1. 首页

带你用85行代码实现一个React.js(手摸手的教你)

  啥也不说,先来看看用我们85行的React.js直接替换别人写好的扫雷项目中的React.js,看看效果怎么样?哈哈,是不是正常运行了。

带你用85行代码实现一个React.js(手摸手的教你)

  Attention:一定不要被下面辣么多的代码吓到,因为下面辣么多的代码只是每次在上一个commit的基础上添加几行而已,大部分都是重复,只是为了让你在阅读上方便,不需要翻到上一个commit。所以,你一定要坚持看完,你也一定会有收获。没有收获找我推款。
  85行代码分18次commit,如下图所示。每个commit实现一个小目标,每个小目标实现一个相对完整的功能。最后由这些功能组合成自己的React.js框架。墙裂推荐你结合每一个commit和这篇文章一起看,再动手写一写。如果这样你还是没有看懂,到上海腾讯来,我真的手把手(气氛有点不对啊)带你写。

带你用85行代码实现一个React.js(手摸手的教你)

1.第一个commit: initial commit 是在github上建立仓库时自动生成的。然后在README.md中说明了一下仓库的用途。用于实现一个基础的React.js框架
2.第二个commit:实现最简单的React渲染字符的功能(第一版) 这个commit主要实现了用React输出一个helloworld到HTML页面中。期待引用了我们的React.js后直接通过下述方式实现

const helloWorld = React.createElement('div', null, `Hello World`);
ReactDOM.render(helloWorld, document.getElementById('root'));

此时我们React.js中的代码如下:


function createElement(parentEle, props, childEle) { let parentElement = document.createElement(parentEle); parentElement.innerHTML = childEle; return parentElement; } function render(insertEle, rootEle) { rootEle.appendChild(insertEle); } React = { createElement } ReactDOM = { render }

3.第三个commit:添加函数组件的功能,能够实现函数组件(第二版) 主要实现能够在React创建组件时,实现能够传递函数定义的组件,后期还会扩展向函数组件中传递参数


const Hello = function () { return React.createElement('div', null, `Hello Version2.0`); }; const helloWorld = React.createElement(Hello, null, null); ReactDOM.render(helloWorld, document.getElementById('root'));

此时我们的React.js中的代码如下:


function createElement(parentEle, props, childEle) { if(typeof parentEle === 'function') { return parentEle(); } else { let parentElement = document.createElement(parentEle); parentElement.innerHTML = childEle; return parentElement; } } function render(insertEle, rootEle) { rootEle.appendChild(insertEle); } React = { createElement } ReactDOM = { render }

4.第四个commit: 主要实现向React中添加子组件的效果(第三版) 实现能够在React.js中添加子组件的效果,如下代码实现添加多个helloworld字符和div片段等。


const HelloVersion3 = function () { return React.createElement('div', null, `版本3.0`); }; const helloWorld1 = React.createElement(HelloVersion3, null, null); const helloWorld2 = React.createElement(HelloVersion3, null, null); const divEle = React.createElement('div', null, `我被一个div标签包裹`); //Js中文网 - 前端进阶资源教程 https://www.javascriptc.com const parent = React.createElement('div', null, helloWorld1, helloWorld2, divEle, `我是文本内容哦` ); ReactDOM.render(parent, document.getElementById('root'));

具体的React.js实现代码如下所示:


function createElement(parentEle, props, ...childEles) { if(typeof parentEle === 'function') { return parentEle(); } else { let parentElement = document.createElement(parentEle); childEles.forEach(child => { if(typeof child === 'string') { parentElement.innerHTML += child; } else if(typeof child === 'object') { parentElement.appendChild(child); } }); return parentElement; } } function render(insertEle, rootEle) { rootEle.appendChild(insertEle); } React = { createElement } ReactDOM = { render }

5.第四个commit:类组件的实现(第四版) 主要是实现采用和React中最常用的使用class来定义组件的方式,如下代码所示:


class Hello { render() { return React.createElement('div', null, `版本四,类组件的实现`); } } const helloWorld = React.createElement(Hello, null, null); ReactDOM.render(helloWorld, document.getElementById('root'));

React.js代码如下:


function createElement(parentEle, props, ...childEles) { if (typeof parentEle === 'function' && /^\s*class\s+/.test(parentEle.toString())) { let component = new parentEle(); return component.render(); }else if (typeof parentEle === 'function'){ return parentEle(); }else { let parentElement = document.createElement(parentEle); childEles.forEach(child => { if(typeof child === 'string') { parentElement.innerHTML += child; } else if(typeof child === 'object') { parentElement.appendChild(child); } }); return parentElement; } } function render(insertEle, rootEle) { rootEle.appendChild(insertEle); } React = { createElement } ReactDOM = { render }

6.第五个commit:React函数组件带参数的实现(第五版) 这个比较简单,其实原理就是向普通函数中传递参数后使用是一个道理,要实现的效果如下代码:


const Hello = ({name}) => { return React.createElement('div', null, `这是 ${name}`); }; const helloWorld = React.createElement(Hello, {name: '版本五'}, null); ReactDOM.render(helloWorld, document.getElementById('root'));

React.js代码如下:


function createElement(parentEle, props, ...childEles) { if (typeof parentEle === 'function' && /^\s*class\s+/.test(parentEle.toString())) { let component = new parentEle(); return component.render(); }else if (typeof parentEle === 'function'){ return parentEle(props); }else {//Js中文网 - 前端进阶资源教程 https://www.javascriptc.com let parentElement = document.createElement(parentEle); childEles.forEach(child => { if(typeof child === 'string') { parentElement.innerHTML += child; } else if(typeof child === 'object') { parentElement.appendChild(child); } }); return parentElement; } } function render(insertEle, rootEle) { rootEle.appendChild(insertEle); } React = { createElement } ReactDOM = { render }

7.第六个commit:类组件属性的传递(第六版) 这个commit主要用来解决如何向内组件中传递参数的问题,实现如下代码效果


class Hello extends React.Component { constructor(props) { super(props); } render() { return React.createElement('div', null, `Hello ${this.props.name}`); } } const helloWorld = React.createElement(Hello, {name: '文字'}, null); ReactDOM.render(helloWorld, document.getElementById('root'));

Js中文网 – 前端进阶资源教程 www.javascriptC.com,typescript 中文文档
一个帮助开发者成长的社区,你想要的,在这里都能找到

React.js代码如下所示:


class Component { constructor(props) { this.props = props; } } function createElement(parentEle, props, ...childEles) { if (typeof parentEle === 'function' && /^\s*class\s+/.test(parentEle.toString())) { // 当为类组件时 let component = new parentEle(props); return component.render(); }else if (typeof parentEle === 'function') { // 当为函数组件时 return parentEle(props); }else { // 当为html标签组件时 let parentElement = document.createElement(parentEle); childEles.forEach(child => { if(typeof child === 'string') { parentElement.innerHTML += child; } else if(typeof child === 'object') { parentElement.appendChild(child); } }); return parentElement; } } function render(insertEle, rootEle) { rootEle.appendChild(insertEle); } React = { createElement, Component } ReactDOM = { render }

8.第七个commit:类组件添加事件属性(第七版) 主要实现能够绑定click事件的功能,需要实现的效果代码如下:


class MyButton extends React.Component { constructor(props) { super(props); } render() { return React.createElement('button', {onclick: this.props.onClick}, `Click me`); } } const myBtn = React.createElement(MyButton, {onClick: () => alert('点击事件触发')}, null); ReactDOM.render(myBtn, document.getElementById('root'));

React.js代码如下:


class Component { constructor(props) { this.props = props; } } function createElement(parentEle, props, ...childEles) { if (typeof parentEle === 'function' && /^\s*class\s+/.test(parentEle.toString())) { // 当为类组件时 let component = new parentEle(props); return component.render(); } else if (typeof parentEle === 'function') { // 当为函数组件时 return parentEle(props); } else { // 当为html标签组件时 let parentElement = document.createElement(parentEle); Object.keys(props).forEach(key => { switch(key) { case 'onclick': parentElement.addEventListener('click', props[key]); break; default: break; } }); childEles.forEach(child => { if(typeof child === 'string') { parentElement.innerHTML += child; } else if(typeof child === 'object') { parentElement.appendChild(child); } }); return parentElement; } } function render(insertEle, rootEle) { rootEle.appendChild(insertEle); } React = { createElement, Component } ReactDOM = { render }

9.第八个commit:实现类组件的动态渲染(数据的变化能够同步到页面上)(第八版) 到这一步已经可以使用上述的功能完成一个小的应用了,我们来试一下,实现一个简单的计数器的应用。技术器的代码如下:


class Counter extends React.Component { constructor(props) { super(props); this.state = {value: 0}; } onPlusClick() { this.setState({value: this.state.value + 1}); } onMinusClick() { this.setState({value: this.state.value - 1}); } render() { return React.createElement('div', null, React.createElement('div', null, `The Famous Dan Abramov's Counter`), React.createElement('div', null, `${this.state.value}`), React.createElement('button', {onClick: this.onPlusClick.bind(this)}, '+'), React.createElement('button', {onClick: this.onMinusClick.bind(this)}, '-') ); } } let myCounter = React.createElement(Counter,null,null); ReactDOM.render(myCounter, document.getElementById('root'));

Js中文网 – 前端进阶资源教程 www.javascriptC.com,typescript 中文文档
一个帮助开发者成长的社区,你想要的,在这里都能找到

React.js代码如下:


let rootElement, rootReactElement; // React基础组件库 class Component { constructor(props) { this.props = props; } setState(state) { this.state = state; reRender(); } } // React.createElement function createElement(parentEle, props, ...childEles) { if (typeof parentEle === 'function' && /^\s*class\s+/.test(parentEle.toString())) { // 当为类组件时 let component = new parentEle(props); return component; } else if (typeof parentEle === 'function') { // 当为函数组件时 return parentEle(props); } else { // 当为html标签组件时 let parentElement = document.createElement(parentEle); Object.keys(props || {}).forEach(key => { switch(key) { case 'onclick': parentElement.addEventListener('click', props[key]); break; case 'onClick': parentElement.addEventListener('click', props[key]); break; default: break; } }); childEles.forEach(child => { if(typeof child === 'string') { parentElement.innerHTML += child; } else if(typeof child === 'object') { parentElement.appendChild(child); } }); return parentElement; } } function render(insertEle, rootEle) { rootElement = rootEle; rootReactElement = insertEle; rootEle.appendChild(insertEle.render()); } function reRender() { while(rootElement.hasChildNodes()) { rootElement.removeChild(rootElement.lastChild); } ReactDOM.render(rootReactElement, rootElement); } React = { createElement, Component } ReactDOM = { render }

实现的计数器的效果图如下所示:

带你用85行代码实现一个React.js(手摸手的教你)

到这里我们就实现了一个最简单的React.js框架。因为如果再叙述到实现扫雷文章的长度实在是太长了,如果你觉得还有必要把接下来的几个commit看完(每个commit其实也就变更了很少行数的代码)你可以移步我的github查看这篇文章的源代码,逐个查看Todo.md和变更的代码,有什么疑问都可以在项目的issue中留言,不要吝啬你的Star哦,你的Star是我继续写文章的动力,谢谢。

作者:殷荣桧
链接:https://juejin.im/post/5ba906eae51d450e78261dbb

看完两件小事

如果你觉得这篇文章对你挺有启发,我想请你帮我两个小忙:

  1. 关注我们的 GitHub 博客,让我们成为长期关系
  2. 把这篇文章分享给你的朋友 / 交流群,让更多的人看到,一起进步,一起成长!
  3. 关注公众号 「画漫画的程序员」,公众号后台回复「资源」 免费领取我精心整理的前端进阶资源教程

JS中文网是中国领先的新一代开发者社区和专业的技术媒体,一个帮助开发者成长的社区,目前已经覆盖和服务了超过 300 万开发者,你每天都可以在这里找到技术世界的头条内容。欢迎热爱技术的你一起加入交流与学习,JS中文网的使命是帮助开发者用代码改变世界

本文著作权归作者所有,如若转载,请注明出处

转载请注明:文章转载自「 Js中文网 · 前端进阶资源教程 」https://www.javascriptc.com

标题:带你用85行代码实现一个React.js(手摸手的教你)

链接:https://www.javascriptc.com/2661.html

« Web 堆栈选择指南:JAMStack vs MEAN vs LAMP
Vuex和Redux都使用的Flux设计模式精简版实现»
Flutter 中文教程资源

相关推荐

QR code