林炳文Evankaka原创作品。转载请注明出处http://www.jb51.cc/tag/http://blog.csdn.net/evankaka

本教程使用AngularJs版本:1.5.3

AngularJs GitHub:https://github.com/angular/angular.js/

AngularJs下载地址:https://angularjs.org/

用有过Spring的人都知道,Spring的核心思想就是DI(依赖注入,Dependency Injection)和IOC(控制反转 Invension of Controller),AngularJS的service其实就是用到了这里的DI思想,那么什么叫做DI?所谓DI,就是指对象是被动接受依赖类而不是自己主动去找,换句话说就是指对象不是从容器中查找它依赖的类,而是在容器实例化对象的时候主动将它依赖的类注入给它。我个人的理解就是,你在实例化一个类时,它会自动去实例化它里面所用到的其它实例。所以这就叫依赖注入。在Angular里面,services作为单例对象在需要到的时候被创建,只有在应用生命周期结束的时候(关闭浏览器)才会被清除。而controllers在不需要的时候就会被销毁了。

AngualrJs中可用来注入的有三种类型,service、factory、provider,这三种写法不样,用法也都不一样。其中,service只实例化一次,其实就是单例模式的思想。无论我们在什么地方注入我们的service,将永远使用同一个实例。所以对很多在controller层中的操作就可以放到serivce层中去。AngularJS提供例如许多内在的服务,如:$http,$route,$window,$location等。每个服务负责例如一个特定的任务,$http是用来创建AJAX调用,以获得服务器的数据。 $route用来定义路由信息等。内置的服务总是前缀$符号。

其中,三者的关系可以看成是下图:


一、Service使用详解

(1)定义

一个学用的service用法如下:

一般直接用this来操作数据、定义函数。

app.service('myService',function() {
    var privateValue = "I am Private";
    this.variable = "This is public";
    this.getPrivate = function() { return privateValue;
};
});

(2)AngularJS中使用DI添加Service的三种方法
方式1(内联注解,推荐使用):

app.controller('myController',['$scope','datefilter',function ($scope,datefilter) { }]);
方式2($inject注解):
 var MyController = function($scope,datefilter) {}
MyController.$inject = ['$scope','datefilter'];
someModule.controller('MyController',MyController);
方式3(隐式注解,不推荐使用):
app.controller('myController',datefilter) { });
推荐使用方式1的理由是:
写法上比方法2更简单明了
比方法3更可靠(由于Javascript可以被压缩,AngularJS又是通过解析服务名称找到对应Service的,因此Javascript压缩之后AngularJS将无法找到指定的Service,但字符串不会被压缩,因此单独以字符串指定Service的名称可以避免这个问题)
使用方式1或方式2的注意点:
由于上述第二点原因,AngularJS在编译Html时,由$injector将数组中Service的名称与方法体中的Service进行一一映射。这种映射关系必须遵守由AngularJS的约定:
数组中Service名称的个数必须与方法体中Service名称的个数一致
数组中Service的顺序必须与方法体中Serivce的顺序一致

(3)什么时候适合使用service()方法
service()方法很适合使用在功能控制比较多的service里面
注意:需要使用.config()来配置service的时候不能使用service()方法

(4)service使用实例

一个完整实例:

<!DOCTYPE html>
<html lang="zh" ng-app="myApp">
<head>
    <Meta charset="UTF-8">
    <title>AngularJS学习</title>
    <script type="text/javascript" src="./1.5.3/angular.min.js"></script>
</head>
<body>
    <div ng-controller="myCtrl">
        <button ng-click="getPrivate()">按钮一</button>
        <button ng-click="getPUbluc()">按钮二</button>
    </div>
    <div ng-controller = "myCtrl2">
    </div>
</body>
<script type="text/javascript">
var app = angular.module('myApp',[]);
app.controller('myCtrl',function($scope,myService) {
    $scope.getPrivate = function() {
        alert(myService.getPrivate());
    };
    $scope.getPUbluc = function() {
        alert(myService.variable);
    };
});
app.controller('myCtrl2',myService) {

});
app.service('myService',function() {
     console.log('instance myService');
    var privateValue = "I am Private";
    this.variable = "This is public";
    this.getPrivate = function() {
        return privateValue;
    };
});
</script>
</html>
效果如下:

另外,从这里可以看到两个controller注入同一个service,但是最终只实例化了一次!


注意:seivce定义的服务不能在.config中使用!只有provider定义的才可以

二、Factory使用详解

Factory 一般就是创建一个对象,然后在对这个对象添加方法与数据,最后将些对象返回即可。然后注入到Controller层中即可。

使用实例:
<!DOCTYPE html>
<html lang="zh" ng-app="myApp">
<head>
    <Meta charset="UTF-8">
    <title>AngularJS学习</title>
    <script type="text/javascript" src="./1.5.3/angular.min.js"></script>
</head>
<body>
    <div ng-controller="myCtrl">
        <button ng-click="getPrivate()">按钮一</button>
        <button ng-click="getPUbluc()">按钮二</button>
    </div>
    <div ng-controller = "myCtrl2">
    </div>
</body>
<script type="text/javascript">
var app = angular.module('myApp',myFactory) {
    $scope.getPrivate = function() {
        alert(myFactory.getPrivate());
    };
    $scope.getPUbluc = function() {
        alert(myFactory.variable);
    };
});
app.controller('myCtrl2',myFactory) {

});
app.factory('myFactory',function() {
    console.log('instance myFactory');
    var factory = {};
    var privateValue = "I am Private";
    factory.variable = "This is public";
    factory.getPrivate = function() {
        return privateValue;
    };
    return factory;
});
</script>
</html>
效果:

从下面可以看到只实例化了一次,两下Controller注入同一个factory,但是最终只实例化了一次!

记得一定要return一个Object对象,否则会报如下错误:


AngularJS中的Services:

三、provider使用详解

$provide服务负责告诉Angular如何创造一个新的可注入的东西:即服务。服务会被叫做供应商的东西来定义,你可以使用$provide来创建一个供应商。你需要使用$provide中的provider()方法来定义一个供应商,同时你也可以通过要求$provide被注入到一个应用的config函数中来获得$provide服务。使用方法是返回一个$get函数,注意在config阶段,只有provider能被注入。其它用法和service一样。
下面是一个实例。
<!DOCTYPE html>
<html lang="zh" ng-app="myApp">
<head>
    <Meta charset="UTF-8">
    <title>AngularJS学习</title>
    <script type="text/javascript" src="./1.5.3/angular.min.js"></script>
</head>
<body>
    <div ng-controller="myCtrl1">
        <button ng-click = "onclick1()">请点击我1</button>
    </div>
    <div ng-controller="myCtrl2">
         <button ng-click = "onclick2()">请点击我2</button>
    </div>
</body>
<script type="text/javascript">
var app = angular.module('myApp',[]);
app.controller('myCtrl1',testProvider) {
    $scope.onclick1 = function() {
        testProvider("林炳文Evankaka");
    };
});
app.controller('myCtrl2',testProvider) {
      $scope.onclick2 = function() {
        testProvider("我到底是谁");
    };
});

app.provider('testProvider',function(){
    console.log('instance testProvider');
    var f = function(name) {
       alert("Hello," + name);
    };
    this.$get = function(){ //一定要有!
      return f;
   };
});
</script>
</html>
效果:

从以下可以看到只实例化了一次:
再来做一个provider实例化的时间测试:
<!DOCTYPE html>
<html lang="zh" ng-app="myApp">

<head>
    <Meta charset="UTF-8">
    <title>AngularJS学习</title>
    <script type="text/javascript" src="./1.5.3/angular.min.js"></script>
</head>

<body>
    <div ng-controller="myCtrl1">
        <button ng-click="onclick1()">请点击我1</button>
    </div>
    <div ng-controller="myCtrl2">
        <button ng-click="onclick2()">请点击我2</button>
    </div>
</body>
<script type="text/javascript">
var app = angular.module('myApp',function($scope) {
/*    $scope.onclick1 = function() {
        test("林炳文Evankaka");
    };*/
});
app.controller('myCtrl2',function($scope) {
/*    $scope.onclick2 = function() {
        test("我到底是谁");
    };*/
});
app.provider('test',function() {
    console.log('instance test');
    var f = function(name) {
        alert("Hello," + name);
    };
    this.$get = function() { //一定要有!
        return f;
    };
});
/*app.config(function(testProvider) {
    testProvider('I am config');
});*/

app.config(function($provide) {
    $provide.provider('greeting',function() {
        this.$get = function() {
            return function(name) {
                alert("Hello," + name);
            };
        };
    });
  /*  greetingProvider('ff');*/
});
</script>

</html>
页面刷新后,我们发现即使不注入这个providrer,但也它也进行实例化了,而service/factory则是第一次注入时才会初始化。而也这是为什么它可以注入到config的一个原因吧!
什么时候使用provider()方法

(1)当我们希望在应用开始前对service进行配置的时候就需要使用到provider()。比如,我们需要配置services在不同的部署环境里面(开发,演示,生产)使用不同的后端处(2)理的时候就可以使用到了
当我们打算发布开源provider()也是首选创建service的方法,这样就可以使用配置的方式来配置services而不是将配置数据硬编码写到代码里面。


四、Service、Factory、Provider三者区别


1 ) 用 Factory 就是创建一个对象,为它添加属性,然后把这个对象返回出来。你把 service 传进 controller 之后,在 controller 里这个对象里的属性就可以通过 factory 使用了。
2) Service 是用"new"关键字实例化的。因此,你应该给"this"添加属性,然后 service 返回"this"。你把 service 传进 controller 之后,在controller里 "this" 上的属性就可以通过 service 来使用了。
3) Providers 是唯一一种你可以传进 .config() 函数的 service。当你想要在 service 对象启用之前,先进行模块范围的配置,那就应该用 provider。
4)Factory/service是第一个注入时才实例化,而provider不是,它是在config之前就已实例 化好
同一函数三种不同写法:
<!DOCTYPE html>
<html lang="zh" ng-app="myApp">
<head>
    <Meta charset="UTF-8">
    <title>AngularJS学习</title>
    <script type="text/javascript" src="./1.5.3/angular.min.js"></script>
</head>
<body>
    <div ng-controller="myCtrl">
        <p>{{ output1 }}</p>
        <p>{{ output2 }}</p>
        <p>{{ output3 }}</p>
    </div>
    <div ng-controller="myCtrl2">
    </div>
</body>
<script type="text/javascript">
var app = angular.module('myApp',testService,testFactory,testProvider) {
    $scope.output1 = testService.lable;
    $scope.output2 = testFactory.lable();
    $scope.output3 = testProvider;
});
app.controller('myCtrl2',testProvider) {
});
app.service('testService',function(){
      console.log('instance testService');
     this.lable = 'this is service';
});
app.factory('testFactory',function () {
      console.log('instance testFactory');
     return{
        lable: function(){
        return 'this is factory';
        }
    };
});
app.provider('testProvider',function(){
    console.log('instance testProvider');
    this.$get = function(){
        return 'this is provider';
    };
});
</script>
</html>


看console的输出,可以知道都只实例 化了一次:


参考文章:
http://www.cnblogs.com/wushangjue/p/4533953.html?utm_source=tuicool&utm_medium=referral
http://www.oschina.net/translate/angularjs-factory-vs-service-vs-provider
http://my.oschina.net/tanweijie/blog/295067
http://www.360doc.com/content/15/1010/18/28162318_504719466.shtml

跟我学AngularJs:Service、Factory、Provider依赖注入使用与区别的更多相关文章

  1. html5录音功能实战示例

    这篇文章主要介绍了html5录音功能实战示例的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

  2. 基于 HTML5 WebGL 实现的医疗物流系统

    物联网( IoT ),简单的理解就是物体之间通过互联网进行链接。这篇文章给大家介绍基于 HTML5 WebGL 实现的医疗物流系统,感兴趣的朋友跟随小编一起看看吧

  3. HTML5页面无缝闪开的问题及解决方案

    这篇文章主要介绍了HTML5页面无缝闪开方案,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

  4. HTML5跳转小程序wx-open-launch-weapp的示例代码

    这篇文章主要介绍了HTML5跳转小程序wx-open-launch-weapp的相关知识,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

  5. ios – 实例化ViewController w / xib时的帧大小不正确

    我有一个视图控制器,看起来像:当我通过让spotViewController=SpotViewController实例化视图控制器并将其推送到导航控制器时,结果框架在viewDidLoad和viewWillAppear中都是不正确的.它给了我这是界面构建器中的大小.为什么会发生这种情况以及使用xib实例化视图控制器以确保帧正确的正确方法是什么?解决方法这将解决您的问题,UIViewController从xib加载,在viewDidLoad期间保持xib大小:

  6. ios – 如何在从SlideMenuViewController实例化的ViewControllers中打开TabBarController

    我希望在所有ViewControllers中都有tabBar.我已经实现了SWRevealViewController,它有两个链接的视图控制器,一个是TabBarController,另一个是TableViewController我希望在我的所有ViewControllers中都有与TableViewController相同的tabBar.解决方法IwanttohavetabBarinallmy

  7. ios – 无法实例化名为ADBannerView的类

    我试图在我的应用程序的底部插入一个iad横幅,但在遵循教程后不断收到错误.代码如下.我一直收到以下错误*由于未捕获的异常’NSInvalidUnarchiveOperationException’终止应用程序,原因:’无法实例化名为ADBannerView的类’*第一次抛出调用堆栈:解决方法请确保您已添加“iAd.framework”…要执行此操作,请转到“App.Target”,“General

  8. ios – 通过AppDelegate以Storyboard方式实例化View Controller

    我正在建立一个应用程序–当第一次启动它时,它要求用户做两件事情:>选择国家>接受T&Cs从那里到家庭视图控制器.我目前面临的问题是将第一个视图控制器从我的应用程序代理推送到屏幕上.我正在使用storyboards/Xcode5/iOS7这是我提出的代码:问题是应用程序崩溃时遇到以下错误的最后一行代码:*Terminatingappduetouncaughtexception‘NSinvalidar

  9. ios – 与Xcode Bots持续集成

    我想使用Xcode机器人进行持续集成.我已经安装了OSXMavericks和Server(版本3).我可以使用Xcode5.0.1创建机器人.在集成时,它正在成功地执行分析测试,但总是最终的集成结果是失败的.IntegrationFailed.Unexpectedinternalservererror.Seetheintegration’slogsformoredetails.`我没有从服务器错误

  10. ios – 使用UINavigationController的initWithNavigationBarClass的正确方法:toolbarClass:

    从苹果文档我知道一个UiNavigationController可以使用另一个Uinavigationbar使用initWithNavigationBarClass:toolbarClass:方法实例化.如何通过定制的UiNavigationBar子类和IB正确地执行此操作?解决方法您可以像这样使用它来初始化导航控制器,这里CustomNavigationBar是通过子类化UINavigation

随机推荐

  1. Angular2 innerHtml删除样式

    我正在使用innerHtml并在我的cms中设置html,响应似乎没问题,如果我这样打印:{{poi.content}}它给了我正确的内容:``但是当我使用[innerHtml]=“poi.content”时,它会给我这个html:当我使用[innerHtml]时,有谁知道为什么它会剥离我的样式Angular2清理动态添加的HTML,样式,……

  2. 为Angular根组件/模块指定@Input()参数

    我有3个根组件,由根AppModule引导.你如何为其中一个组件指定@input()参数?也不由AppModalComponent获取:它是未定义的.据我所知,你不能将@input()传递给bootstraped组件.但您可以使用其他方法来做到这一点–将值作为属性传递.index.html:app.component.ts:

  3. angular-ui-bootstrap – 如何为angular ui-bootstrap tabs指令指定href参数

    我正在使用角度ui-bootstrap库,但我不知道如何为每个选项卡指定自定义href.在角度ui-bootstrap文档中,指定了一个可选参数select(),但我不知道如何使用它来自定义每个选项卡的链接另一种重新定义问题的方法是如何使用带有角度ui-bootstrap选项卡的路由我希望现在还不算太晚,但我今天遇到了同样的问题.你可以通过以下方式实现:1)在控制器中定义选项卡href:2)声明一个函数来改变控制器中的散列:3)使用以下标记:我不确定这是否是最好的方法,我很乐意听取别人的意见.

  4. 离子框架 – 标签内部的ng-click不起作用

    >为什么标签标签内的按钮不起作用?>但是标签外的按钮(登陆)工作正常,为什么?>请帮我解决这个问题.我需要在点击时做出回复按钮workingdemo解决方案就是不要为物品使用标签.而只是使用divHTML

  5. Angular 2:将值传递给路由数据解析

    我正在尝试编写一个DataResolver服务,允许Angular2路由器在初始化组件之前预加载数据.解析器需要调用不同的API端点来获取适合于正在加载的路由的数据.我正在构建一个通用解析器,而不是为我的许多组件中的每个组件设置一个解析器.因此,我想在路由定义中传递指向正确端点的自定义输入.例如,考虑以下路线:app.routes.ts在第一个实例中,解析器需要调用/path/to/resourc

  6. angularjs – 解释ngModel管道,解析器,格式化程序,viewChangeListeners和$watchers的顺序

    换句话说:如果在模型更新之前触发了“ng-change”,我可以理解,但是我很难理解在更新模型之后以及在完成填充更改之前触发函数绑定属性.如果您读到这里:祝贺并感谢您的耐心等待!

  7. 角度5模板形式检测形式有效性状态的变化

    为了拥有一个可以监听其包含的表单的有效性状态的变化的组件并执行某些组件的方法,是reactiveforms的方法吗?

  8. Angular 2 CSV文件下载

    我在springboot应用程序中有我的后端,从那里我返回一个.csv文件WheniamhittingtheURLinbrowsercsvfileisgettingdownloaded.现在我试图从我的角度2应用程序中点击此URL,代码是这样的:零件:服务:我正在下载文件,但它像ActuallyitshouldbeBook.csv请指导我缺少的东西.有一种解决方法,但您需要创建一个页面上的元

  9. angularjs – Angular UI-Grid:过滤后如何获取总项数

    提前致谢:)你应该避免使用jQuery并与API进行交互.首先需要在网格创建事件中保存对API的引用.您应该已经知道总行数.您可以使用以下命令获取可见/已过滤行数:要么您可以使用以下命令获取所选行的数量:

  10. angularjs – 迁移gulp进程以包含typescript

    或者我应该使用tsc作为我的主要构建工具,让它解决依赖关系,创建映射文件并制作捆绑包?

返回
顶部