我们经常使用隐藏控件或者是js全局变量来临时存储数据,全局变量容易导致命名污染,隐藏控件导致经常读写dom浪费性能。jQuery提供了自己的数据缓存方案,能够达到和隐藏控件、全局变量相同的效果,但是jQuery实现方式更优雅。为了更好地使用jQuery数据缓存方案,我们需要掌握$.data()、$.cache、$.expando、$.hasData()、$.removeData()。

$.hasData()用来判断某个对象是否有附加的属性,可以给任何JavaScript对象和HTMLElement对象附加属性。$.data()用来读取或者修改属性值。$.removeData()用来删除已经添加的属性,这是为了释放内存,避免过多无用属性浪费内存。

  1. var myObj = {};
  2. // hasData用来判断HTMLElement或JS对象是否具有数据
  3. console.log(jQuery.hasData($("#a")));// false
  4. // data()添加属性
  5. $.data(myObj, 'name', 'aty');
  6. console.log(jQuery.hasData(myObj));// true
  7. // data()读取属性
  8. console.log($.data(myObj, 'name'));//aty
  9. // removeData删除属性
  10. $.removeData(myObj, 'name');
  11. console.log($.data(myObj, 'name'));//undefined
  12. // 如果所有属性都被删除,那么hasData返回false
  13. console.log(jQuery.hasData(myObj));// false

给HTMLElement对象添加属性,使用方式跟为普通js对象添加属性一模一样。

  1. <div id="content"></div>
  2. <script>
  3. var el = document.getElementById('content');
  4. $.data(el, 'name', 'aty');
  5. console.log($.data(el, 'name')); // aty
  6. </script>

可以看到使用jQuery数据缓存的API是很容易的,现在我们要大致看下jQuery是如何实现缓存方案。为普通JS对象提供缓存时,jquery直接将数据保存在原始的JS对象上。此时会偷偷的给JS对象添加个属性(类似于jQuery16101803968874529044),属性值也是一个对象。

  1. var myObj = {};
  2. $.data(myObj, 'name', 'aty');
  3. console.log(myObj);
  4. console.log($.expando);

我们可以看到myObj结构发生了变化:jquery给普通对象偷偷添加的属性名称其实就是$.expando。

jQuery.expando是一个字符串,使用Math.random生成,去掉了非数字字符。它作为HTMLElement或JS对象的属性名。页面引入jQuery框架的时候,会随机生成一个字符串。

  1. expando: "jQuery" + ( jQuery.fn.jquery + Math.random() ).replace( /\D/g, "" ),

现在我们知道了jQuery如何给普通对象添加属性,以及expando的含义,那么我们就可以通过下面的代码获取添加的属性。

  1. var hisObj = {};
  2. $.data(hisObj, 'name', 'him');
  3. console.log(hisObj[jQuery.expando].data.name);//him

为HTMLElement提供缓存时,却不会直接保存在HTMLElement上。而是保存在jQuery.cache这个全局对象上。此时先给HTMLElement添加属性(jQuery.expando),属性值为数字(1,2,3递增)。即只将一些数字保存在了HTMLElement上,不会直接将数据置入。这是因为IE老版本中可能会存在内存泄露危险。而HTMLElement如何与jQuery.cache建立联系呢? 还是id。刚刚提到属性值数字就是id。

  1. <div id="a"></div>
  2. <script>
  3. var dom = document.getElementById("a");
  4. $.data(dom, 'name', 'aty');
  5. console.log(dom[jQuery.expando]); // 1
  6. console.log(jQuery.cache); // {1 : {data:{name:'aty'}}}
  7. </script>


知道了jQuery如何为dom对象添加属性,我们就可以通过下面的代码获取属性。

  1. console.log(jQuery.cache[dom[jQuery.expando]].data.name);

现在我们看下DOM对象和jQuery封装后的对象有什么区别。

  1. // 给dom元素添加数据
  2. var dom = document.getElementById("a");
  3. $.data(dom, 'name', 'aty');
  4. console.log(jQuery.hasData(dom));//true
  5. console.log($.data(document.getElementById("a"), 'name'));//aty
  6. console.log($.data($("#a")[0], 'name'));//aty
  7. // 给jquery包装后的dom附加对象
  8. var $dom = $("#a");
  9. $.data($dom, 'age', '25');
  10. console.log(jQuery.hasData($dom));//true
  11. console.log($.data($dom, 'age'));//25
  12. console.log($.data($("#a"), 'age'));//undefined


这是因为本质区别在于:原始的DOM对象会被jQuery特殊对待,而jQuery包装后的对象与普通JS对象无异。通过jQuery选择器每次获取的对象并不是同一个对象

  1. var $dom = $("#a");
  2. $.data($dom, 'age', '25');
  3. // dom对象支持多次获取,jQuery对象不支持
  4. console.log( document.getElementById("a") ===  document.getElementById("a"));///true
  5. console.log( document.getElementById("a") ===  $("#a")[0]);///true
  6. console.log($("#a") ===  $("#a"));//false
  7. // jQuery包装后的对象,与普通javascript对象无异
  8. console.log($dom[jQuery.expando].data.age);//25

最后再提一下,可以使用$.data获取某个对象上附加的所有属性。

    1. var $dom = $("#a");
    2. $.data($dom, 'age', '25');
    3. console.log($.data($dom));//Object {age: "25"}

参考文章:

读jQuery之六(缓存数据)

http://www.cnblogs.com/weihengblogs/p/6093067.html

jQuery数据缓存方案详解:$.data()的使用的更多相关文章

  1. jquery源码解析:jQuery数据缓存机制详解2

    上一课主要讲了jQuery中的缓存机制Data构造方法的源码解析,这一课主要讲jQuery是如何利用Data对象实现有关缓存机制的静态方法和实例方法的.我们接下来,来看这几个静态方法和实例方法的源码解 ...

  2. jquery源码解析:jQuery数据缓存机制详解1

    jQuery中有三种添加数据的方法,$().attr(),$().prop(),$().data().但是前面两种是用来在元素上添加属性值的,只适合少量的数据,比如:title,class,name等 ...

  3. java——包装类数据缓存 ==号详解

    Java对部分经常使用的数据采用缓存技术,即第一次使用该数据则创建该数据对象并对其进行缓存, 当再次使用等值对象时直接从缓存中获取,从而提高了程序执行性能.(只对常用数据进行缓存) Java中只是对部 ...

  4. Oracle DataGuard数据备份方案详解

    Oracle DataGuard是一种数据库级别的HA方案,最主要功能是冗灾.数据保护.故障恢复等. 在生产数据库的"事务一致性"时,使用生产库的物理全备份(或物理COPY)创建备 ...

  5. HTML5数据存储方案data与jQuery数据存储方案$.data()的区别

    我们先看下$.fn.data()的使用,这个和$.data()是不一样的,前者是和某个jquery对象相关,后者则是全局方法.主要有data()和removeData()这2个实例方法.通过下面的例子 ...

  6. jQuery数据缓存data(name, value)详解及实现

    一. jQuery数据缓存的作用 jQuery数据缓存的作用在中文API中是这样描述的:“用于在一个元素上存取数据而避免了循环引用的风险”.如何理解这句话呢,看看我下面的举例,不知道合不合适,如果你有 ...

  7. jQuery源代码学习之六——jQuery数据缓存Data

    一.jQuery数据缓存基本原理 jQuery数据缓存就两个全局Data对象,data_user以及data_priv; 这两个对象分别用于缓存用户自定义数据和内部数据: 以data_user为例,所 ...

  8. jQuery数据缓存$.data 的使用以及源码解析

    一.实现原理: 对于DOM元素,通过分配一个唯一的关联id把DOM元素和该DOM元素的数据缓存对象关联起来,关联id被附加到以jQuery.expando的值命名的属性上,数据存储在全局缓存对象jQu ...

  9. 触碰jQuery:AJAX异步详解

    触碰jQuery:AJAX异步详解 传送门:异步编程系列目录…… 示例源码:触碰jQuery:AJAX异步详解.rar AJAX 全称 Asynchronous JavaScript and XML( ...

随机推荐

  1. 《jQuery判断radio、checkbox、select 是否选中和设置选中问题以及获取选中值》总结

    <form> <input type="radio" name="gender" id="man" value=" ...

  2. css负边距之详解

    自从1998年CSS2作为推荐以来,表格的使用渐渐退去,成为历史.正因为此,从那以后CSS布局成为了优雅代码的代名词. 对于所有设计师使用过的CSS概念,负边距作为最少讨论到的定位方式要记上一功.这就 ...

  3. css中margin的应用

    1.margin用于设置外边距,没有继承性,父元素设置的margin属性子元素不会继承. 2.margin存在重叠的问题. 水平边距永远不会发生重叠. 垂直边距在特定情况下会重叠. 重叠问题都很容易解 ...

  4. MVC开发模式下的用户角色权限控制

    前提: MVC开发模式 大概思想: 1.在MVC开发模式下,每个功能都对应着不同的控制器或操作方法名(如修改密码功能可能对应着User/changepd),把每个功能对应的控制器名和操作方法名存到数据 ...

  5. mysql使用load导入txt文件所遇到的问题及解决方法

    导入txt文件,有导入向导这种方式: 另外可以使用load的方式导入.最开始使用以下代码插入: load data local infile 'F:\\Data\\predict_data.txt' ...

  6. IIS网站或系统验证码不显示问题——"使用了托管的处理程序,但是未安装或未完整安装 ASP.NET"

    在IIS上发布了一个系统,但是登陆页面的验证码图片一直出不来,尝试了各种办法,权限.路径.继承父类路径等都不管用,进入Login.html,对着无验证码图片的图标,右键复制图片的网址,粘贴到地址栏,出 ...

  7. $_SERVER 详情

    $_SERVER['HTTP_ACCEPT_LANGUAGE']//浏览器语言 $_SERVER['REMOTE_ADDR'] //当前用户 IP . $_SERVER['REMOTE_HOST'] ...

  8. WebService -- Java 实现之 CXF ( 使用Spring添加拦截器)

    最重要的就是在ApplicationContext.xml下面添加配置 <!-- service provider --> <jaxws:endpoint implementor=& ...

  9. 《转载》跟我学SpringMVC

    在线版目录 第一章 Web MVC简介 第二章 Spring MVC入门 第三章 DispatcherServlet详解 第四章 Controller接口控制器详解(1) 第四章 Controller ...

  10. 关于easyui datagrid 表格数据处理

    首先先将easyui 引入到jsp页面中 <link rel="stylesheet" type="text/css" href="easyui ...