provider实现原理

原生angular中的provider和$injector主要作用是:

  1. 注册组件(controller directive service)
  2. 解决组件间的依赖关系
  3. 初始化组件

接下来提供简单的实现

var Provider={

    //保存所有组件对应的工厂函数
    _providers:{},
    
    //注册组件
    _register: function(name,fn) {
      this._providers[name]=fn;
    },
    
    //要来注册service
    service: function(name,fn) {
      return this._register(name+Provider.SERVICE_SUFFIX,fn);
    },
    
    //要来注册directive 
    directive: function(name,fn) {
      return this._register(name+Provider.DIRECTIVE_SUFFIX,fn);
    },
    
    //要来注册controller
    //初始化controller缓存里保存的是工厂函数,可以多次初始化同一个controller
    controller: function(name,fn) {
      return this._register(name,function() {
        return fn;
      })
    }
}

上面实现了3种组件的注册过程,接下来实现组件初始化及解决依赖

var Provider = {
    // ...
    
   //缓存初始化后的组件
    _cache: {},
    
    //返回一个数组 数组为当前函数工厂函数的传参,也就是依赖组件的名称
    annotate: function (fn) {
        var res = fn.toString()
            .replace(/((\/\/.*$)|(\/\*[\s\S]*?\*\/))/mg, '')
            .match(/\((.*?)\)/);
        if (res && res[1]) {
            return res[1].split(',').map(function (d) {
                return d.trim();
            });
        }
        return [];
    },
    
    //接下来两个函数将互相调用,调用组件的初始化工厂函数,并循环遍历解决依赖(以及依赖的依赖)
    //解决依赖的意思就是将依赖组件的名称转换成对应组件的工厂函数,传入。
    
    //传入组件名,locals为所有本地依赖的对象集合,返回初始化后的组件
    get: function(name,locals) {
    
      //如果已有缓存的初始化工厂函数,返回他
      if (this._cache[name]) {
          return this._cache[name];
      }
      var provider=this._providers[name];
      if (!provider || typeof provider !=='function') {return;}
      return (this._cache[name] = this.invoke(provider,locals));
    },
    
    //用于初始化一个组件。如果该组件依赖的组件还未初始化,则会先初始化依赖的组件
    invoke: function(fn,locals) {
      locals= locals || {};
      
        //通过依赖名找出依赖的工厂函数
      var deps=this.annotate(fn).map(function(item) {
        return locals[item] || this.get(item,locals);
      },this);
      return fn.apply(null,deps);
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 136,350评论 19 139
  • 工厂模式类似于现实生活中的工厂可以产生大量相似的商品,去做同样的事情,实现同样的效果;这时候需要使用工厂模式。简单...
    舟渔行舟阅读 8,075评论 2 17
  • 单例模式 适用场景:可能会在场景中使用到对象,但只有一个实例,加载时并不主动创建,需要时才创建 最常见的单例模式,...
    Obeing阅读 2,288评论 1 10
  • Angular面试题 一、ng-show/ng-hide与ng-if的区别? 第一点区别是,ng-if在后面表达式...
    w_zhuan阅读 5,676评论 0 26
  • 1、angularjs的几大特性是什么? 双向数据绑定、依赖注入、模板、指令、MVC/MVVM 2、列举几种常见的...
    秀才JaneBook阅读 1,598评论 0 22

友情链接更多精彩内容