jQuery静态方法globalEval使用和源码分析

Eval函数大家都很熟悉,但是globalEval方法却很少使用,大多数参考手册也没有相关api,下面就对其用法和源码相应介绍:

jQuery.globalEval()函数用于全局性地执行一段JavaScript代码。

示例:

var name = "全局变量";

function test(){
    var name = "局部变量";

    alert(name);                     // 局部变量

    eval( "alert(name);" );          // 局部变量

    $.globalEval( "alert(name);" );  // 全局变量
}

test();

可以看到该方法跟eval方法相比有一个作用域的范围差异即始终处于全局作用域下面,下面进行源码分析:

// Evaluates a script in a global context
// Workarounds based on findings by Jim Driscoll
// http://weblogs.java.net/blog/driscoll/archive/2009/09/08/eval-javascript-global-context
globalEval: function( data ) {
    if ( data && rnotwhite.test( data ) ) {
         // We use execScript on Internet Explorer
         // We use an anonymous function so that context is window
         // rather than jQuery in Firefox
            ( window.execScript || function( data ) {
                window[ "eval" ].call( window, data );
            } )( data );
     }
 },

注释里的意思是提到了此方法的实现是在Jim Driscoll的基础之上的而且把相关的文章链接还附了上面,索性链接还可以打开瞅瞅,里面大致介绍了怎么让js代码在全局执行的方法,对于ie浏览器而言则提供了一个不太常用的方法(反正我是第一次听到)---window.execScript

window.execScript 方法会根据提供的脚本语言执行一段脚本代码。window.execScript 方法有两个参数,第一个参数指定被执行的脚本代码段,第二个参数指定脚本代码语言类别 (缺省值为 JScript),execScript 方法执行后的代码位于全局作用域下。举例:

 var str=‘全局‘;
    (function(){
        var str=‘局部‘;
        window.execScript(‘alert(str)‘);  //ie 全局
}());  

但是此方法现在只有ie支持,早期的chrome版本也支持现在已经放弃,具体的此方法介绍请参考http://ued.sina.com/?p=789;

打开注释中的网址是一个英文网站四级英语水品的我毫不犹豫的使用了谷歌网页翻译功能,可以谷歌不给力告诉我无法翻译,那没办法只能硬着头皮读下去,怕译错把原文附上请以原文为准

For more standards-respecting browsers, the way to do this should be to use the call function, which is a standard function attached to every Function object. So, eval.call(window, src) should work. But to understand why, it‘s important to know about context, in addition to scope. Every function call has it‘s own context: this is the object that‘s represented by the special value this. When we use the call function, the first parameter is the context object we‘ll use for this. This is handy for all kinds of purposes, but for us, it‘s just nice to use to set the context to the window object - which, you‘ll recall, is the global.

这段是提供解决方法的,对于标准浏览器而言可以采用eval函数解析js代码字符串,然后通过对象冒充的方式把作用域指向window,下面就是自己的测试:

var str=‘全局‘;
(function(){
     var str=‘局部‘;
     eval.call(window, "alert(str)");
}());  

经过测试ie9及以上和非ie均弹出“全局“ ,ie9以下弹出局部,说明这个方法有见兼容性有问题啊,但是这个方法本来就是给非IE用的,你ie爱咋咋地吧

Sadly, eval.call(window,src) breaks on Chrome - it complains about contexts not matching. Odd - and I was unable to Google up why this might be so. But a couple lucky guesses later, and I discovered that window.eval.call(window,src) works on all non-IE browsers. Now, when I say "var j = 1", the window[j] is the variable that‘s set... So, that‘s good. Why do we have to add the extra window. on Chrome? Not sure - I could guess, but it‘s too likely to be wrong.

该文作者用了多次用了”odd“,我也觉得很”odd",作者说在谷歌中报错了,然后把代码修改为window.eval.call(window,src)就可以啦,正如作者所困惑的,全局方法是不需要加window直接调用的这个大家都清楚,但是我做检测的时候是没有出现问题的,应该是年代久远了谷歌已经解决了,最后作者还提到火狐在执行解析this存在问题

At this point, I thought we‘d licked the problem. No such luck. Sure, global variables are getting set, but it turns out that if you say: alert(this) - then you would correctly receive the global object back on Chrome and Safari, but not Firefox - there, you‘d get back the object that was the enclosing object before the call function got called. Very odd, and likely a bug in their implementation.

既然全局解析alert(this)肯定是window才对,而火狐弹出最近的作用链的对象,对于需要测试下,代码如下:

 var str=‘全局‘;
      (function(){
           var str=‘局部‘;
           window.eval.call(window, "alert(this)");
  }());  

经检测火狐没有问题,估计火狐也是早把这个问题处理了,在文章的最后作者留了一个方法

var globalEval = function globalEval(src) {
    if (window.execScript) {
        window.execScript(src);
        return;
    }
    var fn = function() {
        window.eval.call(window,src);
    };
    fn();
};

可以说这个方法已经够用了,但是回过头来看看我们的jQuery源码会发现精简了很多

data && rnotwhite.test( data )

该条件保证有数据而且是不带空格的数据,当然你也可以不传字符串但是没什么意义rnotwhite的定义在构造函数中

// Check if a string has a non-whitespace character in it
    rnotwhite = /\S/,

至于为什么要检测不能有空格还是这篇外文中有其他开发者指出了如果是空白的在ie8出现错误

by vesperaba - 2012-10-18 08:54
If src is a blank string you will get an error in IE8. To avoid that I added the following check at the beginning:

if (src == undefined || src == ‘‘) {
     return;
}

//The whole function will be:
var globalEval = function globalEval(src) {
if (src == undefined || src == ‘‘) {
     return;
}
if (window.execScript) {
    window.execScript(src);
       return;
   }
   var fn = function() {
       window.eval.call(window,src);
   };
   fn();
};

通过逻辑或运算来选择何种方式解析,如果是采用eval则是处在一个自执行的匿名函数中,结构很精简

( window.execScript || function( data ) {
      window[ "eval" ].call( window, data );
 } )( data );

另需注意的事该方法没有返回值,或者可以理解为返回undefined

时间: 2024-10-05 19:01:41

jQuery静态方法globalEval使用和源码分析的相关文章

jQuery静态方法parseXML使用和源码分析

jQuery.parseXML( data ) 接受一个格式良好的 XML 字符串,返回解析后的 XML 文档. 方法 jQuery.parseXML() 使用浏览器原生的 XML 解析函数实现. 在 IE 9+ 和其他浏览器中,会使用 DOMParser 对象解析: 在 IE 9 以下的浏览器中,则使用 ActiveXObject 对象解析 源码分析 // Cross-browser xml parsing parseXML: function( data ) { var xml, tmp;

jQuery静态方法type使用和源码分析

jQuery.type方法是检测数据类型的工具方法,在分析其用法之前先总结下js给我们提供了那些监测数据类型的方法: 一.typeof 操作符 下面是测试代码 var data=[],a='123',b=0,c=true,d={1:23},e=[123],f=function(){},g=null,h=undefined,i=Math,j=/$.+^/,k= new Date(); data.push(a,b,c,d,e,f,g,h,i,j,k); for(var key=0;key<data.

jQuery静态方法isPlainObject,isEmptyObject方法使用和源码分析

isPlainObject方法 测试对象是否是纯粹的对象(通过 "{}" 或者 "new Object" 创建的) 示例: //测试是否为纯粹的对象 jQuery 代码: jQuery.isPlainObject({}) // true jQuery.isPlainObject("test") // false 源码分析: isPlainObject: function( obj ) { // Must be an Object. // Beca

转载Aaron ---- jQuery 2.0.3 源码分析core - 选择器

jQuery 2.0.3 源码分析core - 选择器(02) 声明:本文为原创文章,如需转载,请注明来源并保留原文链接Aaron,谢谢! 打开jQuery源码,一眼看去到处都充斥着正则表达式,jQuery框架的基础就是查询了,查询文档元素对象,所以狭隘的说呢,jQuery就是一个选择器,并这个基础上构建和运行查询过滤器! 工欲善其事,必先利其器,所以先从正则入手 我们来分解一个表达式 // A simple way to check for HTML strings // Prioritize

jQuery.lazyload使用及源码分析

前言: 貌似以前自己也写过图片懒加载插件,但是新公司使用的是jQuery.lazyload插件,为了更好的运用,自己还是把源码看了遍,分别记录了如何使用, 插件原理,各个配置属性的完整解释,demo实例,源码分析(较简短),源码分析可以配合使用,配置属性,原理进行阅读,如需转载,请注明出处 博客园 华子yjh 一.如何使用 // 最简单的使用,不带参数 $('img').lazyload(); // 带参数(配置对象),下面配置对象中的各个属性值都是默认的 $('img').lazyload({

Android Debuggerd 简要介绍和源码分析(转载)

转载: http://dylangao.com/2014/05/16/android-debuggerd-%E7%AE%80%E8%A6%81%E4%BB%8B%E7%BB%8D%E5%92%8C%E6%BA%90%E7%A0%81%E5%88%86%E6%9E%90/ 码字很辛苦,转载请注明来自Dylan‘s Zone的<Android Debuggerd 简要介绍和源码分析> 本文以android4.1为基础,分析debuggerd这个工具的使用方法和源码. 1.Debuggerd 简介

java集合框架10——TreeMap和源码分析(一)

前面讨论完了HashMap和HashTable的源码,这一节我们来讨论一下TreeMap.先从整体上把握TreeMap,然后分析其源码,深入剖析TreeMap的实现. 1. TreeMap简介 TreeMap是一个有序的key-value集合,它内部是通过红-黑树实现的,如果对红-黑树不太了解,请先参考下这篇博文:红-黑树.下面我们先来看看TreeMap的继承关系: java.lang.Object ? java.util.AbstractMap<K, V> ? java.util.TreeM

转载Aaron博客 ---- jQuery 2.0.3 源码分析core - 整体架构

jQuery 2.0.3 源码分析core - 整体架构 整体架构 拜读一个开源框架,最想学到的就是设计的思想和实现的技巧. 废话不多说,jquery这么多年了分析都写烂了,老早以前就拜读过, 不过这几年都是做移动端,一直御用zepto, 最近抽出点时间把jquery又给扫一遍 我也不会照本宣科的翻译源码,结合自己的实际经验一起拜读吧! github上最新是jquery-master,加入了AMD规范了,我就以官方最新2.0.3为准 整体架构 jQuery框架的核心就是从HTML文档中匹配元素并

转:《Linux设备驱动程序3》源码目录结构和源码分析经典链接

转自:http://blog.csdn.net/geng823/article/details/37567557 [原创][专栏]<Linux设备驱动程序>--- LDD3源码目录结构和源码分析经典链接 [专栏]Linux设备驱动程序学习(总目录) [专栏]LDD3源码分析链接(总目录) 1. LDD3源码分析之hello.c与Makefile模板 2. LDD3源码分析之字符设备驱动程序 其他错误: 我的Linux内核为 3.2.0-65-generic-pae,在scull目录下make时