作者: Tang Wei
当你使用AngularJs的时候你遇到过性能问题吗?使用ReactJs可以使页面渲染得更快。下面通过几个案例来对比用React来渲染AngularJs和直接使用Angular来渲染之间的性能差异。
我喜欢AngularJs,我不仅在一些有趣的小项目中使用它,同时也在一些大型的专业项目中来使用AngularJs。我同时也尝试其他框架,像BackboneJs和EmberJs,这俩也是很好的工具,以上三种都属于一类叫做MVC的框架(可以叫MVVC),但是每次使用它们的时候,我都会在渲染方面遇到的一些性能的不足。对于渲染列表项目,双向数据绑定和单向数据绑定并没有实际的区别。对我而言在不考虑双向数据绑定情况下BackboneJS渲染性能优于AngularJs。
最近出现一个叫做ReactJs的优秀框架,一个看起来有点弱小的库接下来一段时间将对JavaScript MVC框架造成很大的影响。简而言之,这个库能够使网页渲染性能得到提升,ReactJs主要使用的部分是针对MVC框架中的View部分,刚开始我不明白设计者为什么会把Model和Controller这两部分给抛弃呢,难道像EmberJs, BackboneJS 和AngularJS这些框架中最有趣的地方不就是Model和Controller部分吗?仔细思考后,其实ReactJs在Controller部分也有很特色的地方,然而ReactJs不是一个大而全的MVC框架,ReactJs的工作原理是保持一个虚拟的DOM,渲染只有在当UI更新时才会改变,这是否有意义呢,听起来新颖但又有理,通过发送指令来更新UI,React只改变和更新现存的DOM(使用虚拟DOM超级快),通过虚拟DOM技术可以使变化的差异很快的被展现,Facebook和Instagram已经开始在实际项目中使用ReactJs了。
我听过ReactJs的开发者Pete Hunt的谈论ReactJs意图的一个podcast,很建议你也去听一下这个podcast,他开源reactjs目的是在现有框架的View部分制造一些噪音,我发现这些现存的框架在UI部分将打算采取这种相同的策略,一些可以使用的新库将会出现。
ReactJs当中的那个V很容易和现存的框架中那个V做比较,backbone,还有angular的ngRepeat,可以和Coffeescript一起使用,所以使用React来渲染可以在你应用性能上面派上用场,例如通过ng-repeat来重复几百项,在我以前的文章中,我写了一篇如何使长列表在AngularJs有效,但是所有的使渲染更快的技术是通过渲染列表中的一部分实现的,使用ReactJS来渲染AngularJs,可以使你的渲染时间较少80%,我玩ngReact的时候那种性能的提升让我质疑代码是不是哪儿出错了,渲染时间从原来直接使用AngularJs的4200ms降低到通过使用ReactJs渲染的120ms,你也可以去尝试下,或者直接看这篇文章下面的例子。
我仍然经历这一些问题当使用AngularJs进行大量的DOM绑定,但是另外一个问题是特殊的双向数据绑定。如果你对这个问题有些担心,或许你应该考虑使用比AngularJs更强大的框架,然而ReactJs的重大作用将就是帮助我们尽可能快的渲染元素到用户的屏幕上。
我将会用一个小案例来向你们解释如何来使用ReactJs来渲染一个AngularJs的应用,步骤如下:
- 默认你已经安装了Bower,然后新建目录,安装reactjs和AngularJs到你的目录下:
$ mkdir fast-angular
$ cd fast-angular
$ bower install --save react
$ bower install --save angular
- 当这些命令运行好后,我们需要的AngularJs和ReactJs已经在我们本地安装好了,接着我们建立一个简单的HTML文件,将这两个脚本引入。
<!doctype html>
<html>
<head>
<meta charset="utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<title></title>
<meta name="description" content="">
<meta name="viewport" content="width=device-width">
</head>
<body>
<h1>Fast AngularJs</h1> <h3>AngularJs with ReactJs</h3>
<script src="bower_components/angular/angular.js"></script>
<script src="bower_components/react/react.js"></script>
</body>
</html>
- 我们需要创建一个来渲染我们输入的字符串的ReactJs组件,所以我们使用ReactJs来渲染我们的模型,创建一个名叫做MYAPP的组件,传递给它来呈现一个props,接着我们创建一个传统的angular指令和控制器(增加标记在我们的html来启动这个应用),通过使用指令来告诉它渲染,而不是去调用ReactJs的组件,我们使用$watch来重新渲染和更新我们的框架,当组件已安装更新现有的实例中我们叫做createComponent 。 实际代码演示http://plnkr.co/edit/FXK3lU?p=preview
<!doctype html>
<html>
<head>
<meta charset="utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<title></title>
<meta name="description" content="">
<meta name="viewport" content="width=device-width">
</head>
<body ng-app="fasterAngular">
<h1>Fast AngularJs</h1> <h3>AngularJs with ReactJs</h3>
<div ng-controller="mycontroller">
<input ng-model="framework"/>
<hr>
<fast-ng framework="framework"></fast-ng>
<hr>
Rendering with traditional AngularJs
</div>
<script src="bower_components/angular/angular.js"></script>
<script src="bower_components/react/react.js"></script>
<script >
var MYAPP = React.createClass({
displayName:'MYAPP',
render:function(){
return React.DOM.div(null,
"Rendering faster in AngularJs with ",
this.props.framework);
}
});
</script>
<script>
angular.module('fasterAngular', []).
controller('mycontroller', ['$scope', function($scope){
$scope.framework = 'ReactJs';
}]).directive('fastNg', function(){
return{
restrict:'E',
scope:{
framework:'='
},
link:function(scope, el, attrs){
scope.$watch('framework', function(newValue, oldValue){
React.renderComponent(
MYAPP({framework:newValue}),
el[0]
);
})
}
}
})
</script>
</body>
</html>
- 当然上面这个例子没有使我们看到性能上面的提升,但是它阐述了如何通过ReactJs来渲染我们的模块。下面这个渲染一长串数字例子就是展现出性能的例子,是一个来自ng-react的例子,我们生成了一个含有1500个数据的数组并将其渲染到表格中,假如使用AngularJs中原生的ng-repeat通常会带来一些性能上面的问题,看这个使用ReactJs来渲染的Plunkr,另外一个是使用原生ng-repeat来渲染的Plunkr。使用ReactJs渲染的代码如下:
<!doctype html>
<html>
<head>
<meta charset="utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<title></title>
<meta name="description" content="">
<meta name="viewport" content="width=device-width">
</head>
<body ng-app="fasterAngular">
<h1>Fast AngularJs</h1> <h3>AngularJs with ReactJs</h3>
<div ng-controller="mycontroller">
<fast-repeat data="data"></fast-repeat>
</div>
<script src="bower_components/angular/angular.js"></script>
<script src="bower_components/react/react.js"></script>
<script >
var MYLIST = React.createClass({displayName: 'MYLIST',
render: function() {
var data = this.props.data;
var rows = data.map(function(datum) {
var clickHandler = function(ev){
console.log("Still in reactJs");
console.log(ev);
}
return (
React.DOM.tr( {onClick:clickHandler},
React.DOM.td(null, datum['0']),
React.DOM.td(null, datum['1']),
React.DOM.td(null, datum['2']),
React.DOM.td(null, datum['3']),
React.DOM.td(null, datum['4'])
)
);
});
return (
React.DOM.table(null,
rows
)
);
}
});
</script>
<script>
angular.module('fasterAngular', []).
controller('mycontroller', ['$scope', function($scope){
$scope.framework = 'ReactJs';
$scope.data = [];
// Fill the data map with random data
for(var i = 0; i < 1500; ++i) {
$scope.data[i] = {};
for(var j = 0; j < 5; ++j) {
$scope.data[i][j] = Math.random();
}
}
}]).directive('fastRepeat', function(){
return{
restrict: 'E',
scope:{
data: '='
},
link:function(scope, el, attrs){
scope.$watch('data', function(newValue, oldValue){
React.renderComponent(
MYLIST({data:newValue}),
el[0]
);
})
}
}
})
</script>
</body>
</html>
<!doctype html>
<html>
<head>
<meta charset="utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<title></title>
<meta name="description" content="">
<meta name="viewport" content="width=device-width">
</head>
<body ng-app="fasterAngular">
<h1>Fast AngularJs</h1> <h3>AngularJs with ReactJs</h3>
<div ng-controller="mycontroller">
<button ng-click="refresh()">Refresh Data</button>
<fast-repeat data="data"></fast-repeat>
<!-- <table>
<tr ng-repeat="line in data" ng-click="clickHandler(ev)">
<td></td>
<td></td>
<td></td>
<td></td>
<td></td>
</tr>
</table> -->
</div>
<script src="bower_components/angular/angular.js"></script>
<script src="bower_components/react/react.js"></script>
<script >
var MYLIST = React.createClass({displayName: 'MYLIST',
render: function() {
var data = this.props.data;
var rows = data.map(function(datum) {
var clickHandler = function(ev){
console.log("Still in reactJs");
console.log(ev);
}
return (
React.DOM.tr( {onClick:clickHandler},
React.DOM.td(null, datum['0']),
React.DOM.td(null, datum['1']),
React.DOM.td(null, datum['2']),
React.DOM.td(null, datum['3']),
React.DOM.td(null, datum['4'])
)
);
});
return (
React.DOM.table(null,
rows
)
);
}
});
</script>
<script>
angular.module('fasterAngular', []).
controller('mycontroller', ['$scope', function($scope){
$scope.framework = 'ReactJs';
$scope.data = [];
// Fill the data map with random data
$scope.clickHandler = function(){
console.log("in AngularJS");
}
$scope.refresh = function(){
for(var i = 0; i < 1500; ++i) {
$scope.data[i] = {};
for(var j = 0; j < 5; ++j) {
$scope.data[i][j] = Math.random();
}
}
}
$scope.refresh()
}]).directive('fastRepeat', function(){
return{
restrict: 'E',
scope:{
data: '='
},
link:function(scope, el, attrs){
scope.$watchCollection('data', function(newValue, oldValue){
React.renderComponent(
MYLIST({data:newValue}),
el[0]
);
})
}
}
})
</script>
</body>
</html>
- 使用AngularJs渲染1500行大概用了1.35秒,然而使用ReactJs渲染仅仅用了320ms,详细见下面两张从chrome开发者工具的截图:
使用ReactJs渲染320ms
使用原生AngularJs渲染1200ms
总结
这仅仅是关于使用使用react提高angular的渲染性能一个简短的介绍,在我接下来的一些项目中,毫无疑问我会使用ReactJs来渲染我认为性能不足的部分。正如 Pete Hunt所说,ReactJs是完全改变游戏规则的,背后的意图是希望改变人们对于内部运作的一般想法,我确信不久所有主流的框架将会有一个类似的想法来使用ReactJs思维,或者设计成很容易和ReactJs集成,我喜欢ReactJs,使用虚拟DOM这种想法让我觉得它意义非凡。