AngularJS深入(5)——provider
太精彩,不得不全文引用。
到这个层次,可能才敢说自己懂了吧。。。
http://syaning.com/2015/07/21/dive-into-angular-5/
在使用AngularJS的时候,可能需要创建各种各样的服务,这个时候,需要告诉AngularJS如何创建这些服务,这便是Provider。在实际使用的时候,会有provider
,factory
,service
,value
,constant
,事实上,它们都是Provider,从factory
到constant
,只不过是对provider
的一步步封装。相关源码都在函数createInjector
中。
1. provider
源码如下:
function provider(name, provider_) {
assertNotHasOwnProperty(name, 'service');
if (isFunction(provider_) || isArray(provider_)) {
provider_ = providerInjector.instantiate(provider_);
}
if (!provider_.$get) {
throw $injectorMinErr('pget', "Provider '{0}' must define $get factory method.", name);
}
return providerCache[name + providerSuffix] = provider_;
}
首先,一个Provider的名字不可以是hasOwnProperty
,然后,如果provider_
是函数或者数组的话,调用providerInjector.instantiate
将其实例化。之后进行判断,如果provider_
没有$get
属性,则报错。然后将provider_
缓存在providerCache
中。
下面来分析对参数provider_
的约束。
(1) 如果provider_
是函数,那么会调用provider_ = providerInjector.instantiate(provider_)
对其再次赋值,相当于以provider_
作为构造函数来创建实例。因此在构造函数中,必须要有对$get
的操作,因此如下形式都是可以的:
app.provider('greeting', function() {
this.$get = function() {
return {
sayHello: function() {
console.log('hello world');
}
};
};
});
// or
app.provider('greeting', function() {
return {
$get: function() {
return {
sayHello: function() {
console.log('hello world');
}
};
}
};
});
(2) 如果provider_
是数组,同样的接下来会调用provider_ = providerInjector.instantiate(provider_)
,所以数组必须是[deps, fn]
的形式,其中deps
为依赖,fn
与(1)中函数的限制相同。例如:
app.provider('greeting', ['$injector', function($injector) {
this.$get = function() {
return {
sayHello: function() {
console.log('hello world', $injector);
}
};
};
}]);
(3)如果provider_
是一个对象字面量,则它必须要有$get
属性,例如:
app.provider('greeting', {
$get: function() {
return {
sayHello: function() {
console.log('hello world');
}
};
}
});
接下来讨论对$get
的约束。在定义instanceInjector
的时候,代码如下:
instanceInjector = (instanceCache.$injector =
createInternalInjector(instanceCache, function(serviceName, caller) {
var provider = providerInjector.get(serviceName + providerSuffix, caller);
return instanceInjector.invoke(provider.$get, provider, undefined, serviceName);
}));
由于会被invoke
调用,因此$get
的值必须为一个函数或者[deps, fn]
形式的数组。而函数的返回值,则可以为原始数据类型、对象、函数等等,并无限制。
总之,在调用provider(name, provider_)
的时候,会将(name + 'Provider', {$get:function(){/* ... */}})
键值对缓存在providerCache
中,在注入的时候,则会调用$get
函数,将其返回值进行注入,并缓存在instanceCache
中。
2. factory
源码如下:
function factory(name, factoryFn, enforce) {
return provider(name, {
$get: enforce !== false ? enforceReturnValue(name, factoryFn) : factoryFn
});
}
可以看到,factory
事实上是调用了provider
方法,而第二个参数factoryFn
,事实上也就是上面的$get
的值,因此应当符合对$get
的约束。
因此上面的例子可以这样写:
app.factory('greeting', function() {
return {
sayHello: function() {
console.log('hello world');
}
};
});
注意到factory
的源码中函数有第三个参数enforce
,也就是说是否强制返回值,如果函数没有返回值(包括显式返回值为undefined
)且第三个参数不为false
的时候,就回报错,相关源码如下:
function enforceReturnValue(name, factory) {
return function enforcedReturnValue() {
var result = instanceInjector.invoke(factory, this);
if (isUndefined(result)) {
throw $injectorMinErr('undef', "Provider '{0}' must return a value from $get factory method.", name);
}
return result;
};
}
然而在使用的过程中,会发现,即使显示声明了enforce
为false
,还是会报错,例如:
app.factory('greeting', function() {
console.log('hello world');
}, false);
// Error: [$injector:undef] Provider 'greeting' must return a value from $get factory method.
这主要是由supportObject
引起的,因为providerCache.$provide.factory
的值实际上是supportObject(factory)
,而supportObject
源码如下:
function supportObject(delegate) {
return function(key, value) {
if (isObject(key)) {
forEach(key, reverseParams(delegate));
} else {
return delegate(key, value);
}
};
}
可以发现,经过supportObject
处理后,函数的有效参数值最多只有两个了,因此第三个参数false
也就被忽略掉了。
3. service
源码如下:
function service(name, constructor) {
return factory(name, ['$injector', function($injector) {
return $injector.instantiate(constructor);
}]);
}
可以看到,实质上是调用了factory
,第二个参数constructor
是作为构造函数,最终返回的是该构造函数表示的类的一个实例。由于会调用$injector.instantiate(constructor)
,因此constructor
必须是一个函数或者[deps, fn]
的形式。
因此上面的例子可以写成:
app.service('greeting', function() {
this.sayHello = function() {
console.log('hello world');
};
});
// or
app.service('greeting', function() {
return {
sayHello: function() {
console.log('hello world');
}
};
});
4. value
源码如下:
function value(name, val) {
return factory(name, valueFn(val), false);
}
可以看到,也是调用了factory
。其中valueFn
的作用是将一个值包装为函数,源码如下:
function valueFn(value) {
return function() {
return value;
};
}
5. constant
源码如下:
function constant(name, value) {
assertNotHasOwnProperty(name, 'constant');
providerCache[name] = value;
instanceCache[name] = value;
}
可以看到,该方法只是对providerCache
和instanceCache
进行了属性设定。
6. value & constant
(1)value
设定的值是可以改变的,而constant
设定的值是不可变的,例如:
app.value('greeting', 'hello value')
.value('greeting', 'hello world ')
.controller('ctrl', ['greeting', function(greeting) {
console.log(greeting); // hello world
}]);
////////////////////
app.constant('greeting', 'hello constant')
.constant('greeting', 'hello world ')
.controller('ctrl', ['greeting', function(greeting) {
console.log(greeting); // hello constant
}]);
实现机制在函数setUpModuleLoader
中。部分相关源码如下:
var moduleInstance = {
_invokeQueue: invokeQueue,
value: invokeLater('$provide', 'value'),
constant: invokeLater('$provide', 'constant', 'unshift'),
// ... ...
};
function invokeLater(provider, method, insertMethod, queue) {
if (!queue) queue = invokeQueue;
return function() {
queue[insertMethod || 'push']([provider, method, arguments]);
return moduleInstance;
};
}
在调用value
的时候,用的是数组的push
方法;而调用constant
的时候,用的是数组的unshift
方法。在加载模块的时候,会依次执行_invokeQueue
中的内容。对于通过value
添加的值,会按照声明的顺序进行设定,因此后面的值会覆盖掉前面的值;而对于通过constant
添加的值,会按照声明的逆序进行设定,因此最后得到的值为第一次设定的值,从而就实现了常量的效果。
(2)在config
中,value
不可以被注入,而constant
可以被注入,例如:
app.constant('greeting', 'hello world')
.config(function(greeting) {
console.log(greeting); // hello world
});
////////////////////
app.value('greeting', 'hello world')
.config(function(greeting) {
console.log(greeting); // Error: [$injector:unpr] Unknown provider: greeting
});
这是因为在使用value
的时候,实际上是将greetingProvider
换存在了providerCache
中;而使用constant
的时候,则是将greeting
换存在了providerCache
中。然后在执行模块的_configBlocks
的时候,会在providerCache
中查找注入的依赖greeting
,因此对于value
来说,自然是找不到了。
7. provider, factory & service
这三者的关系非常密切,看如下例子:
function Greeting() {
this.sayHello = function() {
console.log('hello world');
};
}
app.provider('greetingFromProvider', function() {
this.$get = function() {
return new Greeting();
}
})
.factory('greetingFromFactory', function() {
return new Greeting();
})
.service('greetingFromService', Greeting)
.controller('ctrl', ['greetingFromProvider', 'greetingFromFactory', 'greetingFromService',
function(gp, gf, gs) {
gp.sayHello();
gf.sayHello();
gs.sayHello();
}
]);
可以简单理解为service
接收的参数是构造函数,factory
接收的参数是工厂函数,而该工厂函数是作为provider
中的$get
属性而存在的。但是provider
比factory
更加灵活可配置,因此可以理解为可配置的工厂函数。例如:
app.provider('greeting', function() {
var name = 'world';
this.$get = function() {
return {
sayHello: function() {
console.log('hello ' + name);
}
};
};
this.setName = function(newName) {
name = newName;
};
})
.config(function(greetingProvider) {
greetingProvider.setName('alex');
})
.controller('ctrl', function(greeting) {
greeting.sayHello(); // hello alex
});
AngularJS深入(5)——provider的更多相关文章
- AngularJS中的Provider们:Service和Factory等的区别
引言 看了很多文章可能还是不太说得出AngularJS中的几个创建供应商(provider)的方法(factory(),service(),provider())到底有啥区别,啥时候该用啥,之前一直傻 ...
- angularjs factory,service,provider 自定义服务的不同
angularjs框架学了有一段时间了,感觉很好用.可以把angularjs的app理解成php的class,controller是控制器,而内置服务和自定义服务就可以理解成models了.angul ...
- angularJS——自定义服务provider之$get
可以认为provider有三个部分: 第一部分是私有变量和私有函数,这些变量和函数会在以后被修改. 第二部分是在app.config函数里可以访问的变量和函数,所以,他们可以在其他地方使用之前被修改. ...
- [AngularJS + cryptoJS + Gravatar] Provider vs factory
Configurable Bits Need a Provider We want to be able to configure the characterLength before Tweetab ...
- AngularJS Factory Service Provider
先看看http://www.cnblogs.com/mbydzyr/p/3460501.html http://www.oschina.net/translate/angularjs-factory- ...
- AngularJS 中 Provider 的用法及区别
在一个分层良好的 Angular 应用中,Controller 这一层应该很薄.也就是说,应用里大部分的业务逻辑和持久化数据都应该放在 Service 里. 为此,理解 AngularJS 中的几个 ...
- AngularJS学习之依赖注入
1.什么是依赖注入:简称DI,是一种软件设计模式,在这种模式下,一个或更多的依赖(或服务)被注入(或者通过引用传递)到一个独立的对象(或客户端)中,然后成为了该客户端状态的一部分. 该模式分离了客户端 ...
- AngularJS进阶学习
参考:http://***/class/54f3ba65e564e50cfccbad4b 1. AJAX:Asynchronous JavaScript and XML(异步的 JavaScript ...
- AngularJS(15)-依赖注入
AngularJS 依赖注入 什么是依赖注入 wiki 上的解释是:依赖注入(Dependency Injection,简称DI)是一种软件设计模式,在这种模式下,一个或更多的依赖(或服务)被注入(或 ...
随机推荐
- easyui datagrid 通过复选框删除新追加的数据问题
之前写好的功能在保存好数据后再通过复选框删除是没有问题的,可现在想多追加几行,然后选择删除新追加的某几行或一行,通过$('#dg').datagrid('getChecked')方法返回选中行,然而返 ...
- 锋利的jQuery-4--动画方法总结简表
- 如何用jar命令对java工程进行打包
如何用jar命令对java工程进行打包 有时候为了更方便快捷的部署和执行Java程序,要把java应用程序打包成一个jar包.而这个基础的操作有时候也很麻烦,为了方便java程序员们能够方便的打包ja ...
- 新浪微博客户端(1)-实现Tabbar导航栏效果
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launc ...
- word表格断行的问题
word一个表格如果某一行的 内容 太多,就会自动跑到下一页去了 解决方法是: 在表格上点右键-> 属性 -> "行" -> 去掉"设置行高" ...
- tyvj1213 嵌套矩形
描述 有n个矩形,每个矩形可以用a,b来描述,表示长和宽.矩形X(a,b)可以嵌套在矩形Y(c,d)中当且仅当a<c,b<d或者b<c,a<d(相当于旋转X90度).例如 ...
- zhx's contest (矩阵快速幂 + 数学推论)
zhx's contest Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others) To ...
- document.documentElement和document.body的区别
网页中获取滚动条卷去部分的高度,可以通过 document.body.scrollTop 来获取,比如使div跟着滚动条滚动: <div id="div" style=&qu ...
- SGU-169 Numbers(找规律)
题目链接:http://acm.sgu.ru/problem.php?contest=0&problem=169 解题报告: P(n)定义为n的所有位数的乘积,例如P(1243)=1*2*3* ...
- IOS model的getter和setter方法
总结: 当使用 self.str1 = @"xxx";时, 系统自动调用 setter方法 param_str = self.str1; 自动调用getter方法注意: 只在对象点 ...