理解angular中的module和injector,即依赖注入

本文详细介绍了AngularJS中的依赖注入(DI)概念及其核心组件,包括angular.module()、$provide、$injector的作用及使用方法,并阐述了AngularJS中三种声明依赖的方式。

依赖注入(DI)的好处不再赘言,使用过spring框架的都知道。AngularJS作为前台js框架,也提供了对DI的支持,这是JavaScript/jQuery不具备的特性。angularjs中与DI相关有angular.module()、angular.injector()、 $injector$provide。对于一个DI容器来说,必须具备3个要素:服务的注册、依赖关系的声明、对象的获取。比如spring中,服务的注册是通过xml配置文件的<bean>标签或是注解@Repository、@Service、@Controller、@Component实现的;对象的获取可以ApplicationContext.getBean()实现;依赖关系的声明,即可以在xml文件中配置,也可以使用@Resource等注解在Java代码中声明。在angular中,module和$provide相当于是服务的注册;injector用来获取对象(angular会自动完成依赖的注入);依赖关系的声明在angular中有3种方式。下面从这3个方面,介绍下angular的DI。


1、angular.module()创建、获取、注册angular中的模块

The angular.module() is a global place for creating, registering and retrieving Angular modules.When passed two or more arguments, a new module is created. If passed only one argument, an existing module (the name passed as the first argument to module) is retrieved。

[javascript]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. // 传递参数不止一个,代表新建模块;空数组代表该模块不依赖其他模块  
  2. var createModule = angular.module("myModule", []);  
  3.   
  4. // 只有一个参数(模块名),代表获取模块  
  5. // 如果模块不存在,angular框架会抛异常  
  6. var getModule = angular.module("myModule");  
  7.   
  8. // true,都是同一个模块  
  9. alert(createModule == getModule);  
该函数既可以创建新的模块,也可以获取已有模块,是创建还是获取,通过参数的个数来区分。

angular.module(name, [requires], [configFn]);

name:字符串类型,代表模块的名称;

requires:字符串的数组,代表该模块依赖的其他模块列表,如果不依赖其他模块,用空数组即可;

configFn:用来对该模块进行一些配置。

现在我们知道如何创建、获取模块了,那么模块究竟是什么呢?官方的Developer Guide上只有一句话:You can think of a module as a Container for the different parts of your app – controllers, services, filters, directives, etc.现在我还不太理解,大致就是说模块是一些功能的集合,如控制器、服务、过滤器、指令等子元素组成的整体。现在解释不了,先遗留。


2、$provide和模块的关系

The $provide service has a number of methods for registering components with the $injector. Many of these functions are also exposed on angular.Module.

之前提到过:module和provide是用来注册服务到injector中的。查看官方的API,可以看到$provide提供了provide()、constant()、value()、factory()、service()来创建各种不同性质的服务;angular.Module中也提供了这5个服务注册方法。其实2者功能是完全一样的,就是用来向DI容器注册服务到injector中。

官方API下的auto有$provide 和 $injector,Implicit module which gets automatically added to each $injector.按照字面意思是说,每一个injector都有这2个隐含的服务。但1.2.25版本中,感觉没有办法获取injector中的$provide。不知道这是为什么?一般来说也不需要显示使用这个服务,直接使用module中提供的API即可。

[javascript]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. var injector = angular.injector();  
  2. alert(injector.has("$provide"));//false  
  3. alert(injector.has("$injector"));//true  

3、angular.injector()

使用angular.injector();也能获取到注入器,但是没有和模块绑定。这种做法是没有意义的,相当于是你创建了一个空的DI容器,里面都没有服务别人怎么用呢。正确的做法是,在创建注入器的时候,指定需要加载的模块。

[javascript]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. // 创建myModule模块、注册服务  
  2. var myModule = angular.module('myModule', []);  
  3. myModule.service('myService'function() {  
  4.             this.my = 0;  
  5. });  
  6.   
  7. // 创建herModule模块、注册服务  
  8. var herModule = angular.module('herModule', []);  
  9. herModule.service('herService'function() {  
  10.             this.her = 1;  
  11. });  
  12.   
  13. // 加载了2个模块中的服务  
  14. var injector = angular.injector(["myModule","herModule"]);  
  15. alert(injector.get("myService").my);  
  16. alert(injector.get("herService").her);  
如果加载了多个模块,那么通过返回的injector可以获取到多个模块下的服务。这个例子中如果只加载了myMoudle,那么得到的injector就不能访问herMoudle下的服务。这里特别需要注意下:angular.injector()可以调用多次,每次都返回新建的injector对象

[javascript]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. var injector1 = angular.injector(["myModule","herModule"]);  
  2. var injector2 = angular.injector(["myModule","herModule"]);  
  3.   
  4. alert(injector1 == injector2);//false  

4、angular中三种声明依赖的方式

angular提供了3种获取依赖的方式:inference、annotation、inline方式。

[javascript]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. // 创建myModule模块、注册服务  
  2. var myModule = angular.module('myModule', []);  
  3. myModule.service('myService'function() {  
  4.             this.my = 0;  
  5. });  
  6.   
  7. // 获取injector  
  8. var injector = angular.injector(["myModule"]);  
  9.   
  10. // 第一种inference  
  11. injector.invoke(function(myService){alert(myService.my);});  
  12.   
  13. // 第二种annotation  
  14. function explicit(serviceA) {alert(serviceA.my);};  
  15. explicit.$inject = ['myService'];  
  16. injector.invoke(explicit);  
  17.   
  18. // 第三种inline  
  19. injector.invoke(['myService'function(serviceA){alert(serviceA.my);}]);  

其中annotation和inline方式,对于函数参数名称没有要求,是推荐的做法;inference方式强制要求参数名称和服务名称一致,如果JS代码经过压缩或者混淆,那么功能会出问题,不建议使用这种方式。


var app = angular.module("myApp",[]);

app.factory("game",function(){
	return {
		title:"StarCraft"
	}
});
//创建一个injector 参数为绑定的模板  会返回一个injector对象
//通过返回的对象你可以调用模板里面的各种服务
//这种做法会导致所有的当前的模板下面的控制器都会有这个服务  就是弹出game.title  
//如非特殊场景,不建议使用
angular.injector(["myApp"]).invoke(function(game){
	alert(game.title);
})

app.controller("AppController",function($scope,game){
	$scope.title = game.title;
});

app.controller("OtherController",function($scope,$injector) {
	//$injector 服务在angularJS中已经有了  就不需要使用  angular.injector(["myApp"])显示绑定了
	$injector.invoke(function(game){
		$scope.title = game.title;
		alert(game.title);
	});
});

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值