AngularJS中的模块详解


Posted in Javascript onJanuary 29, 2015

在讲angularjs的模块之前,我们先介绍一下angular的一些知识点:

AngularJS是纯客户端技术,完全用Javascript编写的。它使用的是网页开发的常规技术(HTML,CSS,Javascript),目的是让网页应用开发更快更容易。

AngularJS简化应用开发的一个重要方法是,将一个些通用的低级开发操作包装起来提供给开发者。AngularJS会自动处理好这些低级操作。它们包括:

1.DOM操作
2.设置事件的监听
3.输入验证,因为AngularJS会处理大部分这些操作,所以开发者就能更多的专注在应用的业务逻辑上,更少地编写那些重复性的、易错的、低级的代码。

在AngularJS简化开发的同时,它也为客户端带来了一些精巧的技术,它们包括:

1.数据、业务逻辑、视图的分离
2.数据和视图的自动绑定
3.通用服务
4.依赖注入(主要用于聚合服务)
5.可扩展的HTML编译器(完全由JavaScript编写)
6.易于测试
7.客户端对这些技术的需求其实已经存在很久了。

同时,你还可以用AngularJS来开发单页或者多页的应用,不过其主要还是用来开发单页的。 AngularJS支持浏览器的历史操作,向前,向后按钮,单页应用中的收藏操作。

接下来,我们来详细讲解angularJS的模块。

大部分应用都有一个主方法用来实例化、组织、启动应用。AngularJS应用没有主方法,而是使用模块来声明应用应该如何启动。这种方式有以下几个优点:

1.启动过程是声明式的,所以更容易懂。
2.在单元测试是不需要加载全部模块的,因此这种方式有助于写单元测试。
3.可以在特定情况的测试中增加额外的模块,这些模块能更改配置,能帮助进行端对端的测试。
4.第三方代码可以打包成可重用的模块。
5.模块可以以任何先后或者并行的顺序加载(因为模块的执行本身是延迟的)。

举个例子:

<!doctype html>
<html ng-app="myApp">
 <head>
  <script src="http://code.angularjs.org/angular-1.0.2.min.js"></script>
  <script>
      var myAppModule = angular.module('myApp', []);


// configure the module.



// in this example we will create a greeting filter



myAppModule.filter('greet', function() {
 



return function(name) {
  




return 'Hello, ' + name + '!';
 



};



});
  </script>
 </head>
 <body>
  <div>
   {{ 'World' | greet }}
  </div>
 </body>
</html>

上面的例子,我们是通过在<html ng-app="myApp">中进行指定,来实现使用myApp这个模块启动应用的。

以上这个例子写法很简单,但是不适合用同样的写法来写大型应用。我们推荐是将你的应用拆分成以下几个模块:

1.一个服务模块,用来做服务的声明。
2.一个指令模块,用来做指令的声明。
3.一个过滤器模块,用来做过滤器声明。
4.一个依赖以上模块的应用级模块,它包含初始化代码。

举个例子:

<!doctype html>
<html ng-app="xmpl">
 <head>
  <script src="http://code.angularjs.org/angular-1.0.2.min.js"></script>
  <script src="script.js"></script>
 </head>
 <body>
  <div ng-controller="XmplController">
   {{ greeting }}!
  </div>
 </body>
</html>
[/code]

script.js:

[code]
angular.module('xmpl.service', []).   //服务模块
 value('greeter', {    //自定义greeter对象
  salutation: 'Hello',
  localize: function(localization) {
    this.salutation = localization.salutation;
  },
  greet: function(name) {
    return this.salutation + ' ' + name + '!';
  }
 }).
 value('user', {   //自定义user对象
  load: function(name) {
    this.name = name;
  }
 });
angular.module('xmpl.directive', []);  //指令模块
angular.module('xmpl.filter', []);   //过滤器模块
angular.module('xmpl', ['xmpl.service', 'xmpl.directive', 'xmpl.filter']).  //模块xmpl依赖于数组中的模块
 run(function(greeter, user) {
  // 初始化代码,应用启动时,会自动执行
  greeter.localize({
    salutation: 'Bonjour'
  });
  user.load('World');
 })
// A Controller for your app
var XmplController = function($scope, greeter, user) {
   $scope.greeting = greeter.greet(user.name);
}

这样拆分的原因是,在你的测试中常常需要忽略掉初始化代码,因为这些代码比较难测试。通过把它拆分出来就能在测试中方便地忽视掉它。通过只加载和当前测试相关的模块,也能使测试更专一。以上只是一个建议,你可以放心根据你的具体情况来调整。

一个模块就是一系列配置和代码块的集合,它们是在启动阶段就附加到应用上的。一个最简单的模块由两类代码块集合组成的:

配置代码块 - 在注入提供者注入和配置阶段执行。只有注入提供者和常量可以被注入到配置块中。这是为了防止服务在被配置好之前就被提前初始化。
运行代码块 - 在注入器被创建后执行,被用来启动应用的。只有实例和常量能被注入到运行块中。这是为了防止在运行后还出现对系统的配置。

代码实现:

angular.module('myModule', []).  
config(function(injectables) { // provider-injector      配置代码块


// This is an example of config block.


// You can have as many of these as you want.


// You can only inject Providers (not instances)


// into the config blocks.

}). run(function(injectables) { // instance-injector      运行代码块


// This is an example of a run block.


// You can have as many of these as you want.


// You can only inject instances (not Providers)


// into the run blocks

});

模块还有一些配置的简便方法,使用它们的效果等同于使用代码块。举个例子:

angular.module('myModule', []).
 value('a', 123).
 factory('a', function() { return 123; }).
 directive('directiveName', ...).
 filter('filterName', ...);
// is same as
angular.module('myModule', []).
 config(function($provide, $compileProvider, $filterProvider) {
  $provide.value('a', 123)
  $provide.factory('a', function() { return 123; })
  $compileProvider.directive('directiveName', ...).
  $filterProvider.register('filterName', ...);
 });

配置块会按照$provide, $compileProvider, $filterProvider,注册的顺序,依次被应用。唯一的例外是对常量的定义,它们应该始终放在配置块的开始处。

运行块是AngularJS中最像主方法的东西。一个运行块就是一段用来启动应用的代码。它在所有服务都被配置和所有的注入器都被创建后执行。运行块通常包含了一些难以测试的代码,所以它们应该写在单独的模块里,这样在单元测试时就可以忽略它们了。

模块可以把其他模块列为它的依赖。“依赖某个模块”意味着需要把这个被依赖的模块在本块模块之前被加载。换句话说被依赖模块的配置块会在本模块配置块前被执行。运行块也是一样。任何一个模块都只能被加载一次,即使它被多个模块依赖。

模块是一种用来管理$injector配置的方法,和脚本的加载没有关系。现在网上已有很多控制模块加载的库,它们可以和AngularJS配合使用。因为在加载期间模块不做任何事情,所以它们可以以任意顺序或者并行方式加载

Javascript 相关文章推荐
document.all与WEB标准
May 13 Javascript
[原创]提供复制本站内容时出现,该文章转自脚本之家等字样的js代码
Mar 27 Javascript
javascript 延迟加载技术(lazyload)简单实现
Jan 17 Javascript
判断一个变量是数组Array类型的方法
Sep 16 Javascript
AngularJS语法详解(续)
Jan 23 Javascript
Javascript的比较汇总
Jul 25 Javascript
jQuery+CSS3实现四种应用广泛的导航条制作实例详解
Sep 17 Javascript
AJAX和jQuery动态加载数据的实现方法
Dec 05 Javascript
使用vue如何构建一个自动建站项目
Feb 05 Javascript
webpack external模块的具体使用
Mar 10 Javascript
jquery实现动态创建form并提交的方法示例
May 27 jQuery
JavaScript实现联动菜单特效
Jan 07 Javascript
jquery结合CSS使用validate实现漂亮的验证
Jan 29 #Javascript
5个数组Array方法: indexOf、filter、forEach、map、reduce使用实例
Jan 29 #Javascript
推荐一个自己用的封装好的javascript插件
Jan 29 #Javascript
js实现点击左右按钮轮播图片效果实例
Jan 29 #Javascript
JavaScript中实现继承的三种方式和实例
Jan 29 #Javascript
javascript面向对象程序设计(一)
Jan 29 #Javascript
jquery调取json数据实现省市级联的方法
Jan 29 #Javascript
You might like
PHP之COOKIE支持详解
2010/09/20 PHP
php更改目录及子目录下所有的文件后缀扩展名的代码
2010/10/12 PHP
PHP的PDO大对象(LOBs)
2019/01/27 PHP
在你的网页中嵌入外部网页的方法
2007/04/02 Javascript
JQuery 获得绝对,相对位置的坐标方法
2010/02/09 Javascript
Javascript的闭包详解
2014/12/26 Javascript
纯javascript实现的小游戏《Flappy Pig》实例
2015/07/27 Javascript
JS+CSS实现分类动态选择及移动功能效果代码
2015/10/19 Javascript
JS实现IE状态栏文字缩放效果代码
2015/10/24 Javascript
基于jQuery实现左右图片轮播(原理通用)
2015/12/24 Javascript
JS中script标签defer和async属性的区别详解
2016/08/12 Javascript
Javascript中判断一个值是否为undefined的方法详解
2016/09/28 Javascript
基于jQuery实现Tabs选项卡自定义插件
2016/11/21 Javascript
JS实现的base64加密解密操作示例
2018/04/18 Javascript
微信公众号获取用户地理位置并列出附近的门店的示例代码
2019/07/25 Javascript
Layui动态生成select下拉选择框不显示的解决方法
2019/09/24 Javascript
[03:11]DOTA2上海特锦赛小组赛第一日recap精彩回顾
2016/02/28 DOTA
Python的print用法示例
2014/02/11 Python
Python对小数进行除法运算的正确方法示例
2014/08/25 Python
python 提取key 为中文的json 串方法
2018/12/31 Python
Python学习笔记之Break和Continue用法分析
2019/08/14 Python
TensorFlow设置日志级别的几种方式小结
2020/02/04 Python
tensorflow 实现自定义梯度反向传播代码
2020/02/10 Python
python通过文本在一个图中画多条线的实例
2020/02/21 Python
Bergfreunde丹麦:登山装备网上零售商
2017/02/26 全球购物
夏威夷灵感服装及配饰:Reyn Spooner
2018/09/18 全球购物
日本酒店、民宿、温泉旅馆、当地旅行团中文预订:e路东瀛
2019/12/09 全球购物
C++:局部变量能否和全局变量重名
2014/03/03 面试题
甜品店的创业计划书范文
2014/01/02 职场文书
授权委托书格式模板
2014/04/03 职场文书
党员自我评议个人对照检查材料
2014/09/16 职场文书
幼儿园五一劳动节活动总结
2015/02/09 职场文书
2016年秋季新学期致辞
2015/07/30 职场文书
区域销售大会开幕词
2016/03/04 职场文书
PHP获取学生成绩的方法
2021/11/17 PHP
使用Apache Camel表达REST服务的方法
2022/06/10 Servers