javascript中caller和callee詳解
最近學習javascript,碰到caller和callee的問題,去網上百度了很多。搜到的內容大同小益,整理總結了一下與大家分享。
caller:返回一個對調用function函數(shù)的函數(shù)的引用(用法:function.caller)
說明:對于函數(shù)來說,caller屬性只有在函數(shù)執(zhí)行時才有定義。如果函數(shù)由頂層調用,caller則為null。
var time = 3 //控制次數(shù),去掉會一直在caller與handleCaller交替不斷執(zhí)行 function caller() { caller.caller()//返回調用caller函數(shù)的函數(shù)引用 } function handleCaller() { if (time > 0){ time-- alert(handleCaller.caller)//返回調用handleCaller函數(shù)的函數(shù)引用 alert(caller.caller)//返回調用caller函數(shù)的函數(shù)引用 caller() } } handleCaller()
例子分析:第一次handleCaller運行的時候,兩個alert返回的都是null,alert(handleCaller.caller)返回null是因為它是由頂層調用, alert(caller.caller)返回null是因為caller的默認值是null。接下去caller()函數(shù)被調用,caller.caller返回的是調用它的函數(shù)(handleCaller)的引用,通過caller.caller()可以再次調用handleCaller函數(shù)。第二次handleCaller運行的時候,alert(handleCaller.caller)返回的是caller代碼(其實就是caller的引用),alert(caller.caller)返回的是handleCaller代碼。因為函數(shù)之間的調用關系是handleCaller->caller->handleCaller。之后就不斷在2個函數(shù)之間交替執(zhí)行。
caller指向調用當前函數(shù)的函數(shù),但是有一點,如果是在全局作用域內(即頂層window)被調用,則返回null。
代碼走起
==================== function testCaller(){ if(testCaller.caller == null){ console.log('accessed at global'); }else{ console.log('accessed at ' + testCaller.caller); } }
在全局調用
testCaller(); // accessed at global
在一個函數(shù)中調用
function a(){ testCaller(); } a(); // accessed at function a(){testCaller();}
此時,testCaller.caller指向就是 function a
callee:返回相對應的arguments的函數(shù)引用。(多用于匿名函數(shù)遞歸)
說明:也許你在網上看到最多的是callee返回正在執(zhí)行的函數(shù)引用。我是這么理解,每個函數(shù)都有一個自己的arguments,通常是用來存放參數(shù)的。arguments有一個callee 屬性,初始值就是對應自身的函數(shù)引用。當你函數(shù)執(zhí)行到該語句時,arguments是默認對應的是你現(xiàn)在執(zhí)行的函數(shù),那么arguments.callee為當前正在執(zhí)行的函數(shù)的引用。當然如果你有標記過其他函數(shù)的arguments(例子中的args),自然可以用args.callee()去再次調用那個函數(shù)。
function a(){ alert(arguments.callee) var args = arguments function c(){ alert(arguments.callee) args.callee() } c() } a()
例子分析:例子中的arguments.callee都是默認返回當前正在執(zhí)行的函數(shù)的引用(a中返回a自身函數(shù)引用,c中返回c自身函數(shù)引用),而通過用args存放a函數(shù)的arguments,在內置函數(shù)c中使用args.callee()再次調用a函數(shù)。
==================== function a(x){ if(x<=1) return x; else return x + a(x-1); } a(12) // 78
這是一個極簡的遞歸,運行結果正常。
再看看下面的調用方法
var b = a; a = null; // 將a回收 b(12); // erro : 'a' is not a function
原因也簡單,b=a,b=function a(){};在b調用之前,我們用了a=null。所以在 function a 運行的時候,其中的return x + a(x-1);中的a,指向的就是null,而不是 function a。
所以就報錯了,如何解決這樣的問題。我們將a換一種寫法
function a(x){ if(x<=1) return x; else return arguments.callee(x-1); // 這句是改變的地方 }
再調用
var b = a; a = null; b(12); // 78
原因:雖然我們將a=null了,但是函數(shù)a中并沒有用到a,而是通過arguments.callee指向當前函數(shù)。
因為arguments.callee的定義就是:返回正在執(zhí)行的函數(shù)。
- js的隱含參數(shù)(arguments,callee,caller)使用方法
- javascript中callee與caller的用法和應用場景
- js apply/call/caller/callee/bind使用方法與區(qū)別分析
- javascript下arguments,caller,callee,call,apply示例及理解
- javascript中callee與caller的區(qū)別分析
- js中arguments,caller,callee,apply的用法小結
- 理解Javascript的caller,callee,call,apply區(qū)別
- Javascript - 全面理解 caller,callee,call,apply
- js arguments,jcallee caller用法總結
- js中的caller和callee屬性介紹和例子
- JavaScript中callee和caller的區(qū)別與用法實例分析
相關文章
JavaScript實現(xiàn)隨機產生字符串的方法分享
這篇文章主要為大家詳細介紹了JavaScript中實現(xiàn)隨機產生字符串的方法,文中的示例代碼簡潔易懂,對我們學習JavaScript有一定的幫助,需要的可以參考一下2022-11-11使用Auto.js?調用系統(tǒng)短信、電話模塊實現(xiàn)功能
這篇文章主要介紹了如何使用Auto.js調用系統(tǒng)短信與電話模塊,并實現(xiàn)讀取短信與聯(lián)系人的功能,并給出了實現(xiàn)相應功能的代碼2023-03-03?JavaScript?數(shù)據(jù)結構之散列表的創(chuàng)建(2)
這篇文章主要介紹了?JavaScript?數(shù)據(jù)結構之散列表的創(chuàng)建,主要看如何處理散列值沖突的問題,并實現(xiàn)更完美的散列表。下文詳細介紹需要的小伙伴可以參考一下2022-04-04Webpack常見靜態(tài)資源處理-模塊加載器(Loaders)+ExtractTextPlugin插件
這篇文章主要介紹了Webpack常見靜態(tài)資源處理-模塊加載器(Loaders)+ExtractTextPlugin插件,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-06-06