前者工程化学习,致我们一定组件化的Web

前端组件化开发执行

2015/07/12 · CSS,
HTML5,
JavaScript ·
组件化

初稿出处: 美团技术博客 –
spring   

致大家终将组件化的Web

2015/11/25 · HTML5 · 1
评论 ·
组件化

原文出处:
前者工程化学习,致我们一定组件化的Web。AlloyTeam   

那篇小说将从两年前的一次技术争议起来。冲突的聚焦就是下图的八个目录分层结构。我说按模块划分好,他说您傻逼啊,当然是按能源划分。

美高梅开户网址 1 《=》美高梅开户网址 2

”按模块划分“目录结构,把当前模块下的享有逻辑和能源都放一块了,那对于多个人独立开发和爱慕个人模块不是很好吧?当然了,那争辩的结果是自己宝宝地改回主流的”按财富划分“的目录结构。因为,没有形成JS模块化和能源模块化,仅仅物理地点上的模块划分是向来不意义的,只会大增创设的工本而已。

就算如此他说得好有道理作者无言以对,可是本身心不甘,等待他目前端组件化成熟了,再来首次大战!

而前几天就是自家一再正义的生活!只是那时候不行跟你撕逼的人不在。

模块化的缺少

模块一般指可以单独拆分且通用的代码单元。由于JavaScript语言自个儿并未松开的模块机制(ES6有了!!),我们一般会利用CMD或ADM建立起模块机制。未来半数以上稍稍大型一点的门类,都会动用requirejs可能seajs来落实JS的模块化。四个人分工协作开发,其各自定义正视和暴光接口,维护效能模块间独立性,对于项目的支出效能和花色前期增添和掩护,都以是有很大的帮扶功用。

但,麻烦我们有个别略读一下底下的代码

JavaScript

require([
‘Tmpl!../tmpl/list.html’,’lib/qqapi’,’module/position’,’module/refresh’,’module/page’,’module/net’
], function(listTmpl, QQapi, Position, Refresh, Page, NET){ var foo =
”, bar = []; QQapi.report(); Position.getLocaiton(function(data){
//… }); var init = function(){ bind();
NET.get(‘/cgi-bin/xxx/xxx’,function(data){ renderA(data.banner);
renderB(data.list); }); }; var processData = function(){ }; var bind =
function(){ }; var renderA = function(){ }; var renderB =
function(data){ listTmpl.render(‘#美高梅开户网址 ,listContent’,processData(data)); };
var refresh = function(){ Page.refresh(); }; // app start init(); });

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
require([
    ‘Tmpl!../tmpl/list.html’,’lib/qqapi’,’module/position’,’module/refresh’,’module/page’,’module/net’
], function(listTmpl, QQapi, Position, Refresh, Page, NET){
    var foo = ”,
        bar = [];
    QQapi.report();
    Position.getLocaiton(function(data){
        //…
    });
    var init = function(){
        bind();
        NET.get(‘/cgi-bin/xxx/xxx’,function(data){
            renderA(data.banner);
            renderB(data.list);
        });
    };
    var processData = function(){
    };
    var bind = function(){
    };
    var renderA = function(){
    };
    var renderB = function(data){
        listTmpl.render(‘#listContent’,processData(data));
    };
    var refresh = function(){
        Page.refresh();
    };
    // app start
    init();
});

地点是有血有肉有个别页面的主js,已经封装了像Position,NET,Refresh等功效模块,但页面的主逻辑如故是”面向进程“的代码结构。所谓面向进程,是指根据页面的渲染进度来编排代码结构。像:init
-> getData -> processData -> bindevent -> report -> xxx

方法之间线性跳转,你大约也能感受这样代码弊端。随着页面逻辑更是复杂,那条”进度线“也会特别长,并且进一步绕。加之紧缺专业约束,其余品种成员按照各自需要,在”进程线“加插各自逻辑,最后这些页面的逻辑变得难以维护。

美高梅开户网址 3

支出须要谨慎,生怕影响“进度线”前面不奇怪逻辑。并且每一回加插或改动都以bug泛滥,无不令产品有关人士无不忧心忡忡。

 页面结构模块化

根据下面的面向进度的难题,行业内也有无数消除方案,而作者辈社团也总括出一套成熟的消除方案:Abstractjs,页面结构模块化。大家可以把大家的页面想象为八个乐高机器人,必要不一样零件组装,如下图,如果页面划分为tabContainer,listContainer和imgsContainer两个模块。最后把这几个模块add到结尾的pageModel里面,最后使用rock方法让页面运转起来。

美高梅开户网址 4
(原经过线示例图)

美高梅开户网址 5
(页面结构化示例图)

下边是伪代码的落到实处

JavaScript

require([
‘Tmpl!../tmpl/list.html’,’Tmpl!../tmpl/imgs.html’,’lib/qqapi’,’module/refresh’,’module/page’
], function(listTmpl, imgsTmpl, QQapi, Refresh, Page ){ var
tabContainer = new RenderModel({ renderContainer: ‘#tabWrap’, data: {},
renderTmpl: “<li soda-repeat=’item in
data.tabs’>{{item}}</li>”, event: function(){ // tab’s event }
}); var listContainer = new ScrollModel({ scrollEl: $.os.ios ?
$(‘#Page’) : window, renderContainer: ‘#listWrap’, renderTmpl:
listTmpl, cgiName: ‘/cgi-bin/index-list?num=1’, processData:
function(data) { //… }, event: function(){ // listElement’s event },
error: function(data) { Page.show(‘数据再次来到格外[‘ + data.retcode +
‘]’); } }); var imgsContainer = new renderModel({ renderContainer:
‘#imgsWrap’, renderTmpl: listTmpl, cgiName: ‘/cgi-bin/getPics’,
processData: function(data) { //… }, event: function(){ //
imgsElement’s event }, complete: function(data) { QQapi.report(); } });
var page = new PageModel();
page.add([tabContainer,listContainer,imgsContainer]); page.rock(); });

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
require([
    ‘Tmpl!../tmpl/list.html’,’Tmpl!../tmpl/imgs.html’,’lib/qqapi’,’module/refresh’,’module/page’
], function(listTmpl, imgsTmpl, QQapi, Refresh, Page ){
 
    var tabContainer = new RenderModel({
        renderContainer: ‘#tabWrap’,
        data: {},
        renderTmpl: "<li soda-repeat=’item in data.tabs’>{{item}}</li>",
        event: function(){
            // tab’s event
        }
    });
 
    var listContainer = new ScrollModel({
        scrollEl: $.os.ios ? $(‘#Page’) : window,
        renderContainer: ‘#listWrap’,
        renderTmpl: listTmpl,
        cgiName: ‘/cgi-bin/index-list?num=1’,
        processData: function(data) {
            //…
        },
        event: function(){
            // listElement’s event
        },
        error: function(data) {
            Page.show(‘数据返回异常[‘ + data.retcode + ‘]’);
        }
    });
 
    var imgsContainer = new renderModel({
        renderContainer: ‘#imgsWrap’,
        renderTmpl: listTmpl,
        cgiName: ‘/cgi-bin/getPics’,
        processData: function(data) {
            //…
        },
        event: function(){
            // imgsElement’s event
        },
        complete: function(data) {
           QQapi.report();
        }
    });
 
    var page = new PageModel();
    page.add([tabContainer,listContainer,imgsContainer]);
    page.rock();
 
});

大家把那几个常用的伸手CGI,处理数据,事件绑定,上报,容错处理等一多级逻辑形式,以页面块为单位封装成3个Model模块。

这么的2个空洞层Model,大家得以清晰地见到该页面块,请求的CGI是如何,绑定了何等风云,做了何等上报,出错怎么处理。新增的代码就应当放置在对应的模块上相应的气象方法(preload,process,event,complete…),杜绝了往年的无规则乱增代码的著述。并且,依照不同工作逻辑封装不一致种类的Model,如列表滚动的ScrollModel,滑块功用的SliderModel等等,可以展开高度封装,集中优化。

距今依照Model的页面结构开发,已经包括一点”组件化“的寓意。各种Model都含有各自的数码,模板,逻辑。已经算是七个完整的作用单元。但相差真正的WebComponent依然有一段距离,至少知足不断作者的”理想目录结构“。

 WebComponents 标准

大家回想一下接纳三个datapicker的jquery的插件,所急需的步奏:

  1. 引入插件js

  2. 引入插件所需的css(如若有)

  3. copy 组件的所需的html片段

  4. 添加代码触发组件运转

时下的“组件”基本上只好落得是有个别意义单元上的会见。他的财富都以松散地分散在二种财富文件中,而且组件功效域暴光在大局意义域下,缺少内聚性很简单就会跟其他零件暴发争辩,如最简便的css命名争辨。对于那种“组件”,还不如上边的页面结构模块化。

于是乎W3C按耐不住了,制定三个WebComponents标准,为组件化的前途指点了明路。

上边以较为简单的主意介绍这份正经,力求大家可以连忙掌握落成组件化的始末。(对这有的打听的同桌,可以跳过这一小节)

1. <template>模板能力

模板那东西我们最熟稔但是了,前一年见的较多的模版品质大战artTemplate,juicer,tmpl,underscoretemplate等等。而将来又有mustachejs无逻辑模板引擎等新入选手。然则我们有没有想过,这么基础的力量,原生HTML5是不帮助的(T_T)。

而前几日WebComponent将要提供原生的模板能力

XHTML

<template id=”datapcikerTmpl”>
<div>作者是原生的沙盘</div> </template>

1
2
3
<template id="datapcikerTmpl">
<div>我是原生的模板</div>
</template>

template标签内定义了myTmpl的模版,需求采纳的时候将要innerHTML= document.querySelector('#myTmpl').content;可以看看那么些原生的模板够原始,模板占位符等功能都不曾,对于动态数据渲染模板能力只可以自力更新。

2. ShadowDom 封装组件独立的内部结构

ShadowDom可以知晓为一份有单独成效域的html片段。那个html片段的CSS环境和主文档隔离的,各自保持内部的独立性。也多亏ShadowDom的单独天性,使得组件化成为了大概。

JavaScript

var wrap = document.querySelector(‘#wrap’); var shadow =
wrap.createShadowRoot(); shadow.innerHTML = ‘<p>you can not see me
</p>’

1
2
3
var wrap = document.querySelector(‘#wrap’);
var shadow = wrap.createShadowRoot();
shadow.innerHTML = ‘<p>you can not see me </p>’

在切实dom节点上采用createShadowRoot方法即可生成其ShadowDom。就如在整份Html的屋子里面,新建了3个shadow的房间。房间外的人都不知情房间内有哪些,保持shadowDom的独立性。

3. 自定义原生标签

首先接触Angularjs的directive指令功能,设定好组件的逻辑后,壹个<Datepicker
/>就能引入整个组件。如此狂炫酷炸碉堡天的成效,实在令人弹冠相庆,跃地三尺。

JavaScript

var tmpl = document.querySelector(‘#datapickerTmpl’); var
datapickerProto = Object.create(HTMLElement.prototype); //
设置把大家模板内容大家的shadowDom datapickerProto.createdCallback =
function() { var root = this.createShadowRoot();
root.appendChild(document.importNode(tmpl.content, true)); }; var
datapicker = docuemnt.registerElement(‘datapicker’,{ prototype:
datapickerProto });

1
2
3
4
5
6
7
8
9
10
11
12
var tmpl = document.querySelector(‘#datapickerTmpl’);
var datapickerProto = Object.create(HTMLElement.prototype);
 
// 设置把我们模板内容我们的shadowDom
datapickerProto.createdCallback = function() {
    var root = this.createShadowRoot();
    root.appendChild(document.importNode(tmpl.content, true));
};
 
var datapicker = docuemnt.registerElement(‘datapicker’,{
    prototype: datapickerProto
});

Object.create形式继续HTMLElement.prototype,得到3个新的prototype。当解析器发现我们在文档中标记它将检查是还是不是贰个名为createdCallback的主意。若是找到那几个法子它将即时运维它,所以大家把克隆模板的故事情节来创造的ShadowDom。

末尾,registerElement的格局传递大家的prototype来注册自定义标签。

地点的代码初叶略显复杂了,把前面八个能力“模板”“shadowDom”结合,形成组件的中间逻辑。最终通过registerElement的措施注册组件。之后方可愉悦地<datapicker></datapicker>的使用。

4. imports缓解组件间的敬服性

XHTML

<link rel=”import” href=”datapciker.html”>

1
<link rel="import" href="datapciker.html">

其一类php最常用的html导入作用,HTML原生也能协理了。

WebComponents标准内容大体到此处,是的,作者这里没有怎么Demo,也远非实践经验分享。由于webComponents新特点,基本上除了高版本的Chrome资助外,其余浏览器的支撑度甚少。即使有polymer帮助拉动webcompoents的库存在,不过polymer本人的须求版本也是越发高(IE10+)。所以先天的中坚并不是他。

我们大致来回想一下WebCompoents的四局地作用:

1 .<template>定义组件的HTML模板能力

  1. Shadow Dom封装组件的内部结构,并且维持其独立性

  2. Custom Element 对外提供组件的标签,落成自定义标签

  3. import化解组件结合和凭借加载

 组件化实践方案

合法的正经看完了,大家寻思一下。一份真正成熟可信的组件化方案,须求具有的力量。

“财富高内聚”—— 组件能源内部高内聚,组件能源由本人加载控制

“功能域独立”—— 内部结构密封,不与全局或别的零件发生潜移默化

“自定义标签”—— 定义组件的应用办法

“可相互结合”—— 组件正在有力的地点,组件间组装整合

“接口规范化”—— 组件接口有联合标准,大概是生命周期的管住

私家觉得,模板能力是基础力量,跟是或不是组件化没有强联系,所以没有提议二个大点。

既然如此是实践,现阶段WebComponent的支撑度还不成熟,不只怕看做方案的手腕。而除此以外一套以高质量虚拟Dom为切入点的机件框架React,在facebook的造势下,社区拿到了大力发展。此外一名骨干Webpack,负责解决组件财富内聚,同时跟React相当切合形成互补。

所以【Webpack】+【React】将会是那套方案的宗旨技术。

不亮堂你以后是“又是react+webpack”感到失望美高梅开户网址 6,如故“太好了是react+webpack”不用再学几次新框架的愉悦美高梅开户网址 7。无论怎么样下边的始末不会让您失望的。

壹,组件生命周期

美高梅开户网址 8

React天生就是强制性组件化的,所以可以从根特性上化解面向进程代码所带来的费劲。React组件本身有生命周期方法,可以知足“接口规范化”能力点。并且跟“页面结构模块化”的所封装抽离的多少个法子能挨个对应。别的react的jsx自带模板功用,把html页面片间接写在render方法内,组件内聚性越发紧凑。

由于React编写的JSX是会先生成虚拟Dom的,要求时机才真的插入到Dom树。使用React必要求明了组件的生命周期,其生命周期多少个状态:

Mount: 插入Dom

Update: 更新Dom

Unmount: 拔出Dom

mount这单词翻译增添,嵌入等。我倒是指出“插入”更好精晓。插入!拔出!插入!拔出!默念两次,懂了没?别少看黄段子的能力,

美高梅开户网址 9

组件状态就是: 插入-> 更新 ->拔出。

然后各个组件状态会有三种处理函数,一前一后,will函数和did函数。

componentWillMount()  准备插入前

componentDidlMount()  插入后

componentWillUpdate() 准备更新前

componentDidUpdate()  更新后

componentWillUnmount() 准备拔出前

因为拔出后基本都以贤者形态(小编说的是组件),所以并未DidUnmount这几个措施。

除此以外React其它二个为主:数据模型props和state,对应着也有自个状态方法

getInitialState()     获取发轫化state。

getDefaultProps() 获取私自认同props。对于那么些并未父组件传递的props,通过该格局设置专断认同的props

componentWillReceiveProps()  已插入的零件收到新的props时调用

再有二个尤其境况的处理函数,用于优化处理

shouldComponentUpdate():判断组件是不是须要update调用

丰富最要紧的render方法,React本人带的法子刚刚好十一个。对于初学者的话是比较麻烦消化。但实际getInitialStatecomponentDidMountrender三个境况方法都能一鼓作气半数以上零件,不必惧怕。

回去组件化的大旨。

1个页面结构模块化的零部件,能独立包装整个组件的进程线

美高梅开户网址 10

大家换算成React生命周期方法:

美高梅开户网址 11

 

零件的情状方法流中,有两点要求独特表达:

1,二次渲染:

鉴于React的虚构Dom个性,组件的render函数不需协调触发,依据props和state的更改自个通过差距算法,得出最优的渲染。

伸手CGI一般都以异步,所以肯定带来二次渲染。只是空数据渲染的时候,有只怕会被React优化掉。当数码回来,通过setState,触发二次render

 

2,componentWiillMount与componentDidMount的差别

和一大半React的课程小说差距,ajax请求作者指出在威尔Mount的法子内实施,而不是组件初叶化成功未来的DidMount。那样能在“空数据渲染”阶段以前请求数据,尽早地回落二次渲染的光阴。

willMount只会履行四回,相当适合做init的工作。

didMount也只会举办五次,并且这时候真实的Dom已经形成,极度适合事件绑定和complete类的逻辑。

 

 二,JSX很丑,可是组件内聚的根本!

WebComponents的正规化之1、必要模板能力。本是觉得是我们熟练的模版能力,但React中的JSX那样的怪人照旧让人议论纷纭。React还未曾火起来的时候,大家就曾经在博客园上狠狠地吐槽了“JSX写的代码那TM的丑”。那实际上只是Demo阶段JSX,等到实战的大型项目中的JSX,包括多情状多数据多事件的时候,你会意识………….JSX写的代码还是很丑。

美高梅开户网址 12
(即便用sublime-babel等插件高亮,逻辑和渲染耦合一起,阅读性依旧略差)

为啥我们会认为丑?因为大家早就经对“视图-样式-逻辑”分离的做法潜移默化。

依据维护性和可读性,甚至质量,我们都不提议直接在Dom上边绑定事件恐怕直接写style属性。大家会在JS写事件代理,在CSS上写上classname,html上的就是清晰的Dom结构。我们很好地保险着MVC的设计情势,一切有惊无险。直到JSX把他们都夹杂在同步,所守护的技巧栈受到凌犯,难免存有抗拒。

 

只是从组件化的目标来看,那种高内聚的做法未尝不可。

上面的代码,之前的“逻辑视图分离”情势,大家须要去找相应的js文件,相应的event函数体内,找到td-info的class所绑定的事件。

相比较起JSX的中度内聚,所有事件逻辑就是在自己jsx文件内,绑定的就是自家的showInfo方法。组件化的风味能立即展现出来。

(注意:即便写法上大家好像是HTML的内联事件处理器,但是在React底层并不曾实际赋值类似onClick属性,内层依旧使用类似事件代理的法子,高效地保证着事件处理器)

再来看一段style的jsx。其实jsx没有对体制有硬性规定,大家完全可听从从前的定义class的逻辑。任何一段样式都应该用class来定义。在jsx你也统统可以如此做。然则出于组件的独立性,小编提出部分唯有“一遍性”的体裁间接利用style赋值更好。裁减冗余的class。

XHTML

<div className=”list” style={{background: “#ddd”}}> {list_html}
</div>

1
2
3
<div className="list" style={{background: "#ddd"}}>
   {list_html}
</div>

或是JSX内部有负担繁琐的逻辑样式,可JSX的自定义标签能力,组件的黑盒性立马能体会出来,是或不是弹指之间美好了好多。

JavaScript

render: function(){ return ( <div> <Menus
bannerNums={this.state.list.length}></Menus> <TableList
data={this.state.list}></TableList> </div> ); }

1
2
3
4
5
6
7
8
render: function(){
    return (
      <div>
         <Menus bannerNums={this.state.list.length}></Menus>
         <TableList data={this.state.list}></TableList>
      </div>
   );
}

虽说JSX本质上是为着虚拟Dom而准备的,但那种逻辑和视图中度合一对于组件化未尝不是一件善事。

 

上学完React那个组件化框架后,看看组件化能力点的做到意况

“财富高内聚”—— (33%)  html与js内聚

“成效域独立”—— (四分之一)  js的成效域独立

“自定义标签”—— (100%)jsx

“可交互结合”—— (59%)  可整合,但缺少使得的加载格局

“接口规范化”—— (100%)组件生命周期方法

 

Webpack 能源组件化

对于组件化的能源独立性,一般的模块加载工具和营造流程视乎变得勤奋。组件化的营造工程化,不再是事先大家广大的,css合二,js合3、而是体验在组件间的倚重于加载关系。webpack正好适合必要点,一方面填补组件化能力点,另一方帮忙大家周详组件化的全体构建环境。

首先要阐明一(Wissu)点是,webpack是3个模块加载打包工具,用于管理你的模块资源倚重打包难题。那跟大家纯熟的requirejs模块加载工具,和grunt/gulp打造工具的概念,多多少少有个别出入又有点雷同。

美高梅开户网址 13

首先webpak对于CommonJS与英特尔同时支持,满足我们模块/组件的加载方式。

JavaScript

require(“module”); require(“../file.js”); exports.doStuff = function()
{}; module.exports = someValue;

1
2
3
4
require("module");
require("../file.js");
exports.doStuff = function() {};
module.exports = someValue;

JavaScript

define(“mymodule”, [“dep1”, “dep2”], function(d1, d2) { return
someExportedValue; });

1
2
3
define("mymodule", ["dep1", "dep2"], function(d1, d2) {
    return someExportedValue;
});

自然最强劲的,最卓越的,当然是模块打包作用。那正是这一作用,补充了组件化能源看重,以及完整工程化的能力

依照webpack的规划理念,所有财富都是“模块”,webpack内部贯彻了一套财富加载机制,可以把想css,图片等财富等有依靠关系的“模块”加载。那跟大家应用requirejs那种单纯处理js大大不一致。而那套加载机制,通过贰个个loader来完结。

 

JavaScript

// webpack.config.js module.exports = { entry: { entry: ‘./index.jsx’,
}, output: { path: __dirname, filename: ‘[name].min.js’ }, module:
{ loaders: [ {test: /\.css$/, loader: ‘style!css’ }, {test:
/\.(jsx|js)?$/, loader: ‘jsx?harmony’, exclude: /node_modules/},
{test: /\.(png|jpg|jpeg)$/, loader: ‘url-loader?limit=10240’} ] } };

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
// webpack.config.js
module.exports = {
    entry: {
     entry: ‘./index.jsx’,
    },
    output: {
        path: __dirname,
        filename: ‘[name].min.js’
    },
    module: {
        loaders: [
            {test: /\.css$/, loader: ‘style!css’ },
            {test: /\.(jsx|js)?$/, loader: ‘jsx?harmony’, exclude: /node_modules/},
            {test: /\.(png|jpg|jpeg)$/, loader: ‘url-loader?limit=10240’}
        ]
    }
};

上边一份简单的webpack配置文件,留意loaders的配置,数组内二个object配置为一种模块能源的加载机制。test的正则为协作文件规则,loader的为匹配到文件将由什么加载器处理,八个总计机之间用相隔,处理顺序从右到左。

 

style!css,css文件通过css-loader(处理css),再到style-loader(inline到html)的加工处理流。

jsx文件通过jsx-loader编译,‘?’开启加载参数,harmony支持ES6的语法。

图表能源通过url-loader加载器,配置参数limit,控制少于10KB的图纸将会base64化。

 财富文件如何被require?

JavaScript

// 加载组件自己css require(‘./slider.css’); // 加载组件爱惜的模块 var
Clip = require(‘./clipitem.js’); // 加载图片财富 var spinnerImg =
require(‘./loading.png’);

1
2
3
4
5
6
// 加载组件自身css
require(‘./slider.css’);
// 加载组件依赖的模块
var Clip = require(‘./clipitem.js’);
// 加载图片资源
var spinnerImg = require(‘./loading.png’);

在webpack的js文件中大家除了require我们例行的js文件,css和png等静态文件也得以被require进来。大家经过webpack命令,编译之后,看看输出结果什么:

JavaScript

webpackJsonp([0], { /* 0 */ /***/ function(module, exports,
__webpack_require__) { // 加载组件自个儿css
__webpack_require__(1); // 加载组件看重的模块 var Clip =
__webpack_require__(5); // 加载图片能源 var spinnerImg =
__webpack_require__(6); /***/ }, /* 1 */ /***/
function(module, exports, __webpack_require__) { /***/ }, /* 2
*/ /***/ function(module, exports, __webpack_require__) {
exports = module.exports = __webpack_require__(3)();
exports.push([module.id, “.slider-wrap{\r\n position: relative;\r\n
width: 100%;\r\n margin: 50px;\r\n background:
#fff;\r\n}\r\n\r\n.slider-wrap li{\r\n text-align:
center;\r\n line-height: 20px;\r\n}”, “”]); /***/ }, /* 3 */
/***/ function(module, exports) { /***/ }, /* 4 */ /***/
function(module, exports, __webpack_require__) { /***/ }, /* 5
*/ /***/ function(module, exports) { console.log(‘hello, here is
clipitem.js’) ; /***/ }, /* 6 */ /***/ function(module, exports)
{ module.exports = “……” /***/ }
]);

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
webpackJsonp([0], {
/* 0 */
/***/ function(module, exports, __webpack_require__) {
          // 加载组件自身css
          __webpack_require__(1);
          // 加载组件依赖的模块
          var Clip = __webpack_require__(5);
          // 加载图片资源
          var spinnerImg = __webpack_require__(6);
/***/ },
/* 1 */
/***/ function(module, exports, __webpack_require__) {
 
/***/ },
/* 2 */
/***/ function(module, exports, __webpack_require__) {
          exports = module.exports = __webpack_require__(3)();
          exports.push([module.id, ".slider-wrap{\r\n position: relative;\r\n width: 100%;\r\n margin: 50px;\r\n background: #fff;\r\n}\r\n\r\n.slider-wrap li{\r\n text-align: center;\r\n line-height: 20px;\r\n}", ""]);
 
/***/ },
/* 3 */
/***/ function(module, exports) {
 
/***/ },
 
/* 4 */
/***/ function(module, exports, __webpack_require__) {
/***/ },
 
/* 5 */
/***/ function(module, exports) {
          console.log(‘hello, here is clipitem.js’) ;
/***/ },
/* 6 */
/***/ function(module, exports) {
          module.exports = "……"
/***/ }
]);

webpack编译之后,输出文件视乎乱糟糟的,但骨子里每二个能源都被封装在1个函数体内,并且以编号的款型标记(注释)。那么些模块,由webpack的__webpack_require__其中方法加载。入口文件为编号0的函数index.js,可以看出__webpack_require__加载其余编号的模块。

css文件在编号1,由于应用css-loader和style-loader,编号1-4都以拍卖css。其中编号2大家能够看大家的css的string体。最后会以内联的法子插入到html中。

图表文件在编号6,可以看出exports出base64化的图纸。

 组件一体输出

JavaScript

// 加载组件本身css require(‘./slider.css’); // 加载组件看重的模块 var
React = require(‘react’); var Clip = require(‘../ui/clipitem.jsx’); //
加载图片财富 var spinnerImg = require(‘./loading.png’); var Slider =
React.createClass({ getInitialState: function() { // … },
componentDidMount: function(){ // … }, render: function() { return (
<div> <Clip data={this.props.imgs} /> <img
className=”loading” src={spinnerImg} /> </div> ); } });
module.exports = Slider;

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
// 加载组件自身css
require(‘./slider.css’);
// 加载组件依赖的模块
var React = require(‘react’);
var Clip = require(‘../ui/clipitem.jsx’);
// 加载图片资源
var spinnerImg = require(‘./loading.png’);
var Slider = React.createClass({
    getInitialState: function() {
        // …
    },
    componentDidMount: function(){
        // …
    },
    render: function() {
        return (
            <div>
               <Clip data={this.props.imgs} />
               <img className="loading" src={spinnerImg} />
            </div>
        );
    }
});
module.exports = Slider;

假设说,react使到html和js合为紧凑。

那就是说丰硕webpack,两者结合一起的话。js,css,png(base64),html
所有web财富都能合成1个JS文件。那多亏这套方案的基本所在:零件独立一体化。假如要引用一个零部件,仅仅require('./slider.js') 即可已毕。

 

投入webpack的模块加载器之后,大家组件的加载难点,内聚难点也都事业有成地消除掉

“能源高内聚”—— (100%) 所有财富得以一js出口

“可相互结合”—— (100%)  可组合可依靠加载

 

 CSS模块化实践

很高兴,你能翻阅到那里。近期大家的零件完结度非凡的高,财富内聚,易于组合,效率域独立互不污染。。。。等等美高梅开户网址 14,视乎CSS模块的落成度有欠缺。

那么近期组件完毕度来看,CSS功能域其实是全局性的,并非组件内部独立。下一步,我们要做得就是如何让大家组件内部的CSS功效域独立。

这儿只怕有人马上跳出,大喊一句“德玛西亚!”,哦不,应该是“用sass啊傻逼!”。可是项目组件化之后,组件的其中封装已经很好了,其里面dom结构和css趋向简单,独立,甚至是破破烂烂的。LESS和SASS的一体式样式框架的筹划,他的嵌套,变量,include,函数等丰硕的效劳对于全部大型项目标样式管理尤其实用。但对于2个职能单一组件内部样式,视乎就变的多少顶牛。“无法为了框架而框架,合适才是最好的”。视乎原生的css能力已经满意组件的体制要求,唯独就是上边的css效用域难点。

 

那里本身付诸思考的方案:
classname随便写,保持原生的办法。编译阶段,依照组件在品种路线的唯一性,由【组件classname+组件唯一路径】打成md5,生成全局唯一性classname。正当自个儿要写二个loader完成自己的想法的时候,发现歪果仁已经早在先走一步了。。。。

此处具体方案参考小编从前博客的译文:

事先我们谈论过JS的模块。今后经过Webpack被加载的CSS能源叫做“CSS模块”?小编觉得依旧有难题的。以往style-loader插件的落到实处精神上只是开创link[rel=stylesheet]要素插入到document中。那种作为和平常引入JS模块至极不一样。引入另1个JS模块是调用它所提供的接口,但引入三个CSS却并不“调用”CSS。所以引入CSS本人对于JS程序来说并不存在“模块化”意义,纯粹只是表明了一种财富尊敬——即该零件所要完毕的功能还索要一些asset。

所以,那位歪果仁还增添了“CSS模块化”的概念,除了上面的大家需求有的效率域外,还有为数不少成效,那里不详述。具体参考原文 

可怜赞的有些,就是cssmodules已经被css-loader收纳。所以大家不需求倚重额外的loader,基本的css-loader开启参数modules即可

JavaScript

//webpack.config.js … module: { loaders: [ {test: /\.css$/, loader:
‘style!css?modules&localIdentName=[local]__[name]_[hash:base64:5]’
}, ] } ….

1
2
3
4
5
6
7
8
//webpack.config.js
…  
    module: {
        loaders: [
            {test: /\.css$/, loader: ‘style!css?modules&localIdentName=[local]__[name]_[hash:base64:5]’ },
        ]  
    }
….

modules参数代表开启css-modules作用,loaclIdentName为设置大家编译后的css名字,为了有利于debug,大家把classname(local)和零部件名字(name)输出。当然能够在结尾输出的本子为了节约提交,仅仅使用hash值即可。此外在react中的用法大概如下。

JavaScript

var styles = require(‘./banner.css’); var Banner = new
React.createClass({ … render: function(){ return ( <div> <div
className={styles.classA}></div> </div> ) } });

1
2
3
4
5
6
7
8
9
10
11
var styles = require(‘./banner.css’);
var Banner = new React.createClass({
    …
    render: function(){
        return (
            <div>
                <div className={styles.classA}></div>
            </div>
        )
    }
});

终极那里关于出于对CSS一些思考,

至于css-modules的别样功用,作者并不打算采取。在其间分享【大家竭尽所能地让CSS变得复杂】中提及:

咱俩项目中多数的CSS都不会像boostrap那样须要变量来安装,身为一线开发者的咱们大致可以感受到:设计师们改版UI,相对不是简单的换个色或改个间距,而是万物更新的全新UI,那纯属不是2个变量所能消除的”维护性“。

相反项目实战进程中,真正要缓解的是:在本子迭代进程中那么些淘汰掉的超时CSS,大批量地堆放在类型当中。我们像极了家中的欧巴酱不舍得丢掉没用的东西,因为那不过大家使用sass或less编写出具有高度的可维护性的,肯定有复用的一天。

那些堆积的过期CSS(or
sass)之间又有部分器重,一部分逾期失效了,一部分又被新的体制复用了,导致没人敢动这一个历史样式。结果现网项目迭代还带着大量两年前没用的样式文件。

组件化之后,css的方式同样被改良了。或然postcss才是你今后手上最适合的工具,而不在是sass。

 

到那边,大家到底把组件化最后二个难题也化解了。

“功能域独立”—— (100%) 如同shadowDom功能域独立

 

到此处,大家得以开一瓶82年的Pepsi-Cola,好好庆祝一下。不是吗?

美高梅开户网址 15

 

 组件化之路还在连续

webpack和react还有过多新卓殊重大的表征和功能,介于本文仅仅围绕着组件化的为着力,没有各样解说。其余,配搭gulp/grunt补充webpack创设能力,webpack的codeSplitting,react的机件通讯难点,开发与生育条件安插等等,都是整整大型项目方案的所必须的,限于篇幅难题。可以等等小编更新下篇,或大家可以自行查阅。

唯独,不得不再安利一下react-hotloader神器。热加载的费用形式相对是下一代前端开发必备。严厉说,只要没有了热加载,作者会很泼辣地扬弃那套方案,即便那套方案再怎么完美,作者都讨厌react必要5~6s的编译时间。但是hotloader能够在自个儿不刷新页面的气象下,动态修改代码,而且不单单是样式,连逻辑也是即时生效。

美高梅开户网址 16

如上在form表单内。使用热加载,表单不需求重新填写,修改submit的逻辑立时见效。那样的成本功效真不是增加仅仅三个品位。必须安利一下。

 

唯恐你发觉,使用组件化方案将来,整个技术栈都被更新了一番。学习话费也不少,并且可以预见到,基于组件化的前端还会众多欠缺的标题,例如质量优化方案须求再行考虑,甚至最大旨的机件可复用性不必然高。前边十分长一段时间,需要大家不停磨砺与优化,探求最优的前端组件化之道。

至少我们得以想象,不再担心本人写的代码跟有个别何人哪个人争持,不再为找某段逻辑在八个公文和形式间不停,不再copy一片片逻辑然后改改。大家每便编写都以可采用,可整合,独立且内聚的组件。而各样页面将会由3个个嵌套组合的零部件,互相独立却相互功用。

 

对此这么的前端以后,有所指望,不是很可以吗

由来,谢谢您的翻阅。

1 赞 6 收藏 1
评论

美高梅开户网址 17

摘要


  • 后端懒人一枚,临时必要搞个管理端,从前在全校有用jquery+bootstrap做过类似的前端项目,然则今后追思,旧代码如屎一坨,更别提维护了。有了此前的训诫,碰到那么些需要就控制认真做做。
  • 对前者如故略小白,写得不得了,还望多多指教。
  • 已形成的档次选拔的结合是react+webpack+redux-form(扶助ie8)

原科目内容详见精益
React
学习指南,那只是本身在攻读进程中的一些读书笔记,个人觉得该课程讲解通俗,比近日多数学科专业、系统。

前言

1位处理器前辈曾说过:

Controlling complexity is the essence of computer programming

1
Controlling complexity is the essence of computer programming

乘势前端开发复杂度的渐渐升级,组件化开发应运而生,并乘胜 FIS、React
等脍炙人口框架的产出四处开花。这一历程一样发出在美团,面临工作范围的飞速发展和工程师团队的接连不断增加,大家历经引入组件化解决能源整合难点、逐步提高组件功效推进开发作用、重新制作新一代组件化方案适应全栈开发和共享共建等阶段,努力“controlling
complexity”。本文将介绍大家组件化开发的实施进程。

背景与题材


  • Header,Banner,Footer等前端布局的集体部分,重复编写代码或倚靠php等后端语言举办渲染,会造成前后端代码高耦合。
  • 舍弃全局变量的存在,给品种增添带来未知问题。
  • 随着项目花费持续拓展,项目也会变得臃肿,不佳的连串协会会使保证工作尤为困难。复杂的页面必要控制越多处境,并依据景况的变化,执行越多相应的逻辑。
  • 本着管理端,举行ie8+的合作。
  • 动用的第三方库较多,前端发送的静态文件(js,css)请求较多,导致网页加载速度较慢。
  • 治本平台的体裁多有同一,若样式定制开发,会爆发不须要的工作量。

1.1 Introduction to React

组件化 1.0:能源整合

在美团早期,前端能源是安分守己页面只怕类似事情页面集合的款型开展集体的。例如
order.js 对应订单相关页面的相互,account.css
对应账户相关页面的体裁。那种措施在过去的较长一段时间内,持续帮衬了全套项目标不奇怪化推进,居功至伟。

美高梅开户网址 18

乘胜工作规模的伸张和开发团队的壮大,那套机制稳步展现出它的一些相差:

  • 能源冗余页面的逐步增多,交互的逐步复杂化,导致对应的 css 和 js
    都有大幅度升高,进而出现为了借助有些 js
    中的三个函数,要求加载整个模块,大概为了利用有些 css
    中的部分样式正视整个 css,冗余财富较多
  • 对应提到不直观没有明确的应和规则,导致的2个标题是修改某些业务模块的
    css 或许 js 时,几乎只能借助 grep。靠人来保证页面模块 html、css 和
    js 之间的依靠关系,简单犯错,平常出现内容已经去除然则 css 或 js
    还存在的题材
  • 难于单元测试以页面为最小粒度进行能源整合,不相同作用的业务模块彼此影响,复杂度太高,自动化测试难以推进

2011 年初叶,在调研了 FIS、BEM
等方案以后,结合美团开发框架的实际上,我们开头完毕了一套轻量级的组件化开发方案。首要的精益求精是:

  • 以页面效果组件为单位聚合前端能源
  • 自行加载符合约定的 css、js 资源
  • 将事情数据到渲染数据的更换进程独立出来

美高梅开户网址 19

举例来说来说,美团顶部的搜索框就被完毕为3个零件。

美高梅开户网址 20

代码构成:

www/component/smart-box/ ├── smart-box.js # 交互 ├── smart-box.php #
渲染数据生产、组件配置 ├── smart-box.scss # 样式 ├── smart-box.tpl #
内容 └── test ├── default.js # 自动化测试 └── default.php # 单测页面

1
2
3
4
5
6
7
8
www/component/smart-box/
├── smart-box.js    # 交互
├── smart-box.php   # 渲染数据生产、组件配置
├── smart-box.scss  # 样式
├── smart-box.tpl   # 内容
└── test
    ├── default.js  # 自动化测试
    └── default.php # 单测页面

调用组件变得丰富不难:

JavaScript

echo View::useComponent(‘smart-box’, [ ‘keyword’ => $keyword ]);

1
2
3
echo View::useComponent(‘smart-box’, [
    ‘keyword’ => $keyword
]);

对照以前,可以看出组件化的一对性情:

  • 按需加载只加载须求的前端财富
  • 对应提到十显然显组件所需要的前端能源都在同等目录,任务鲜明且唯壹,对应提到明确
  • 不难测试组件是具有独立显示和互相的细小单元,可利用 Phantom
    等工具自动化测试

除此以外,由于前端财富集中开展调度,组件化也为高阶品质优化提供了上空。例如落实组件级其他BigRender、通过数据解析举行能源的合并加载等等。

关键词


据悉此前对背景和题材的分析,这一次做前端的严重性就是做到以下二个根本词。

  • 模块化
  • 组件化
  • 工程化

1.1.1 What is React?

React IS A JAVASCRIPT LIBRARY FOR BUILDING USER INTERFACES , INCLUDING

  1. React.js
  2. ReactRenders: ReactDOM / ReactServer / ReactCanvas
  3. Flux方式及其已毕
  4. React Components
  5. React Native
  6. GraphQI + Relay

组件化 2.0:趋于成熟

组件化 1.0
上线后,由于不难易用,很快得到工程师的认可,并早先在各项事情中运用起来。新的必要接连不断,平素频频到
2015 年终,那几个等级我们称为组件化 2.0。上边介绍下第一的几个立异。

难点及对应化解方案


1.1.2 Basic concept in React

  1. React.js
    React.js 是 React 的主干库,在动用中务必先加载核心库

    <script src="http://facebook.github.io/react/js/react.js"</script>
    <script src="http://facebook.github.io/react/js/react-dom.js"></script>
    <script src="https://cdnjs.cloudflare.com/ajax/libs/babel-core/5.8.23/browser.min.js"></script>
  1. ReactDOM.js
    DOM渲染器,为了在 web 页面中显得开发的零部件,需求调用 ReactDOM.render
    方法, 首个参数是 React 组件,第三个参数为 HTMLElement。
  2. JSX
    JSX 是 React 自定义的语法,最终 JSX 会转化为 JS 运转于页面当中
  3. Components
    Component是 React 中的大旨概念,页面当中的富有因素都以经过 React
    组件来宣布。
  4. Virtual DOM
    React 抽象出来的虚构 DOM 树,虚拟树是 React 高质量的第一。
  5. one-way reactive data flow
    React 应用的基本设计格局,数据流向自顶向下

Lifecycle

零件在高内聚的还要,往往必要揭发一些接口供外界调用,从而可以适应复杂的页面必要,例如提交订单页面要求在付出密码组件运转完毕后绑定提交时的检讨。Web
Components、React 等都选取了生命周期事件/方法,大家也是如出一辙。

组件的生命周期:

美高梅开户网址 213个零部件的全部生命周期包蕴:

  • init,开首化组件根节点和布署
  • fetch,加载 css 和 js 资源
  • render,内容渲染,默许的渲染内容措施是 BigRender
  • ready,进行数量绑定等操作
  • update,数据更新
  • destroy,解除所有事件监听,删除所有组件节点

零件提供 pause、resume 方法以方便进行生命周期控制。各种阶段采用 Promise
串行举办,异步的军事管制更显然。使用自定义语义事件,在改动暗中同意行为、组件间通讯上丰裕利用了
YUI 强大的自定义事件连串,有效下跌了花费爱戴花费。

举个例证,页面起先化时组件的启航进度实际上也是依靠生命周期完毕的:

JavaScript

var afterLoadList = []; Y.all(‘[data-component]’).each(function
(node) { var component = new Y.mt.Component(node); // 绑定 init
生命周期事件,在 init 暗中认同行为落成后举办回调 component.after(‘init’,
function (e) { // 要是安排了延期运营 if (e.config.afterLoad) { //
暂停组件生命周期 e.component.pause(); // 压入延迟运维数组
afterLoadList.push(e.component); } }); // 发轫进入生命周期
component.start(); }); Y.on(‘load’, function () { // 在页面 load
事件暴发时回升组件生命周期 afterLoadList.forEach(function (component) {
component.resume(); }); });

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
var afterLoadList = [];
Y.all(‘[data-component]’).each(function (node) {
    var component = new Y.mt.Component(node);
    // 绑定 init 生命周期事件,在 init 默认行为完成后执行回调
    component.after(‘init’, function (e) {
        // 如果配置了延迟启动
        if (e.config.afterLoad) {
            // 暂停组件生命周期
            e.component.pause();
            // 压入延迟启动数组
            afterLoadList.push(e.component);
        }
    });
    // 开始进入生命周期
    component.start();
});
 
Y.on(‘load’, function () {
    // 在页面 load 事件发生时恢复组件生命周期
    afterLoadList.forEach(function (component) {
        component.resume();
    });
});

回过头来看,引入生命周期除了带来扩张性外,更首要的是理顺了组件的相继阶段,有助于更好的知道和选拔。

题材:前后端代码高耦合+全局变量

1.1.3 Features of React

  1. Component的构成格局
    组合模式有时候又叫做部分-整体模式,它使我们树型结构的问题中,模糊了简单元素和复杂元素的概念,客户程序可以向处理简单元素一样来处理复杂元素,从而使得客户程序与复杂元素的内部结构解耦。
    React 就是基于组合形式,
    无论是应用等级或然二个表单亦或然二个按钮都说是贰个组件,
    然后基于组件的三结合构建整个应用,这样的协会一直是前端界想要却迟迟不来的
    web component。
    基于组合格局的长处:
  • 打造可采取的零件:组件的费用可以形成集团的组件库,每一个事情的花费都能累积可采纳的机件。
  • 无学习障碍:天然符合 HTML 的布局,
    对前者开发者来说大约没有读书障碍。
  • 有着弹性的架构:组合格局很简短却很管用,可以创设简单的页面也能构建大型的前端选拔。
  • 源码高可维护性:开发只是工作中的一部分,应用的上线才是惊恐不已的梦的上马,很多重型应用因为复制的事务逻辑导致力不从心快捷响应工作需要,可尊崇性低。
  1. One way data flow
    Javascript 是脚本语言,不能像静态语言一样通过编译定位为题,想要清晰的定位到应用中的 bug 需要深入了解业务代码,对于大型前端应用来说,因为业务代码量很大且复杂,很难定位到 bug。 然而 React 的单向数据流的设计让前端 bug 定位变得简单, 页面的 UI 和数据的对应是唯一的,我们可以通过定位数据变化就可以定位页面展现问题。
  2. High efficiency
    据悉VirtualDOM算法可以让唯有须要变更的成分才去重渲染
  3. Separate frame design
    React.js 未来的本子现已将源码分开为 ReactDOM 和 React.js .
    那就表示 React 不仅仅可以在 web 端工作,
    甚至足以在服务端(nodejs),Native 端运转。
    与此同时, 咱们可以自定义本人的渲染器, 落成比如 Three.js,
    Pixi.js, D3.js 的 React 形式渲染。

Data Binding

数码绑定是我们望穿秋水已久的作用,将 View 和 ViewModel
之间的相互自动化无疑会省去工程师的大气小时。在组件化减弱关心点和下落复杂度后,完结数量绑定变得进一步只怕。

大家最终兑现的数目绑定方案主要参照了 Angular,通过在 html
节点上添加特定的性质申明绑定逻辑,js
扫描这一个内容并进行相应的渲染和事件绑定。当数码发生变化时,对应的内容总体重复渲染。

XHTML

<ul class=”addressList”> <li mt-bind-repeat=”addr in addrList”
mt-bind-html=”addr.text” > </li> </ul> <script>
Y.use([‘mt-bind’, ‘mt-scope’], function () {
Y.mt.bind.init(document.body); var scope =
Y.one(‘.addressList’).getScope(); // 将 scope.addrList
设置为1个数组,DOM 中校自行渲染其内容 scope.$set(‘addrList’, [ { text:
“first address” }, { text: “second address” } ]); }); </script>

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
<ul class="addressList">
    <li
        mt-bind-repeat="addr in addrList"
        mt-bind-html="addr.text"
    >
    </li>
</ul>
 
<script>
Y.use([‘mt-bind’, ‘mt-scope’], function () {
    Y.mt.bind.init(document.body);
    var scope = Y.one(‘.addressList’).getScope();
    // 将 scope.addrList 设置为一个数组,DOM 上将自动渲染其内容  
    scope.$set(‘addrList’, [
        { text: "first address" },
        { text: "second address" }
    ]);
});
</script>

使用性质申明绑定逻辑的利益是足以同时援救后端渲染,那对于美团团购那样的偏突显型业务是非凡须要的,用户可以很快看到页面内容。

方案:模块化


第1、将前端的公家部分提取出来,举行模块化编程;
下一场,划定功效域,不相同模块间功用域不共享,那样就足以做到,公共代码模块化,防止了全局变量的麻烦。

近年来,流行的模块化编程的工具有RequireJS, SeaJS,
Webpack,其中SeaJS是由阿里的公司开发的,在中华有外向的社区。Webpack由德意志人编写,体现了更强的类型工程化特点。
内部,RequireJS和SeaJS分别是速龙和CMD二种标准的表示。Intel推崇倚重前置,在概念模块的时候就要声明其借助的模块;CMD推崇异步保养加载的,只有在接纳有些模块的时候再去require。

RequireJS(AMD写法)

SeaJS(CMD写法)

而Webpack则属于集大成者,不仅包容两者书写规范,而且因采用异步IO及体系缓存,使Webpack在增量编译上更快。但实际,Webpack在实际开支中更加多的是担任编译者和打包者的角色,它结合React同样可以落到实处模块化开发,那么些大家在后面浮现。

1.1.4 Where can React apply?

  1. web 端
  2. 原生应用
    IOS、Android、Native 应用
    那要归功于 facebook 开源的 React Native。 基于 React Native ,
    大家将得以动用 jsx 来促成所有原生应用品质的 UI 运营于 IOS 和 android
    中,同时大家也足以通过 NW.js 恐怕 Electron 来兑现基于 React
    的桌面应用。
  3. 劳务器端渲染
    React 除了在 Web 和 Native 环境以外, 也足以因此 React
    实今后劳动器端渲染出 HTML。

Flux

贯彻多少绑定后,大家不得不面对其余一个题材:怎样一同八个零部件间的数量。因为有些组件的数目变化,很有恐怕滋生其余零件的浮动。例如当修改购买数量,总金额会转变,而总金额超越500 后,还索要出示大额消费指示。

为了缓解那个题材,大家引入了
Flux,使用全局音讯总线的思绪开展跨组件交互。

比如说因为互相复杂而直白让大家那些头痛的项目购买页,在运用组件 + Flux
重构后,各模块之间的交互更是清楚:

美高梅开户网址 22

此外方面的改良还有为数不少,蕴含引入模板引擎 LightnCandy约束模板逻辑、帮忙组件任意嵌套、协理异步加载并自行起头化等。

随着组件化 2.0
的日趋健全,基本已经可以从容应对平时支付,在效能和品质方面都上了八个台阶。

标题:不好的门类协会,控制更加多的气象

1.2 JSX Grammar

组件化 3.0:重启征程

光阴的车轮滚滚前行,二零一四 年终,大家蒙受有的新的机遇和挑战:

  • 依照 Node 的全栈开发形式开始利用,前后端渲染有了越来越多的可能
  • YUI 为止维护,需求一套新的财富管理方案
  • 新工作不断加码,须求找到一种组件共享的情势,幸免重复造轮子

组成从前的执行,以及在这一经过中国和扶桑渐积累的对业内方案的咀嚼,大家指出了新的组件化方案:

  • 基于 React 开发页面组件,使用 NPM 举办分发,方便共建共享
  • 按照 Browserify 二次开发,建设能源打包工具 Reduce,方便浏览器加载
  • 建设适应组件化开发格局的工程化开发方案
    Turbo,方便工程师将零件应用于工作支付中

方案:组件化


组件化建立在模块化的根基上,模块化是对能源的管理,组件化是对作业逻辑的管制,各自要有两样的前行方向。经调研,NPM,Webpack(或Browserify)和React的组合会便于我们对项目结构有壹个完好无缺的认识。在那个基础上,我们便能丰硕有利地创建一个布局特别鲜明的品种。

而针对性不相同组件所负有的频频变动的地方,通过对Flux和Redux的学习和运用,可以方便我们对于复杂组件状态处理的建制,有愈来愈的精通。(但因为我到底不是前者,且较懒,关于redux,未做过多读书,直接上手redux-form是为着快捷支付管理端表单。)那里要提一下,redux-form充足强大,官方已交付很多例子,那里不再熬述。

Flux

Redux

1.2.1 Prepare running environment

  1. Babel REPL
    Babel
    REPL
    一贯在 REPL 中写 JSX 语法,可以实时的查阅编译后的结果。
  2. JSFiddle
    在线格局 React
    Fiddle
  3. Local development

React

在组件化 2.0 的经过中,大家发现众多效率和 React 重合,例如 Data
Binding、Lifecycle、前后端渲染,甚至平昔借鉴的 Flux。除此之外,React
的函数式编程思想、增量更新、兼容性卓越的事件种类也让我们越发向往。借着前端全栈开发的节骨眼,大家最先考虑基于
React 举办零部件化 3.0 的建设。

问题:IE8兼容

1.2.2 JSX Grammar

创设 JSX 语法的本来面目目标是为了接纳基于 xml 的主意发挥组件的嵌套,保持和
HTML 一致的协会,语法上除了在描述组件上相比尤其以外,其他和普通的
Javascript 没有区分。 并且最终具备的 JSX 都会编译为原生 Javascript。

  1. jsx componments
    JSX 组件分为 HTML 组件和 React 组件
    HTML 组件就是 HTML 中的原生标签, 如:

function render() {
       return  <p> hello, React World </p>
}
function render() {
       return <ul> 
                 <li>list item 1</li>
                 <li>list item 2</li>
              </ul>
}

React 组件就是自定义的组件,如

// 定义一个自定义组件
var CustomComponnet = React.createClass({
        render: function() {
            return <div> custom component </div>
  }
});
// 使用自定义组件
function render() {
        return <p> <CustomComponent/> </p>
}
  1. properties of jsx components
    和 html 一样,JSX 中组件也有总体性,传递属性的不二法门也一致
    对于 HTML 组件

 function render() {
      return  <p title="title" >hello, React, world </p>
 }

假定是 React 组件可以定义自定义属性,传递自定义属性的办法:

  function render() {
      return <p> <CustomComponent customProps="data"/> </p>
      }
}

质量即可以是字符串,也足以是随机的 Javascript 变量,
传递方式是将变量用花括号, eg:

  function render() {
      var data = {a: 1, b:2};
      return <p> <CustomComponent customProps={data}/> </p>
  }

内需专注的地点上,属性的写法上和 HTML 存在分歧,在写 JSX
的时候,所有的习性都以驼峰式的写法,重借使由于规范的原由,驼峰式是
Javascript 的规范写法,并且 React 底层是将品质直接对应到原生 DOM
的品质,而原生 DOM 的质量是驼峰式的写法,那里也可以知道为何类名不是
class 而是 className 了, 又因为 class 和 for 依旧 js 关键字,所以 jsx
中:

class => className
for => htmlFor

除了相比非凡的地方是 data-*aria-*两类天性是和 HTML 一致的。

  1. curly braces
  • 来得文本

  function render() {
        var text = "Hello, World"
        return <p> {text} </p>
  }
  • 运算

funtion render() {
      var text = text;
      var isTrue = false;
      var arr = [1, 2, 3];
      return <p>
        {text}
        {isTrue ? "true" : "false"}
        {arr.map(function(it) {
           return  {it} 
        })}
        </p>
  }
  1. 限制规则
    render 方法重返的组件必须是有且只有1个根组件,错误情状的例证

  // 无法编译通过,JSX 会提示编译错误
  function render() {
    return (<p> .... </p>
           <p> .... </p>)
  }
  1. 组件命名空间
    JSX 可以通过命名空间的主意使用组件,
    通过命名空间的法门得以缓解相同名称不一致用途组件争论的题材。如:

  function render() {
    return <p>
           <CustomComponent1.SubElement/>
           <CustomComponent2.SubElement/>
           </p>
  }

NPM + Reduce

NPM + Reduce 构成了我们新的能源管理方案,其中:

  • NPM
    负责组件的发布和装置。可以认为是“分”的长河,粒度越小,重用的只怕性越大
  • Reduce
    负责将页面财富举行包装。可以认为是“合”的长河,让浏览器更快地加载

三个卓越的零件包:

smart-box/ ├── package.json # 组件包元消息 ├── smart-box.jsx # React
Component ├── smart-box.scss # 样式 └── test └── main.js # 测试

1
2
3
4
5
6
smart-box/
├── package.json    # 组件包元信息
├── smart-box.jsx   # React Component
├── smart-box.scss  # 样式
└── test
    └── main.js     # 测试

NPM 暗中认可只扶助 js 文件的管制,大家对 NPM 中的 package.json
进行了增加,伸张了 style 字段,以使打包工具 Reduce 也可以对 css 和 css
中援引的 image、font 举行辨别和拍卖:

XHTML

{ “style”: “./smart-box.scss” }

1
2
3
{
    "style": "./smart-box.scss"
}

假使在页面中 require 了 smart-box,经过 Reduce 打包后,js、css
甚至图片、字体,都会并发在浏览器中。

JavaScript

var SmartBox = require(‘@mtfe/smart-box’); // 页面 var IndexPage =
React.createClass({ render: function () { return ( <Header>
<SmartBox keyword={ this.props.keyword } /> </Header> … );
} }); module.exports = IndexPage;

1
2
3
4
5
6
7
8
9
10
11
12
13
var SmartBox = require(‘@mtfe/smart-box’);
// 页面
var IndexPage = React.createClass({
    render: function () {
        return (
            <Header>
                <SmartBox keyword={ this.props.keyword } />
            </Header>
            …
        );
    }
});
module.exports = IndexPage;

总体思路和零部件化 1.0 如出一辙,却又那么差别。

方案:React(0.14.x)


最一初阶调研的时候,因为ie兼容的顾虑,所以,调研了相比较流行的React,Angular,Vue以及小众的Avalon。其中后多少个都是MVVM框架(宗旨是多少双向绑定)。关于包容,Angular早在1.3本子就扬弃了对ie8的援助,Vue就没打算协助ie8,可面对中国1/5的ie8用户,只能在Avalon和React做接纳。

MVVM

Avalon是由去哪儿前端架构师“司徒正美”开发的一款基于虚拟DOM与质量胁制的精工细作、易用、高品质的前端MVVM框架,适用于各类气象,包容种种古老刁钻浏览器,吸收最新的技术成果,能很快堆砌组件与运用。它竟然扶助到IE6+。

avalon2

包容性和总体性好外,缺点就是除了文档和较小的论坛,你搜不到过多的素材,那对早先时期学习,或许是别人维护,是很不便利的。而且撰稿人重写了成百上千js的主导措施,那造成即使出错,除了找作者和本身改源码外,很大概无处查询。

末段拔取React(0.14.x),除了包容外,还有诸如社区活泼,能源多,最要害的是有Facebook做支撑。当然,它的ie8帮忙需求展开多插件配置和协理,在本人github上关于react学习的有关推行里有连锁配置文件。其中,package.json和webpack.config.js要求优质看下,其次就是留意项目结构(因为用了react-router,项目协会相对清晰),其他包容相关可以由此上边链接举办学习。
https://github.com/xcatliu/react-ie8
http://www.aliued.com/?p=3240

1.2.3 Understand JSX

  1. JSX 的编译格局
    • 在 HTML 中引入 babel 编译器, 如上 Hello World 程序中千篇一律。
    • 离线编译 JSX,通过 babel 编译 JSX,细节大家将在其次章中助教。
  2. JSX 到 JS 的转化
    Hello World 程序转化为 JS 的代码如下:

  var Hello = React.createClass({
       displayName: 'Hello',
       render: function() {
          return React.createElement("div", null, "Hello ", this.props.name);
     }
  });
  ReactDOM.render(
       React.createElement(Hello, {name: "World"}),
       document.getElementById('container')
  );

Turbo

一味化解分发和包裹的标题还不够,业务支出进度如若变得繁琐、难以
Debug、品质低下的话,只怕不会碰到工程师欢迎。

为了消除那些标题,大家在 Node
框架的底蕴上,提供了一体系中间件和开发工具,逐步打造对组件友好的前端工程化方案
Turbo。首要有:

  • 支持前后端同构渲染,让用户更早看到内容
  • 简化 Flux 流程,数据流特别清晰易维护
  • 引入 ImmutableJS,有限支撑 Store 以外的数额不可变
  • 选用 cursor 机制,保障数据修改/获取同步
  • 支撑 Hot Module Replacement,革新开发流自动化

透过那么些创新,一线工程师可以一本万利的采用各个零件,专注在工作本人上。开发框架层面的支撑也反过来促进了组件化的进化,我们更愿意使用一多元组件来营造页面效果。

题材:前端发送的静态文件(js,css)请求多

1.3 React Components

小结

发觉痛点、分析调研、应用立异的缓解难题思路在组件化开发执行中连连使用。历经七个大本子的朝梁暮陈,组件化开发情势有效化解了业务发展牵动的复杂度提高的压力,并作育工程师具备小而美的工程思想,形成共建共享的特出氛围。毫无疑问,组件化那种“分而治之”的沉思将会长期地影响和促进前端开发格局。我们前几天早已准备好,迎接新的机遇和挑衅,用技术的持续创新进步工程师的幸福感。

1 赞 5 收藏
评论

美高梅开户网址 23

方案:Webpack


Webpack援助转译,打包,压缩等功能。转译可以将react的jsx语言转成js,es6写法转成es5(半数以上浏览器包容)等,同时,将洋洋文本的转译结果及器重关系等,打包压缩到二个文件中,只需四次呼吁,便也等于加载了多少个公文及其关系,极大地升级了前者页面加载速度。

1.3.1 Create a component

  • 创设二个 React 组件的法子为,调用 React.createClass
    方法,传入的参数为1个对象,对象必须定义贰个 render 方法,render
    方法重回值为组件的渲染结构,也可以驾驭为3个组件实例(React.createElement
    工厂方法的重返值),重临值有且不得不为壹个零部件实例,恐怕返回null/false,当再次回到值为 null/false 的时候,React 内部通过
    <noscript/> 标签替换

var MyComponent = React.createClass({
        render: function() {
            return <p>....</p>;
        }
});
  • Component命名空间
    可以看来 React.createClass 生成的机件类为1个 Javascript 对象。
    当大家想设置命名空间组件时,可以在组件上面添加子组件

 MyComponent.SubComponent = React.createClass({...});
 MyComponent.SubComponent.Sub = React.createClass({....});
  • 无状态组件
    除却可以通过 React.createClass
    来成立组件以外,组件也足以经过3个常常的函数定义,函数的重临值为组件渲染的结果

function StatelessComponent(props) {
        return  <div> Hello {props.name} </div>
}

无状态组件可以优化质量,因为其里面不会维护状态,React
内部不会有3个心心相印的机件实例,并且也绝非生命周期 hook

题材:样式须求不高,但重复性高

1.3.2 将零件渲染到 DOM 中

当创设好了组件过后,为了将零件渲染到 DOM 中突显出来,须要两个步骤:

  1. 在 HTML 中定义1个因素,设置 id 属性
  2. JSX 中调用 ReactDOM.render 方法, 第二个参数为
    组件,第贰个为刚刚概念的 DOM 成分

<!-- 定义的 DOM 元素 -->
<div id="example"></div>
<script type="text/babel">
    // 自定义组件 
    var MyComponent = React.createClass({
        render: function() {
            return <p>....</p>;
        }
    });
    // 调用 render 方法
    ReactDOM.render(
    <MyComponent/>,
    document.getElementById('example')
    );
</script>

方案:Ace Admin(ie8)


团结首要如故后端开发,管理端样式不是上下一心探究的基本点,故决定选取模板进行付出。通过调研,Ace
Admin和AdminLTE都是github上相比较受欢迎的前端样式模板,其中Ace协理ie8,故选拔前者用于实际开发。但AdminLTE相对赏心悦目,提供的体制接纳较多,大家也足以动用。

QQ截图20161027190718.jpg

1.3.3 States of components

React
的渲染结果是由组件属性和景况共同决定的,状态和性质的区分是,状态维护在组件内部,属性是由外部控制,我们先介绍组件状态相关细节:
决定状态的 API 为:

  • this.state:组件的当下地方
  • getInitialState:获取组件的始发状态,在组件加载的时候会被调用三次,再次回到值赋予
    this.state 作为开端值
  • this.setState:组件状态改变时,可以经过 this.setState 修改状
    • setState 方法援救按需修改,如 state 有多个字段,仅当 setState
      传入的目标涵盖字段 key 才会修改属性
  • 历次调用 setState 会导致重渲染调用 render 方法
  • 直接改动 state 不会重渲染组件

 var Switch = React.createClass({
        // 定义组件的初始状态,初始为关
        getInitialState: function() {
            return {
                open: false
            }
        },
        // 通过 this.state 获取当前状态
        render: function() {
            console.log('render switch component');
            var open = this.state.open;
            return <label className="switch"> 
                        <input type="checkbox" checked={open}/> 
                    </label>
        },
        // 通过 setState 修改组件状态
        // setState 过后会 React 会调用 render 方法重渲染
        toggleSwitch: function() {
            var open = this.state.open;
            this.setState({
                open: !open
            });
        }
    })

其三方库


React的考虑是由此操作虚拟dom完结指定任务。可是,在实际用度中,其实有很多方便的车轱辘并从未React化,还是要求通过外部引入的法门借助第三方库开展支付。
在这里,React组件的生命周期为引入三方库,提供了componentDidMount等办法,方便对第三方库开展加载。
此处顺便安利多少个库,如uploadify(flash版上传文件)、ZeroClipboard(点击即可复制)、dataTables(列表前端组件)

1.3.4 Properties of components

前边已经涉及过 React 组件可以传递属性给组件,传递格局和 HTML 中同样,
可以透过 this.props 获取组件属性
质量相关的 API 为:

  • this.props: 获取属性值
  • getDefaultProps:
    获取暗中认同属性对象,会被调用四回,当组件类创建的时候就会被调用,重返值会被缓存起来,当组件被实例化过后一旦传入的性质没有值,会再次来到默许属性值
  • this.props.children:子节点属性
  • propTypes: 属性类型检查

// props.name 表示代办事项的名称
    var TodoItem = React.createClass({
        render: function() {
            var props = this.props;
            return <div className="todo-item">
                        {props.name}
                    </div>
        }
    });
    ReactDOM.render(
        <TodoItem name="代办事项1"/>, 
         document.getElementById('example'));
  1. children属性
    零件属性中会有一个奇特属性 children ,表示子组件,
    照旧以上边一个零部件为例子,大家可以换一种方法定义 name:

var TodoItem = React.createClass({
        render: function() {
            var props = this.props;
            return <div className="todo-item">
                        {props.children}
                    </div>
        }
    });
    ReactDOM.render(
        <TodoItem>代办事项1</TodoItem>, 
         document.getElementById('example')); 
  1. 品质类型检查
    为了确保组件传递属性的不利, 大家可以透过定义 propsType
    对象来兑现对组件属性的严厉校验

var MyComponent = React.createClass({
        propTypes: {
            optionalArray: React.PropTypes.array,
            optionalBool: React.PropTypes.bool,
            optionalFunc: React.PropTypes.func,
            optionalNumber: React.PropTypes.number,
            optionalObject: React.PropTypes.object,
            optionalString: React.PropTypes.string,
            // 任何可以被渲染的包括,数字,字符串,组件,或者数组
            optionalNode: React.PropTypes.node,
            // React 元素
            optionalElement: React.PropTypes.element,
            // 枚举
            optionalEnum: React.PropTypes.oneOf(['News', 'Photos']),
            // 任意一种类型
            optionalUnion: React.PropTypes.oneOfType([
              React.PropTypes.string,
              React.PropTypes.number,
              React.PropTypes.instanceOf(Message)
            ]),
            // 具体类型的数组
            optionalArrayOf: React.PropTypes.arrayOf(React.PropTypes.number),
            // 具体类型的对象
            optionalObjectOf: React.PropTypes.objectOf(React.PropTypes.number),
            // 符合定义的对象
            optionalObjectWithShape: React.PropTypes.shape({
              color: React.PropTypes.string,
              fontSize: React.PropTypes.number
            }),
            requiredFunc: React.PropTypes.func.isRequired,
            requiredAny: React.PropTypes.any.isRequired,
            // 自定义校验
            customProp: function(props, propName, componentName) {}
        }
    });
  1. 天性传递的单向性
    大家曾经关系过 React 的单向数据流情势,数据的流动管道就是
    props,流动的样子就是组件的层级自定向下的大方向。所以三个组件是不大概修改本身的性质的,组件的习性一定是经过父组件传递而来(恐怕暗中同意属性)。
  2. 无状态组件属性
    对于无状态组件,可以添加 .propTypes 和 .defaultProps 属性到函数上。

代码相关


项目代码不太好开源,但在此以前本身写的多少个react相关的就学例子与项目契合度较高。
那里贴出react学习的事例链接,感兴趣的爱侣可以自行下载。欢迎各位互相互换。

1.3.5 组件的嵌套组合

在JSX 实例子中,当大家循环输出 todo 列表的时候,React
会指示对于循环输出的组件,需求有多个唯一的 key 属性。那一个标题标原由在于
React 的调停机制(Reconciliation)上。

  1. 什么叫调和?
    在历次数据更新之后,React 会重新调用 render
    渲染出新的零部件结构,新的布局采纳到 DOM 中的进程就叫做调和进程。
  2. 干什么须求调和?
    只要大家有一个输入组件,这几个时候大家正聚焦在输入框中,当修改值之后触及事件致使了多少变动,数据变动导致了重渲染,
    那一个时候输入框被替换成了新的 DOM。
    那个进度对用户来说应该是无感知的,所以这本来的聚焦状态应该被保留,
    那咋做到的呢? DOM 都被轮换了,输入状态,采纳景况为啥还是能保留。
    大家先不急着明亮 How,近年来只须求驾驭那就是调和进程。
    除却保留情形以外,调和进度还做了众多 DOM 优化。
    比如输出3个数组的时候,数据新扩张依然减少了刹那间,只怕数组项值改变了,实际上大家并未要求删除原来的
    DOM 结构,只须要修改 DOM 的值或然去除 DOM 就能完结重渲染。
    那就是干吗要有 key 属性,key 属性可以协助稳定 DOM
    与数组元素的关联,在重渲染的时候可以完成渲染优化。

1.4 Life circle of React Components

14.1 Components

React
中组件有协调的生命周期方法,简单了解可以为组件从出生(实例化) -> 激活 -> 销毁生命周期
hook。通过那一个 hook 方法可以自定义组件的表征。
除此之外,还足以设置有些附加的尺度配备。

美高梅开户网址 24

零件生命周期.png

这个生命周期方法都足以在调用 React.createClass 的参数对象中盛传,
此前运用过了有个别方法:
render getInitialState getDefaultProps propTypes

1.4.2 mixins

Type: array mixins
mixins 可以清楚为 React
的插件列表,通过那种情势在不一致组件之间共享方法数据依旧表现只需共享 mixin
就行,mixins 内定义的生命周期方法在组件的生命周期内都会被调用。

var MyMixin1 = {
    componentDidMount: function() {
        console.log('auto do something when component did mount');
    }
};

var MyMixin2 = {
    someMethod: function() {
        console.log('doSomething');
    }
};

var MyComponnet = React.createClass({
    mixins: [MyMixin1, MyMixin2],
    componentDidMount: function() {
        // 调用 mixin1 共享的方法
        this.someMethod();
    }
});

1.4.3 statics

Type:object statics
statics可以定义组件的类措施
React 的零件是 面向对象OOP 的切磋,MyComponent 是3个 class,class
分为类措施和实例方法,实例方法可以访问 this,
但是类形式不能够,所以大家不恐怕在 Class 中回到状态恐怕性质。

var MyComponent = React.createClass({
  statics: {
    customMethod: function(foo) {
      return foo === 'bar';
    }
  }
});

MyComponent.customMethod('bar');  // true

1.4.4 displayName

Type: string displayName
为了显示调试消息,各种组件都会有1个名称,JSX 在转为 JS 的时候自动的安装
displayName,当然我们也得以自定义 displayName

// Input (JSX):
var MyComponent = React.createClass({ });

// Output (JS):
var MyComponent = React.createClass({displayName: "MyComponent", });

1.4.5 生命周期方法

美高梅开户网址 25

零件生命周期.png

1.4.6 componentWillMount

void componentWillMount()

  • 规格:第一遍渲染阶段在调用 render 方法前会被调用
  • 作用:该方式在全路组件生命周期只会被调用两遍,所以可以运用该办法做一些组件内部的初叶化工作

1.4.7 componentDidMount

void componentDidMount()

  • 标准:第三次渲染成功过后,组件对应的 DOM 已经添加到页面后调用
  • 效益:这几个等级表示组件对应的 DOM
    已经存在,大家得以在那几个时候做一些依靠 DOM
    的操作依然别的的有个别如请求数据,和第三方库整合的操作。假若嵌套了子组件,子组件会比父组件优先渲染,所以这一个时候可以获取子组件对应的
    DOM。

1.4.8 componentWillReceiveProps(newProps)

void componentWillReceiveProps(
   object nextProps
)
  • 标准: 当组件获取新属性的时候,第五回渲染不会调用
  • 用处: 这一个时候可以依照新的天性来修改组件状态

    componentWillReceiveProps: function(nextProps) {
       this.setState({
           likesIncreasing: nextProps.likeCount > this.props.likeCount
      });
    }

注意:
那几个时候就算是获取新属性,但并无法确定属性一定改变了,例如一个零件被一再渲染到
DOM 中,如下边:

    var Component = React.createClass({
        componentWillReceiveProps: function(nextProps) {
            console.log('componentWillReceiveProps', nextProps.data.bar);
        },
        rener: function() {
            return <div> {this.props.data.bar} </div>
        }
    });

    var container = document.getElementById('container');
    var mydata = {bar: 'drinks'};
    ReactDOM.render(<Component data={mydata} />, container);
    ReactDOM.render(<Component data={mydata} />, container);
    ReactDOM.render(<Component data={mydata} />, container);

结果会输出一回component威尔ReceiveProps,即便属性数据没有改动,可是依然会调用
component威尔ReceiveProps 方法。

1.4.9 shouldComponentUpdate(nextProps, nextState)

boolean shouldComponentUpdate(
   object nextProps, object nextState
)
  • 规则: 接收到新属性恐怕新情景的时候在 render 前会被调用(除了调用
    forceUpdate 和初步化渲染以外)
  • 用处: 该方法让大家有时机决定是或不是重渲染组件,若是回到
    false,那么不会重渲染组件,借此可以优化利用质量(在组件很多的情况)。

1.4.10 componentWillUpdate

void componentWillUpdate(
  object nextProps, object nextState
)
  • 标准化:当组件确定要立异,在 render 以前调用
  • 用处:那么些时候可以规定一定会更新组件,可以执行更新前的操作
  • 瞩目:方法中不能运用 setState ,setState 的操作应该在
    component威尔ReceiveProps 方法中调用

1.4.11 componentDidUpdate

void componentDidUpdate(
  object prevProps, object prevState
)
  • 基准:更新被应用到 DOM 之后
  • 用处:可以实施组件更新之后的操作

1.4.12 生命周期与单向数据流

作者们驾驭 React
的着力格局是单向数据流,那不仅仅是对此组件级其他形式,在组件内部
的生命周期中也是应该符合单向数据的情势。数据从组件的性子流入,再结合组件的状态,流入生命周期方法,直到渲染甘休那都应当是贰个一边的长河,其间无法随意变更组件的动静。

美高梅开户网址 26

组件内部数据流.png

1.5 React & DOM

1.5.1获取DOM元素

DOM真正被添加到HTML中的hook为

  • componentDidMount
  • componentDidUpdate

在那七个 hook 函数中, 大家得以博得真正的 DOM 成分,React
提供的收获情势二种方法

  1. findDOMNode()
    透过 ReactDOM 提供的 findDOMNode 方法, 传入参数

var MyComponent = React.createClass({
    render: function() {
        return <div> .... </div>
    },
    componentDidMount: function() {
        var $root = ReactDOM.findDOMNode(this);
        console.log($root);
    }
})

必要留意的是此办法无法运用到无状态组件上

  1. Refs
    上面的主意只好获取到 root 成分,那尽管自身的 DOM
    有成百上千层级,笔者想拿到1个子级的因素呢?React 提供了 ref
    属性来贯彻那种要求。
    各个组件实例都有3个 this.refs 属性,会自行引用所有包涵 ref
    属性组件的 DOM

var MyComponent = React.createClass({
    render: function() {
        return  <div>
                    <button ref="btn">...</button>
                    <a href="" ref="link"></a>
                </div>
    },
    componentDidMount: function() {
        var $btn = this.refs.btn;
        var $link = this.refs.link;
        console.log($btn, $link);
    }
})

1.5.2 DOM事件

  1. 绑定事件
    在 React
    中绑定事件的法门很简短,只必要在要素中加上事件名称的属性已经相应的处理函数,如:

var MyComponent = React.creatClass({
    render: function() {
        return  <div>
                    <button onClick={this.onClick}>Click Me</button>
                </div>
    },
    onClick: function() {
        console.log('click me');
    }
});

事件名称和其余质量名称相同,遵循驼峰式命名。

  1. 合成事件(Synthetic伊夫nt)
    在 React 中,
    事件的处理由其中间协调达成的事件系统成功,触发的风浪都称之为
    合成事件(Synthetic伊夫nt),事件系统对浏览器做了合作,其提供的 API
    与原生的事件相同。

boolean bubbles
boolean cancelable
DOMEventTarget currentTarget
boolean defaultPrevented
number eventPhase
boolean isTrusted
DOMEvent nativeEvent
void preventDefault()
boolean isDefaultPrevented()
void stopPropagation()
boolean isPropagationStopped()
DOMEventTarget target
number timeStamp
string type

和原生事件的分别在于,事件不只怕异步话,如:

function onClick(event) {
     console.log(event); // => nullified object.
     console.log(event.type); // => "click"
     var eventType = event.type; // => "click"

     setTimeout(function() {
         console.log(event.type); // => null
        console.log(eventType); // => "click"
  }, 0);

     this.setState({clickEvent: event}); // Won't work. this.state.clickEvent will only contain null values.
     this.setState({eventType: event.type}); // You can still export event properties.
}

由来是在事变系统的内部贯彻当中,
五个事件目的可能会被录用(约等于事件做了池化
Pooling)。当2个风云响应函数执行过后,事件的属性被安装为 null,
要是想用保持事件的值的话,可以调用
event.persist()
这么,属性会被保留,并且事件也会被从池中取出。

  1. 事件捕获和冒泡
    在 DOM2.0 事件分为捕获阶段和冒泡阶段,React
    中一般我们报了名的风云为冒泡事件,假若要登记捕获阶段的轩然大波,能够在事变名称后加
    Capture 如:

onClick
onClickCapture
  1. 帮衬事件列表

粘贴板事件 {
      事件名称:onCopy onCut onPaste
      属性:DOMDataTransfer clipboardData
}
编辑事件 {
      事件名称:onCompositionEnd onCompositionStart onCompositionUpdate
      属性:string data
}
键盘事件 {
      事件名称:onKeyDown onKeyPress onKeyUp
      属性: {
        boolean altKey
        number charCode
        boolean ctrlKey
        boolean getModifierState(key)
        string key
        number keyCode
        string locale
        number location
        boolean metaKey
        boolean repeat
        boolean shiftKey
        number which
    }
}
// 焦点事件除了表单元素以外,可以应用到所有元素中
焦点事件 {
      事件名称:onFocus onBlur
      属性:DOMEventTarget relatedTarget
}
表单事件 {
      事件名称:onChange onInput onSubmit
}
鼠标事件 {
      事件名称:{
        onClick onContextMenu onDoubleClick onDrag onDragEnd onDragEnter onDragExit onDragLeave onDragOver onDragStart onDrop onMouseDown onMouseEnter onMouseLeave onMouseMove onMouseOut onMouseOver onMouseUp
    }
      属性:{
        boolean altKey
        number button
        number buttons
        number clientX
        number clientY
        boolean ctrlKey
        boolean getModifierState(key)
        boolean metaKey
        number pageX
        number pageY
        DOMEventTarget relatedTarget
        number screenX
        number screenY
        boolean shiftKey
    }
}
选择事件 {
      事件名称:onSelect
}
触摸事件 {
      事件名称:onTouchCancel onTouchEnd onTouchMove onTouchStart
      属性:{
        boolean altKey
        DOMTouchList changedTouches
        boolean ctrlKey
        boolean getModifierState(key)
        boolean metaKey
        boolean shiftKey
        DOMTouchList targetTouches
        DOMTouchList touches
    }
}
UI 事件 {
      事件名称:onScroll
      属性:{
        number detail
        DOMAbstractView view
    }
}
滚轮事件 {
      事件名称:onWheel
      属性:{
        number deltaMode
        number deltaX
        number deltaY
        number deltaZ
    }
}
媒体事件 {
      事件名称:{
        onAbort onCanPlay onCanPlayThrough onDurationChange onEmptied onEncrypted onEnded onError onLoadedData onLoadedMetadata onLoadStart onPause onPlay onPlaying onProgress onRateChange onSeeked onSeeking onStalled onSuspend onTimeUpdate onVolumeChange onWaiting
    }
}
图像事件 {
      事件名称:onLoad onError
}
动画事件 {
      事件名称:onAnimationStart onAnimationEnd onAnimationIteration
      属性:{
        string animationName
        string pseudoElement
        float elapsedTime
    }
}
渐变事件 {
      事件名称:onTransitionEnd
      属性: {
        string propertyName
        string pseudoElement
        float elapsedTime
    }
}

1.5.3 表单事件

在 React
中相比奇特的风浪是表单事件,半数以上零部件都以由此质量和情景来支配的,可是表单组件如
input, select, option 这几个组件的场地用户能够修改,在 React
中会特殊处理那一个零件的风浪。

  1. onChange 事件
    和经常 HTML 中的 onChange 事件差距, 在原生组件中,唯有 input
    成分失去焦点才会触发 onChange 事件, 在 React
    中,只要成分的值被改动就会触发 onChange 事件。

var MyComponent = React.createClass({
    getInitialState: function() {
        return {
            value: ''
        }
    },
    render: function() {
        return  <div onChange={this.onChangeBubble}>
                    <input value={this.state.value} onChange={this.onChange}/>
                </div>
    },
    onChange: function(ev) {
        console.log('change: ' + ev.target.value);
        this.setState({
            value: ev.target.value
        });
    },
    // onChange 事件支持所有组件,可以被用于监听冒泡事件
    onChangeBubble: function(ev) {
        console.log('bubble onChange event', + ev.target.value);
    }
})
  1. 互动属性
    表单组件中能被用户修改的天性叫交互属性,包蕴:

value => <input> 和 <select> 组件
checked => <input type="checkbox|radio">
selected => <opiton>
  1. textarea
    在 HTML 中,textarea 的值是像如下概念的:

<textarea name="" id="" cols="30" rows="10">
        some value
</textarea>

而在 React 中, TextArea 的应用办法同 input 组件,使用 value 来设置值

var MyComponent = function() {
    render: function() {
        return <div>
                    <textarea value={...} onChange={...}/>
               </div>
    }
}
  1. select 组件
    在 React 中 select 组件协理 value 值,value 值还帮衬多选

  <select value="B">
      <option value="A">Apple</option>
      <option value="B">Banana</option>
      <option value="C">Cranberry</option>
  </select>
  <select multiple={true} value={['B', 'C']}>
      <option value="A">Apple</option>
      <option value="B">Banana</option>
      <option value="C">Cranberry</option>
  </select>
  1. 受控组件
    在 React 中表单组件可分为两类,受控与非受控组件,受控组件是带有了
    value 值的,如:

render: function() {
      return <input type="text" value="....."/>
}

干什么叫受控组件? 因为这些时候用户无法改改 input 的值, input
的值永远是 value 固定了的值。即使去掉 value 属性,那么就足以输入值了。
那怎么修改受控组件的值吗? 如上边的例子中, 添加 onChange
事件,事件内修改 value 属性,value 属性的值会被设置到零部件的 value 中。

  1. 非受控组件
    没有 value 值的 input

render: function() {
     return <input type="text"/>
}

可以由此 defaultValue 属性来安装暗中同意值

render: function() {
      return <input type="text" defaultValue="Default Value">
}

看似的对于 checkbox 有 defaultChecked 属性
需求注意的是,暗许值只适用于第一遍渲染,在重渲染阶段将不会适用。

  1. checkbox & radio
    checkbox 和 radio 相比特殊, 假如在 onChange 事件中调用了
    preventDefault ,那么浏览器不会更新 checked
    状态,尽管事实上组件的值已经 checked 或然 unchecked 了 。

var CheckBox = React.createClass({
      getInitialState: function(){
          return {
              checked: false
        }
    },
    render: function() {
        return  <div>
            <input type="checkbox" 
                checked={this.state.checked} 
                onChange={this.onChange}/>
        </div>
    },
    onChange: function(ev) {
        this.setState({
            checked: true
        });
        ev.preventDefault();
    }
})

本条例子里边,checked 即使更新为 true ,可是 input 的值 checked 为
false
那应什么处理 checkbox 呢?

  • 防止调用 ev.preventDefault
  • 在 setTimeout 中处理 checked 的修改
  • 使用 click 事件

1.5.4 Style属性

在 React 中,可以一向设置 style 属性来支配样式,但是与 HTML 不同的是,
传入的 style 值为3个object, 对象的富有 key 都以驼峰式命名,eg:

render: function() {
    var style = {
        backgroundColor: 'red',
        height: 100,
        width: 100
    }
    return <div style={style}></div>
}

其中还足以见到不一样的地点时,为了简写宽度中度值,能够直接设置数字,对应
100 -> 100px。借使有些质量不须求添加 px 后缀,React 也会自行删除。

经过属性值驼峰式的案由是 DOM 内部访问 style
也是驼峰式。如若急需加上浏览器前缀瑞 -webkit-、-ms- 大驼峰(除了 ms ),
如:

var divStyle = {
  WebkitTransition: 'all', // 'W' 是大写
  msTransition: 'all'      // 'ms' 为小写
};

在以前的前端开发方式是 样式结构和逻辑要分别, 近来后 React
中却有不少人尊重** inline **的体制。 在小编看来一碗水端平,React
的那种形式也能做到体制模块化,样式重用(借用 Js 的特色)。并且因为 React
的兑现方式,Inline 样式的品质甚至比 class 的主意高。

1.6 Flux

1.6.1 Flux 介绍

简单易行来讲,Flux 是 非死不可 引入到 React
中的一种前端架构,通过定义其宗旨单向数据流的点子,让 React
应用越来越健康。同时,这种应用架构也持有普适性,可以动用到任何随意前端项目中,甚至足以使用到客户端应用开发中,约等于说
Flux 更应当叫做一种架构情势(Pattern)。

1.6.2 MVC 架构之痛

MVC
的落到实处大概有很多样方法,比较灵活,但核心精神不会变动,只是三者间的多少传递方向大概会转移,即使是
MVP 格局也只是 MVC 的变种,所以为了统一我们且以下图的 MVC 格局来研究。

美高梅开户网址 27

MVC.png

  1. 概念
  • Model: 负责保存应用数据,和后端交互联合运用数据
  • View: 负责渲染页面 HTML DOM
  • Controller: 负责连接 View 和 Model , Model 的其他变动会利用到 View
    中,View 的操作会通过 Controller 应用到 Model 中
  • 事关:Model, View, Controller 都以多对多涉及。
  1. 流程
    以 TODOMVC 为例子用户增加3个 todo 的互相流程:
    View -> Action -> Controller -> Model -> View
  • View -> Action: 添加按钮事件可能 input 输入的交付事件
  • Action -> Controller: 控制器响应 View 事件
  • Controller -> Model: 控制器依赖 Model, 调用 Model 添加 todo
  • Model -> View: View 监听 Model 的转移添加 todo 事件,在 HTML
    中添加多个新的 Todo 视图
  1. 问题
    对此新增2个 todo
    ,须求编制二个视图渲染处理函数,函数内添加新品类到列表中。同理对于删除1个todo,也会有2个处理函数。当工作逻辑变多过后,只怕有过多模子须要做增删改的机能,与之对应的就是大家需求仔细营造那样多的渲染处理函数。
    那种局地更新格局是高质量的关键所在,但难题是:
  • 创新逻辑复杂,必要编制大量的一些渲染函数
  • 难点一定困难,页面的近年来意况是有数量和那么些有些更新函数确定的
  1. 何以化解
    若是渲染函数惟有贰个,统一放在 App
    控制器中,每一遍换代重渲染页面,那样的话:

    • 任何数据的更新都只用调用重渲染就行
    • 多少和当前页面的状态是绝无仅有确定的
      重渲染也有坏处,会牵动惨重的习性难点,重渲染和局地渲染各有优劣,对
      MVC 来说那是一个难堪的挑选,不可以做到鱼和熊掌兼得。

1.6.3 Flux 架构

通过 React + Flux 就可以健全消除 MVC 的难题。简单来讲在 Flux
架构中直接退出了决定器层,MVC 架构成为了 MV + Flux 架构。

  • 重渲染: 在 React
    中老是渲染都以重渲染,且不影响页面品质,是因为重渲染的是 Virtual
    Dom。那就象征完全不用去关系重渲染难题,增删改的渲染都和初叶化渲染相同入口
  • 数量和景色一致性: Store 的数目确定应用唯一的情况
  1. 概念
  • one way data flow

    那是 Flux 架构的大旨绪想,从图中得以看看,数据的流向从action 到 view
    的壹个单向流。

    美高梅开户网址 28

    单项数据流.png

  • Action

    Action
    可以领略为对使用数据修改的授命,任何改动应用数据的一言一行都必须需通过触发
    action 来修改。Action 可以来自于 View,也足以来自服务端的数量更新。

    美高梅开户网址 29

    action.png

  • Action creator
    为了架空 Action ,提供部分帮忙的语义化的主意来创造Action,这一个帮扶方法叫做 Action Creator。

![](https://upload-images.jianshu.io/upload_images/5691460-97ad656f6ce94c41.png)

Action creator.png
  • Stores
    利用的数目基本,所有应用数据都存放在此地控制,同时富含数据的控制行为,大概含有两个store.
  • Dispatcher
    action 的通晓者,所有 action 都会经过 dispatcher,由 dispatcher 控制
    action 是不是应当传入到 store 中,Dispatcher 是多少个单例。
  • View
    页面的视图,对应 React 的 Component, 视图可以触发 action 到
    dispatcher。
    内需区分出一种叫控制器 View(Controller View)的类型,那种 View
    能够知晓 store 数据,把 store
    数据转载为本人的情状,在将数据传递给任何 view 。 并且可以监听 store
    数据的改变,当 store 数据变动之后再一次设置意况触发重渲染。
    可以将控制器 View 对应 MVC 中的控制器,不过差距很大,控制器 View
    唯一多做的事体就是监听 store 数据变动,没有其他任何事情处理逻辑。
  1. 流程
    同等以 TODOMVC 的丰裕 todo 为例,Flux 中的流程为:

View -> Action(Action Creator -> Action) -> Dispatcher -> Store -> Controller View -> View
  • View -> Action: 添加按钮事件还是 input 输入的交给事件,View
    上校事件转化为 action, action 由 Action Creator 创造。
  • Action -> Dispatcher: action 统一由 Dispatcher 分配
  • Dispatcher -> Store: Dispatcher 分配 action 到 Store
  • Store -> Controller View: 控制器 View 监听 store
    的数量变动,将数据转载为自家性质
  • Controller View -> View: 数据变动机关重渲染所有视图
    与MVC的对比
  • 渲染策略: 数据变动 Flux 自动渲染,MVC 手动编写更新函数
  • 事件触发策略: Flux 中具有 action 交给 dispather 分配,MVC
    中提交对应的控制器分配
  • Flux 在主导策略上的两样是不留余地 MVC 架构难题的重点

1.6.4 理解 Flux 架构

Flux 架构是分外优雅凝练的,合理使用了部分大好的架构思想

  1. 分而治之(Divide And Conquer)
    数据的处理进度是 Store -> Controller View -> View。
    所有数据来源于 Store,页面的渲染层级为 Store 将数据传入 Controller
    View, 再由 Controller View 传入子 View , 平昔到 View 的纸牌节点。
    以此是多个典型的分而治之策略,将大的页面拆分为小的模块,再由小的模块拆分为小的零件,具体组件负者组件自个儿的难点,所有子组件都是自私的,不用关切“大家”,只用关爱“小家”。
  2. 合而治之 – 中央化控制
    Flux 把所有的 View 都作为愚民,Store
    视作能源的拥有者为统治者,统治者要求提供能源(数据)给公民,不过一旦老百姓企图对能源修改(Mutation),必须得先通告给统治者,让统治者决定是不是做处理。
    作者们为 Flux 中的概念分配角色
    View: 平民
    Action: 能源修改操作
    Dispatcher: 审核官
    Store: 统治者
    1个策划修改财富的操作可以描述为:

View Require Mutation -> Action -> Dispatcher -> Store -> Mutate Handler

国民提交 Mutation
请求,由审核官控制,审核通过后递交给统治者,统治者再分配给亲信做资源Mutation
合而治之的方针也相当大旨化控制策略,
作为统治者既要了然放职务(能源的分红),也要领悟控制权利(能源的改动),这种裁减自如的合理性是
Flux 简洁的有史以来。
再就是那种思维带来的独到之处如下:

  • View 的独立性和简单性:View
    自己的逻辑简单,不须求明白太多事情,只关怀上级传来的多少,那种方式使得
    View 是低耦合的,简洁的。
  • 高可维护性:主题化控制知道所有对财富的操作,假设暴发 bug,
    可以便捷定位难题
  1. 函数式编程思想
    在 Flux 中数量的单向流动正视于 View
    的显明,相同的数码传入相同的零件,得到的结果自然要平等,那是函数式编程的考虑。
    为了保证组件也能不负众望 “纯函数”
    的天性,相同的属性会取得一致的渲染结果。 在写 React
    组件的时候尽量准守一下预订:

1.尽量使用无状态组件
2.除了控制类组件以外其他组件避免使用组件状态
3.可以通过属性计算出来的状态不要用状态来表示
4.组件的渲染避免外部依赖,按照纯函数的方式写

函数式的亮点也是无副功效组件的亮点:

  • 无耦合,可移植性强: 组件可重用性高
  • 可测试性高:组件无依靠,可以很不难的独门测试组件

发表评论

电子邮件地址不会被公开。 必填项已用*标注

网站地图xml地图