jQuery版本冲突
jQuery多个版本或和其他js库冲突主要是常用的$符号的冲突。
一、冲突的解决 1、同一页面jQuery多个版本冲突解决方法(function($){ ); })(jq164); jq142(function($){ ); });
2、jQuery库在其他库之后导入jQuery noConflict() 方法会释放会 $ 标识符的控制,这样其他脚本就可以使用它了。
1、可以通过jQuery全名替代简写的方式来使用 jQuery在其他库和jQuery库都加载完毕后,可以在任何时候调用jQuery.noConflict()函数来将变量$的控制权移交给其他JavaSript库。然后就可以在程序里将jQuery()函数作为jQuery对象的制造工厂。
test---prototypetest---jQuery jQuery.noConflict(); (){ alert( jQuery(this).text() ); }); }); //此处不可以再写成$,此时的$代表prototype.js中定义的$符号。 $(
2、自定义一个快捷方式noConflict() 可返回对 jQuery 的引用,可以把它存入自定义名称,例如jq,$J变量,以供稍后使用。
这样可以确保jQuery不会与其他库冲突,同时又使用自定义一个快捷方式。
(){ alert( $j(this).text() ); }); }); $(
3、在不冲突的情况下仍然用$如果想在jQuery 代码块使用 $ 简写,不愿意改变这个快捷方式,可以把 $ 符号作为变量传递给 ready 方法。这样就可以在函数内使用 $ 符号了 , 而在函数外,依旧不得不使用 "jQuery"。
<script type="text/javascript"> jQuery.noConflict(); //将变量$的控制权让渡给prototype.js jQuery(document).ready(function($){ $("p").click(function(){ //继续使用 $ 方法 alert( $(this).text() ); }); }); //或者如下 jQuery(function($){ //使用jQuery $("p").click(function(){ //继续使用 $ 方法 alert( $(this).text() ); }); }); </script>
或者使用IEF语句块,这应该是最理想的方式,因为可以通过改变最少的代码来实现全面的兼容性。
在我们自己写jquery插件时,应该都使用这种写法,因为我们不知道具体工作过程中是如何顺序引入各种js库的,而这种语句块的写法却能屏蔽冲突。
<script type="text/javascript"> jQuery.noConflict(); //将变量$的控制权让渡给prototype.js (function($){ //定义匿名函数并设置形参为$ $(function(){ //匿名函数内部的$均为jQuery $("p").click(function(){ //继续使用 $ 方法 alert($(this).text()); }); }); })(jQuery); //执行匿名函数且传递实参jQuery $("pp").style.display = 'none'; //使用prototype </script>
3、jQuery库在其他库之前导入jQuery库在其他库之前导入,$的归属权默认归后面的JavaScript库所有。那么可以直接使用"jQuery"来做一些jQuery的工作。
同时,可以使用$()方法作为其他库的快捷方式。这里无须调用jQuery.noConflict()函数。
Test-prototype(将被隐藏)Test-jQuery(将被绑定单击事件) jQuery((){ alert( jQuery(this).text() ); }); }); $(
二、原理 1、源码源码:看一下源码里怎么做到的
_jQuery = window.jQuery, // Map over the $ in case of overwrite _$ = window.$, jQuery.extend({ noConflict: function( deep ) { if ( window.$ === jQuery ) { window.$ = _$; } if ( deep && window.jQuery === jQuery ) { window.jQuery = _jQuery; } return jQuery; } });
在jQuery加载的时候,通过事先声明的_jQuery变量获取到当前window.jQuery,通过_$获取到当前window.$。
通过jQuery.extend()把noConflict挂载到jQuery下面。所以我们在调用的时候都是jQuery.noConflict()这样调。
在调用noConflict()时做了2个判断,
第一个if,把$的控制权交出去。
第二个if,在noConflict()传参的时候把,jQuery的控制权交出去。
最后noConflict()返回jQuery对象,用哪个参数接收,哪个参数将拥有jQuery的控制权。
2、 验证$ = 123; //假设其他库中$为123 $( function () { console.log($); //报错Uncaught TypeError: $ is not a function } );
解决冲突
jq = $.noConflict(); var $ = 123; jq(function () { alert($); //123 });
释放$控制权例子
<script> var $ = 123; // window.$是123,存储在私有的_$上。 </script> <script src="https://code.jquery.com/jquery-2.2.4.js"></script> <body> <div>aaa</div> <script> var jq = $.noConflict();//当window.$===jQuery的时候,把_$赋给了window.$。 jq(function () { alert($); //123 }); </script>
释放jQuery控制权例子
参数deep的作用:deep用来放弃jQuery对外的接口。
如下,noConflict()不写参数,弹出jQuery为构造函数。