天天躁日日躁狠狠躁AV麻豆-天天躁人人躁人人躁狂躁-天天澡夜夜澡人人澡-天天影视香色欲综合网-国产成人女人在线视频观看-国产成人女人视频在线观看

javascript下動態this與動態綁定實例代碼

那么函數就是被掰成兩部分儲存于對象,一是其函數名(鍵),一是函數體(值),那么函數中的this一般都指向函數所在的對象。但這是一般而已,在全局調用函數時,我們并沒有看到調用者,或者這時就是window。不過,函數聲明后,其實并沒有綁定到任何對象,因此我們可以用call apply這些方法設置調用者。

一個簡單的例子:
[script]
<script>
window.name = "window";
var run = function() {
alert("My name is " + this.name);
}
run();
</script>
[/html]
這里你不能說run是作為window的一個屬性而存在,但它的確是被window屬性調用了。實質上大多數暴露在最上層的東西都則window接管了。在它們需要調用時被拷貝到window這個對象上(不過在IE中window并不繼承對象),于是有了window['xxx']與window.xxx性能上的差異。這是內部實現,不深究了。
另一個例子,綁定到一個明確的對象上

[Ctrl+A 全選 注:如需引入外部Js需刷新才能執行]
答案顯然易見,this總是為它的調用者。但如果復雜一點呢?

[Ctrl+A 全選 注:如需引入外部Js需刷新才能執行]
盡管它是定義在object內部,盡管它是定義run函數內部,但它彈出的既不是object也不是run,因為它既不是object的屬性也不是run的屬性。它松散在存在于run的作用域用,不能被前兩者調用,就只有被window拯救。window等原生對象浸透于在所有腳本的內部,無孔不入,只要哪里需要到它做貢獻的地方,它都義不容辭。但通常我們不需要它來幫倒忙,這就需要奠出call與apply兩大利器了。

[Ctrl+A 全選 注:如需引入外部Js需刷新才能執行]
call與apply的區別在于第一個參數以后的參數的形式,call是一個個,aplly則都放到一個數組上,在參數不明確的情況,我們可以借助arguments與Array.slice輕松搞定。

[Ctrl+A 全選 注:如需引入外部Js需刷新才能執行]
由此Prototype開發者搞了一個非常有名的函數出來,bind!以下是它的一個最簡單的版本:
復制代碼 代碼如下:
var bind = function(context, fn) {
return function() {
return fn.apply(context, arguments);
}
}


[Ctrl+A 全選 注:如需引入外部Js需刷新才能執行]
不過為了面對更復雜的情況建議用以下版本。
復制代碼 代碼如下:
function bind(context,fn) {
var args = Array.prototype.slice.call(arguments, 2);
return args.length == 0 ? function() {
return fn.apply(context, arguments);
} : function() {
return fn.apply(context, args.concat.apply(args, arguments));
};
};

它還有一個孿生兄弟叫bindAsEventListener ,綁定事件對象,沒什么好說的。
復制代碼 代碼如下:
var bindAsEventListener = function(context, fn) {
return function(e) {
return fn.call(context, (e|| window.event));
}
}

Prototype的版本
復制代碼 代碼如下:
Function.prototype.bind = function() {
if (arguments.length < 2 && (typeof arguments[0]==='undefined'))
return this;
var _slice = Array.prototype.slice
var __method = this, args = _slice.call(arguments,0), context = args.shift();
return function() {
return __method.apply(context, args.concat(_slice.call(arguments,0)));
}
}


[Ctrl+A 全選 注:如需引入外部Js需刷新才能執行]
bind函數是如此有用,google早早已把它加入到Function的原型中了(此外還有inherits,mixin與partial)。
復制代碼 代碼如下:
//在chrome中
var a = function(){};
alert(a.bind)

有綁定就有反綁定,或者叫剝離更好!例如原生對象的泛化方法我們是無法通過遍歷取出它們的。

[Ctrl+A 全選 注:如需引入外部Js需刷新才能執行]
要取出它們就需要這個東西:
復制代碼 代碼如下:
var _slice = Array.prototype.slice;
function unbind(fn) {//第一步取得泛化方法
return function(context) {//第二部用對應原生對象去重新調用!
return fn.apply(context, _slice.call(arguments, 1));
};
};

示例以前也給過了,請見文章
總結:
this 的值取決于 function 被調用的方式,一共有四種,
如果一個 function 是一個對象的屬性,該 funtion 被調用的時候,this 的值是這個對象。如果 function 調用的表達式包含句點(.)或是 [],this 的值是句點(.)或是 [] 之前的對象。如myObj.func 和myObj["func"] 中,func 被調用時的 this 是myObj。
如果一個 function 不是作為一個對象的屬性,那么該 function 被調用的時候,this 的值是全局對象。當一個 function 中包含內部 function 的時候,如果不理解 this 的正確含義,很容易造成錯誤。這是由于內部 function 的 this 值與它外部的 function 的 this 值是不一樣的。解決辦法是將外部 function 的 this 值保存在一個變量中,在內部 function 中使用它來查找變量。
如果在一個 function 之前使用 new 的話,會創建一個新的對象,該 funtion 也會被調用,而 this 的值是新創建的那個對象。如function User(name) {this.name = name}; var user1 = new User("Alex"); 中,通過調用new User("Alex") ,會創建一個新的對象,以user1 來引用,User 這個 function 也會被調用,會在user1 這個對象中設置名為name 的屬性,其值是Alex 。
可以通過 function 的 apply 和 call 方法來指定它被調用的時候的 this 的值。 apply 和 call 的第一個參數都是要指定的 this 的值。由于它們存在,我們得以創建各種有用的函數。

JavaScript技術javascript下動態this與動態綁定實例代碼,轉載需保留來源!

鄭重聲明:本文版權歸原作者所有,轉載文章僅為傳播更多信息之目的,如作者信息標記有誤,請第一時間聯系我們修改或刪除,多謝。

主站蜘蛛池模板: 国产99久久亚洲综合精品西瓜tv | 调教女M屁股撅虐调教 | 夜夜穞狠狠穞 | av天堂网2014在线 | 野花韩国视频中文播放 | 免费看午夜高清性色生活片 | 欧美 亚洲 日韩 在线综合 | 日操夜操天天操 | 看全色黄大色大片免费久黄久 | 久久精品黄色 | 久久精品国产欧美成人 | 国内国外精品影片无人区 | 伊人在线高清视频 | 插我一区二区在线观看 | 果冻传媒在线观看进入窗口 | 亚洲午夜久久久久久久久电影网 | 亚洲 日本 欧美 中文字幕 | 推倒美女总裁啪啪 | 丝袜足控免费网站xx91 | 久久人人玩人妻潮喷内射人人 | 亚洲精品久久一区二区三区四区 | 国产精品一区二区资源 | 手机在线亚洲日韩国产 | 亚洲精品国产精麻豆久久99 | 狠狠色狠狠色狠狠五月ady | A国产一区二区免费入口 | 武侠艳妇屈辱的张开双腿 | 国产精品成久久久久三级四虎 | 亚洲欧美一区二区三区导航 | 黑吊大战白女出浆 | 天天躁日日躁狠狠躁午夜剧场 | 最近中文字幕完整版高清 | 日本欧美午夜三级 | 欧美精品亚洲精品日韩专区一 | 女人张开腿让男人桶爽免 | 千禧金瓶梅 快播 | 久久青青无码AV亚洲黑人 | 丝瓜视频在线免费 | 色欲人妻AAAAAAA无码 | 国产亚洲精品欧洲在线视频 | 色欲色香天天天综合 |