数据缓存是为了解决内存泄露,他的原理是,当我们将数据存储到一个对象上面,实际上是将所有的数据存到一个单独的数据对象里,而这个对象只提供一个接口,这个接口可以访问自己存在数据对象里自己的数据。

这是一个简单的数据缓存

var cache= {};
function set(obj,name,value){
//在对象上存储一个属性,用作访问数据的接口
obj.expando = 1;
//初始化对象,并在我们接口所提供的位置存放数据
cache[obj.expando] = {}
cache[obj.expando][name] = value
}
function get(obj,name){
//实际就是通过对象上的expando指定在数据中位置去访问的自己的数据。
return cache[obj.expando][name]
}
var obj2 = {};
set(obj2,"name","winder");
console.log(get(obj2,"name")) //winder
console.log(obj2) // Object { expando: 1}
console.log(cache[1]) // Object { name: "winder" }

由代码可以很清楚看到,我们实际访问的是数据对象中,1引用的对象;而如何找到数据的位置,则由存储在对象上的一个接口属性提供。

当我们把他运用在jQuery中

//mini的jQuery框架。
var jQuery = function( selector, context ) {
return new jQuery.fn.init( selector, context );
};
jQuery.fn = jQuery.prototype = {
selector: "",
init: function(selector){
var result = document.querySelectorAll(selector);
for(var i = 0;i < result.length;i++){
this[i] = result[i]
}
this.length = result.length;
},
constructor : jQuery
}
jQuery.fn.init.prototype = jQuery.fn; //定义date构造函数
function Data(){
this.cache = {}
//默认带有随机数属性
this.expando = Math.random()
}
//设置在数据对象中的位置的辅助属性
Data.uid = 1;
Data.prototype = {
//判断对象是否有接口属性,
//没有接口时,新建接口
key: function( owner ) { var descriptor = {},
unlock = owner[ this.expando ];
if ( !unlock ) {
unlock = Data.uid++; descriptor[ this.expando ] = { value: unlock };
Object.defineProperties( owner, descriptor ); }
//确保在缓存中存在
if ( !this.cache[ unlock ] ) {
this.cache[ unlock ] = {};
}
return unlock;
},
//向缓存中添加数据
set: function( owner, data, value ) {
var unlock = this.key( owner );
var cache = this.cache[ unlock ];
cache[ data ] = value; return cache;
},
//获取数据
get: function( owner, key ) {
var cache = this.cache[ this.key( owner ) ]; return cache[ key ];
},
//access实际就是一个访问路由,根据参数判断是设置还是获取
access: function( owner, key, value ) {
//访问
if ((typeof key === "string") && value === undefined ) { return this.get( owner, key );
}
//设置
this.set( owner, key, value ); return value
},
}
var data_user = new Data()
jQuery.data = function( elem, name, data ) {
return data_user.access( elem, name, data );
}
//测试
var obj = {}
jQuery.data(obj,"name","winder")
console.log(jQuery.data(obj,"name")) //winder
console.log(obj) //里面的里面是0.6637197330750032:1
console.log(data_user.cache[1])//Object { name: "winder" }

采用向对象中添加随机数属性,是为了屏蔽除内部外任何访问,保证缓存中数据安全。

添加jQuery对象方法

//mini的jQuery框架。
var jQuery = function( selector, context ) {
return new jQuery.fn.init( selector, context );
};
jQuery.fn = jQuery.prototype = {
selector: "",
init: function(selector){
var result = document.querySelectorAll(selector);
for(var i = 0;i < result.length;i++){
this[i] = result[i]
}
this.length = result.length;
},
constructor : jQuery
}
jQuery.fn.init.prototype = jQuery.fn; //定义date构造函数
function Data(){
this.cache = {}
//默认带有随机数属性
this.expando = Math.random()
}
//设置在数据对象中的位置的辅助属性
Data.uid = 1;
Data.prototype = {
//判断对象是否有接口属性,
//没有接口时,新建接口
key: function( owner ) { var descriptor = {},
unlock = owner[ this.expando ];
if ( !unlock ) {
unlock = Data.uid++; descriptor[ this.expando ] = { value: unlock };
Object.defineProperties( owner, descriptor ); }
//确保在缓存中存在
if ( !this.cache[ unlock ] ) {
this.cache[ unlock ] = {};
}
return unlock;
},
//向缓存中添加数据
set: function( owner, data, value ) {
var unlock = this.key( owner );
var cache = this.cache[ unlock ];
cache[ data ] = value; return cache;
},
//获取数据
get: function( owner, key ) {
var cache = this.cache[ this.key( owner ) ]; return cache[ key ];
},
//access实际就是一个访问路由,根据参数判断是设置还是获取
access: function( owner, key, value ) {
//访问
if ((typeof key === "string") && value === undefined ) { return this.get( owner, key );
}
//设置
this.set( owner, key, value ); return value
},
}
var data_user = new Data()
jQuery.data = function( elem, name, data ) {
return data_user.access( elem, name, data );
}
//简化的jQuery对象API方法
jQuery.fn.data = function( key, value ){
for(var i = 0;i < this.length; i++){
return jQuery.data(this[i],key,value)
}
}
//测试
var $emample = jQuery("#example")
$emample.data("name","winder");
console.log($emample.data("name")) //winder

这是一个极简化的API。而在原版jQuery当中,细致考虑了多种情况。

jQuery源码笔记——数据缓存的更多相关文章

  1. 【菜鸟学习jquery源码】数据缓存与data()

    前言 最近比较烦,深圳的工作还没着落,论文不想弄,烦.....今天看了下jquery的数据缓存的代码,参考着Aaron的源码分析,自己有点理解了,和大家分享下.以后也打算把自己的jquery的学习心得 ...

  2. jQuery源码解读 - 数据缓存系统:jQuery.data

    jQuery在1.2后引入jQuery.data(数据缓存系统),主要的作用是让一组自定义的数据可以DOM元素相关联——浅显的说:就是让一个对象和一组数据一对一的关联. 一组和Element相关的数据 ...

  3. jQuery1.9.1源码分析--数据缓存Data模块

    jQuery1.9.1源码分析--数据缓存Data模块 阅读目录 jQuery API中Data的基本使用方法介绍 jQuery.acceptData(elem)源码分析 jQuery.data(el ...

  4. jQuery源码笔记(一):jQuery的整体结构

    jQuery 是一个非常优秀的 JS 库,与 Prototype,YUI,Mootools 等众多的 Js 类库相比,它剑走偏锋,从 web 开发的实用角度出发,抛除了其它 Lib 中一些中看但不实用 ...

  5. jQuery 2.0.3 源码分析 数据缓存

    历史背景: jQuery从1.2.3版本引入数据缓存系统,主要的原因就是早期的事件系统 Dean Edwards 的 ddEvent.js代码 带来的问题: 没有一个系统的缓存机制,它把事件的回调都放 ...

  6. jQuery源码笔记(二):定义了一些变量和函数 jQuery = function(){}

    笔记(二)也分为三部分: 一. 介绍: 注释说明:v2.0.3版本.Sizzle选择器.MIT软件许可注释中的#的信息索引.查询地址(英文版)匿名函数自执行:window参数及undefined参数意 ...

  7. JavaScipt 源码解析 数据缓存

    常见的内存泄露的几种情况: 循环引用 JavaScript闭包 DOM插入 一个DOM对象被一个JavaScript对象引用,同时又引用同一个或其他的JavaScript对象,这个DOM对象可能回引发 ...

  8. jQuery源码笔记——回调对象

    回调对象是一个多用途的回调列表对象,提供了强大的的方式来管理回调函数列表. 最简单的缓存对象 function Callbacks(){ var list = [], self = { add: fu ...

  9. jQuery源码笔记——延迟对象

    提供一种方法来执行一个或多个对象的回调函数, Deferred对象通常表示异步事件. 它是回调对象的拓展运用,在jQuery当中非常依赖回调对象. 一个简单的,只解决成功状态下的缓存实例 functi ...

随机推荐

  1. poj1220 (高精度任意进制转换)

    http://poj.org/problem?id=1220 高精度任意进制转换 代码是从discuss里找到的,据说是maigo神牛写的. 超精简!! 我自己第一写的时候,还把n进制先转成10进制, ...

  2. 【机房系统知识小结点系列】之遍历窗体中的控件,判断Text是否为空?

    做机房系统时,几乎每个窗体中都会用到判断界面中的控件是否为空的情景.我们曾经是这样走来的: 第一版: 好处:对窗体界面中的Text等控件,逐一做判断,当用户输入某一项为空的时候,会议弹出框的形式,告诉 ...

  3. javascript 【封装AJAX】

    post function createXHR() { if (typeof XMLHttpRequest != 'undefined') { return new XMLHttpRequest(); ...

  4. RDLC报表系列(二) 行分组

    接上一篇文章的内容,今天来说的是行分组.还是打开demo1.rdlc界面,拖入一个文本框和表 1.在表中随便选择一个字段,不然在添加行组的时候不会自动提示.我这里是选择的Dept 2.在下面的行组中右 ...

  5. APP安全测评checklist

    leader不要打我啊,我要借用一下我组app的安全测评检查方案,这些最基本的安全防范措施应该是每个app都要注意的吧: 对了,首先,你的app得先混淆啊~:AndroidStudio 混淆打包 先来 ...

  6. 在ASP.NET MVC自定义错误页面

    异常处理跳转页面 第一步,在项目的Web.config文件中找到节点<system.web> 在此节点下添加配置(Error为定义的控制器也可以多添加些error标签用于区分不同的错误) ...

  7. ZRender源码分析4:Painter(View层)-中

    回顾 上一篇说到:ZRender源码分析3:Painter(View层)-上,接上篇,开始Shape对象 总体理解 先回到上次的Painter的render方法 /** * 首次绘图,创建各种dom和 ...

  8. Android 监听网络变化

    Android 监听网络变化

  9. 【原】YUI Test自动化测试实例详解

    测试在软件开发中至关重要,目前针对不同的开发语言,都有比较成熟的测试框架,如jUnit,cUnit,cppUnit,nUnit等,我们统称为xUnit,他们的都遵守统一的规则: 针对代码测试 断言 启 ...

  10. 初学Django

    纵然有众多大牛写过这些简单入门文章,但作为记录,还是要自己动手写下来的比较靠谱,‘好脑筋不如烂笔头’啊! Python 安装 Django本身是纯Python编写的,所以安装框架的第一步是确保你已经安 ...