CacheFactoryProvider 简介

源码里是这么描述的:

Factory that constructs {@link $cacheFactory.Cache Cache} objects and gives access to

them.

意思就是通过cacheFactory可以构造一个Cache对象来给予访问和执行权限。

这个Cache对象官方文档是这么说的:

A cache object used to store and retrieve data, primarily used by $http and the script directive to cache templates and other data.

用我自己的话来说就是 提供存储和访问缓存对象的服务,angular内部主要被$http,script指令用于

缓存template和其他数据。我们自己可以在Controller内部使用。

CacheFactoryProvider 用法

  1. <!doctype html>
  2. <html lang="en" ng-app="myapp">
  3. <head>
  4. <meta charset="UTF-8">
  5. <meta name="Generator" content="EditPlus®">
  6. <meta name="Author" content="">
  7. <meta name="Keywords" content="">
  8. <meta name="Description" content="">
  9. <title>Document</title>
  10. <script src="js/angular.js"></script>
  11. </head>
  12. <body>
  13. <div ng-controller="MyController">
  14. </div>
  15. <script >
  16. var app=angular.module('myapp',[]);
  17. app.controller('MyController',function($scope,$cacheFactory){
  18. var myCache = $cacheFactory("myCache",{capacity: 6});
  19. //var myCache1 = $cacheFactory("myCache",{capacity: 6}); //会报错
  20. myCache.put("name","john");
  21. myCache.put("name1","wonder");
  22. myCache.put("name","john");
  23. });
  24. app.controller('getCacheController',['$scope','$cacheFactory',
  25. function($scope,$cacheFactory){
  26. var cache = $cacheFactory.get('myCache');
  27. var name = cache.get('name');
  28. console.log(name); //打印john
  29. }]);
  30. </script>
  31. </body>
  32. </html>

看了上面这个一个简单的例子,读者可能会产生如下疑惑:

  1. 不是名为CacheFactoryProvider吗,怎么在代码里只看到cacheFactory呢?
  2. cacheFactory是怎么存储对象的?

    下面我们来依次解答这两个问题

$cacheFactory的注入

我们首先来看第一个问题,这个问题要牵涉到angular里面的依赖注入机制,我们前面的分析也讲过,

angular会在启动之前通过调用publishExternalAPI 函数先发布一些扩展API,同时定义ng

模块,在定义ng模块的时候就传入了注入provider的方法

  1. angularModule('ng', ['ngLocale'], ['$provide',
  2. //通过参数注入$provide
  3. function ngModule($provide) {
  4. ///部分代码省略
  5. $provide.provider({
  6. $cacheFactory: $CacheFactoryProvider,
  7. });
  8. }])

$cacheFactory出现了,它是通过javascript的键值对象作为键传给provider方法。那么它是如何存储

对象的呢?首先我们看它的定义:

CacheFactoryProvider的定义

内部定义了依赖注入核心的$get方法,$get方法返回cacheFactory方法(也就是上面实例代码里的

$cacheFactory参数)。

  1. function $CacheFactoryProvider() {
  2. //定义$get方法供依赖调用
  3. //controller中获取cacheFactory时会调用此方法
  4. //这个$get方法也是获取provider的关键方法
  5. this.$get = function() {
  6. var caches = {};//闭包的一个运用
  7. function cacheFactory(cacheId, options) {
  8. //部分代码省略
  9. //可以用if来判断
  10. if (cacheId in caches) {//如果caches中已经存在cacheId
  11. //实例代码里抛出的错误就在此处、
  12. //统一调用minErr函数
  13. throw minErr('$cacheFactory')('iid', "CacheId '{0}' is already taken!"
  14. , cacheId);
  15. }
  16. var size = 0,
  17. //把options 和{id:cacheId} 放入{} 中 不是深拷贝
  18. stats = extend({}, options, {id: cacheId}),
  19. data = createMap(),//通过Object.create(null) 创建个空对象
  20. capacity = (options && options.capacity) || Number.MAX_VALUE,
  21. lruHash = createMap(),
  22. freshEnd = null,
  23. staleEnd = null;
  24. //返回caches中的一个对象
  25. return caches[cacheId] = {
  26. //省略部分代码
  27. //存储里讲解
  28. put:function(key,value){
  29. },
  30. get: function(key) {
  31. },
  32. remove: function(key) {
  33. },
  34. removeAll: function() {
  35. },
  36. destroy: function() {
  37. },
  38. info: function() {
  39. }
  40. }
  41. //刷新节点次序
  42. function refresh(entry) {
  43. }
  44. //
  45. function link(nextEntry, prevEntry) {
  46. }
  47. }
  48. //所有的缓存
  49. cacheFactory.info = function() {
  50. var info = {};
  51. forEach(caches, function(cache, cacheId) {
  52. info[cacheId] = cache.info();
  53. });
  54. return info;
  55. };
  56. cacheFactory.get = function(cacheId) {
  57. return caches[cacheId];
  58. };
  59. return cacheFactory;
  60. }
  61. }

CacheFactoryProvider的存储

存储分为这几个核心方法:put,refresh,remove,link

put函数

value会放入data对象中,key会放入lruHash链表

  1. put: function(key, value) {
  2. if (isUndefined(value)) return;
  3. //如果设定的capcity小于maxvalue
  4. if (capacity < Number.MAX_VALUE) {
  5. //lruHash 存了当前的key 还有可能是 p 和n (previous和next)
  6. var lruEntry = lruHash[key] || (lruHash[key] = {key: key});
  7. //刷新各节点的次序
  8. refresh(lruEntry);//把当前entry放入链表末尾
  9. }
  10. //如果key 在data里不存在 那么增加size
  11. if (!(key in data)) size++;
  12. data[key] = value;
  13. //当大于capacity时 会清除最早加入的那个
  14. if (size > capacity) {
  15. this.remove(staleEnd.key);//移除淘汰节点stableEnd
  16. }
  17. return value;
  18. }

get函数

Retrieves named data stored in the {@link $cacheFactory.Cache Cache} object

获取存储在cache对象中的指定数据

  1. get: function(key) {
  2. if (capacity < Number.MAX_VALUE) {
  3. var lruEntry = lruHash[key];
  4. if (!lruEntry) return;
  5. // 获取first的时候 因为staleEnd为first 所以会让staleEnd指向 second
  6. // 内部会执行link 使得 second.p = null
  7. // first.p = third third.n = first
  8. //stableEnd为 second freshEnd为first
  9. refresh(lruEntry);
  10. }
  11. return data[key];
  12. }

remove函数

Removes an entry from the {@link $cacheFactory.Cache Cache} object.

从cache对象删除一个entry

  1. remove: function(key) {
  2. //如果capacity小于maxvalue
  3. if (capacity < Number.MAX_VALUE) {
  4. //先取出当前key的entry
  5. var lruEntry = lruHash[key];
  6. if (!lruEntry) return;
  7. //第一次超过时 freshEnd 为third lryEntry为first
  8. if (lruEntry == freshEnd) freshEnd = lruEntry.p;
  9. //第一次超过时 staleEnd 为first lryEntry为first
  10. //所以 会让 stalEnd 指向second 以便于下次移除时
  11. if (lruEntry == staleEnd) staleEnd = lruEntry.n;
  12. //把淘汰节点的一个节点选中
  13. //第一次超过时 lryEntry.n为 second lryEntry.p 为null
  14. //执行结果为 second.p = null
  15. link(lruEntry.n,lruEntry.p);
  16. //把当前key从lruHash中删除
  17. delete lruHash[key];
  18. }
  19. if (!(key in data)) return;
  20. delete data[key];
  21. size--;
  22. }

refresh函数

makes the entry the freshEnd of the LRU linked list。

把entry 放入链表的末尾

  1. function refresh(entry) {
  2. if (entry != freshEnd) {
  3. if (!staleEnd) { //staleEnd为空那么就让他指向当前entry
  4. staleEnd = entry;
  5. } else if (staleEnd == entry) {
  6. //如果淘汰节点等于当前节点
  7. staleEnd = entry.n; //用于把 当前的下一个节点 用作淘汰节点
  8. }
  9. //放入第一个元素时 entry.n,entry.p都为undefined
  10. link(entry.n, entry.p); //当前的上一个节点 和当前的下一个节点
  11. link(entry, freshEnd); // 当前的节点 和 最新的末尾节点
  12. freshEnd = entry;
  13. freshEnd.n = null;
  14. //第一次执行完 结果为: freshEnd = first staleEnd为first
  15. //first.p=null first.n=null
  16. //第二次执行完 结果为:freshEnd = second staleEnd为first
  17. // first.p=null first.n= second
  18. // scecond.p = first scecond.n = null
  19. //第三次执行完 freshEnd = third staleEnd为first first.p=null
  20. //first.n= second
  21. // second.p = first second.n = null
  22. // third.p = second third.n = null
  23. }
  24. }

link函数

bidirectionally(双向链表) links two entries of the LRU linked list

双向链接链表里的两个元素。

  1. function link(nextEntry, prevEntry) {
  2. //undefined 不等于undefined
  3. if (nextEntry != prevEntry) {
  4. //
  5. if (nextEntry) nextEntry.p = prevEntry;
  6. //p stands for previous, 'prev' didn't minify
  7. if (prevEntry) prevEntry.n = nextEntry;
  8. //n stands for next, 'next' didn't minify
  9. }
  10. }

欢迎关注我的公众号,获取最新源码解析文章!

参考资料:
  1. 【算法】—— LRU算法
  2. 缓存淘汰算法--LRU算法
  3. 漫画:什么是LRU算法?

angular源码剖析之Provider系列--CacheFactoryProvider的更多相关文章

  1. angular源码剖析之Provider系列--QProvider

    QProvider 简介 源码里是这么描述的: A service that helps you run functions asynchronously, and use their return ...

  2. angular源码阅读,依赖注入的原理:injector,provider,module之间的关系。

    最开始使用angular的时候,总是觉得它的依赖注入方式非常神奇. 如果你跳槽的时候对新公司说,我曾经使用过angular,那他们肯定会问你angular的依赖注入原理是什么? 这篇博客其实是angu ...

  3. 【安卓网络请求开源框架Volley源码解析系列】定制自己的Request请求及Volley框架源码剖析

    通过前面的学习我们已经掌握了Volley的基本用法,没看过的建议大家先去阅读我的博文[安卓网络请求开源框架Volley源码解析系列]初识Volley及其基本用法.如StringRequest用来请求一 ...

  4. 【java集合框架源码剖析系列】java源码剖析之TreeSet

    本博客将从源码的角度带领大家学习TreeSet相关的知识. 一TreeSet类的定义: public class TreeSet<E> extends AbstractSet<E&g ...

  5. 【java集合框架源码剖析系列】java源码剖析之HashSet

    注:博主java集合框架源码剖析系列的源码全部基于JDK1.8.0版本.本博客将从源码角度带领大家学习关于HashSet的知识. 一HashSet的定义: public class HashSet&l ...

  6. 【java集合框架源码剖析系列】java源码剖析之TreeMap

    注:博主java集合框架源码剖析系列的源码全部基于JDK1.8.0版本.本博客将从源码角度带领大家学习关于TreeMap的知识. 一TreeMap的定义: public class TreeMap&l ...

  7. 【java集合框架源码剖析系列】java源码剖析之ArrayList

    注:博主java集合框架源码剖析系列的源码全部基于JDK1.8.0版本. 本博客将从源码角度带领大家学习关于ArrayList的知识. 一ArrayList类的定义: public class Arr ...

  8. 【java集合框架源码剖析系列】java源码剖析之LinkedList

    注:博主java集合框架源码剖析系列的源码全部基于JDK1.8.0版本. 在实际项目中LinkedList也是使用频率非常高的一种集合,本博客将从源码角度带领大家学习关于LinkedList的知识. ...

  9. 【java集合框架源码剖析系列】java源码剖析之HashMap

    前言:之所以打算写java集合框架源码剖析系列博客是因为自己反思了一下阿里内推一面的失败(估计没过,因为写此博客已距阿里巴巴一面一个星期),当时面试完之后感觉自己回答的挺好的,而且据面试官最后说的这几 ...

随机推荐

  1. 【paddle学习】词向量

    http://spaces.ac.cn/archives/4122/   关于词向量讲的很好 上边的形式表明,这是一个以2x6的one hot矩阵的为输入.中间层节点数为3的全连接神经网络层,但你看右 ...

  2. BUPT复试专题—三元组(2016)

    题目描述 给你一个长度为m的数组(数组元素从0到m-1),如果数组里有a[i]+a[j]==a[k](i,j,k大于等于0并且小于m),便称之为三元组.现在给你一个数组,让你求三元组的个数. 例如m为 ...

  3. ubuntu 添加和删除用户

    Without a home directory sudo useradd myuser With home directory sudo useradd -m myuser Then set the ...

  4. UVA 1482 - Playing With Stones(SG打表规律)

    UVA 1482 - Playing With Stones 题目链接 题意:给定n堆石头,每次选一堆取至少一个.不超过一半的石子,最后不能取的输,问是否先手必胜 思路:数值非常大.无法直接递推sg函 ...

  5. mmall 项目实战(一)项目初始化

    1.创建 数据库 及 表 数据脚本: /* Navicat Premium Data Transfer Source Server : 182.92.82.103 Source Server Type ...

  6. BIOS维修技术

    BIOS是电脑中最基础且最重要的程序,为电脑提供最低级且最直接的硬件控制,电脑的原始操作都是依照固化在BIOS里的程序来完成的.因此如果BIOS出现故障将会导致影响电脑的正常工作.BIOS故障有很多, ...

  7. SGU 194 Reactor Cooling 无源汇带上下界可行流

    Reactor Cooling time limit per test: 0.5 sec. memory limit per test: 65536 KB input: standard output ...

  8. 【python】How to change the Jupyter start-up folder

    Copy the Jupyter Notebook launcher from the menu to the desktop. Right click on the new launcher and ...

  9. (testng多个class文件执行时混乱,不是等一个class内的所有methods执行完再去执行下一个class内的内容)问题的解决

    问题描述如下: We use TestNG and Selenium WebDriver to test our web application. Now our problem is that we ...

  10. Hibernate commit() 和flush() 的区别

    <<精通Hibernate java对象持久化技术详解>> ,flush()方法进行清理缓存的操作,执行一系列的SQL语句,但不会提交事务;commit()方法会先调用flus ...