• <fieldset id="8imwq"><menu id="8imwq"></menu></fieldset>
  • <bdo id="8imwq"><input id="8imwq"></input></bdo>
    最新文章專題視頻專題問答1問答10問答100問答1000問答2000關鍵字專題1關鍵字專題50關鍵字專題500關鍵字專題1500TAG最新視頻文章推薦1 推薦3 推薦5 推薦7 推薦9 推薦11 推薦13 推薦15 推薦17 推薦19 推薦21 推薦23 推薦25 推薦27 推薦29 推薦31 推薦33 推薦35 推薦37視頻文章20視頻文章30視頻文章40視頻文章50視頻文章60 視頻文章70視頻文章80視頻文章90視頻文章100視頻文章120視頻文章140 視頻2關鍵字專題關鍵字專題tag2tag3文章專題文章專題2文章索引1文章索引2文章索引3文章索引4文章索引5123456789101112131415文章專題3
    問答文章1 問答文章501 問答文章1001 問答文章1501 問答文章2001 問答文章2501 問答文章3001 問答文章3501 問答文章4001 問答文章4501 問答文章5001 問答文章5501 問答文章6001 問答文章6501 問答文章7001 問答文章7501 問答文章8001 問答文章8501 問答文章9001 問答文章9501
    當前位置: 首頁 - 科技 - 知識百科 - 正文

    JavaScript打破作用域的牢籠

    來源:懂視網 責編:小采 時間:2020-11-27 20:24:45
    文檔

    JavaScript打破作用域的牢籠

    JavaScript打破作用域的牢籠:JavaScript打破作用域的牢籠JavaScript作為一種松散型語言,有著很多令人瞠目結舌的特性(往往是一些令人捉摸不透的奇怪特性),本文我們將介紹如何使用JavaScript的一些特性來打破常規編程語言作用域的牢籠。1.JavaScript聲明提升很多人應該知道,
    推薦度:
    導讀JavaScript打破作用域的牢籠:JavaScript打破作用域的牢籠JavaScript作為一種松散型語言,有著很多令人瞠目結舌的特性(往往是一些令人捉摸不透的奇怪特性),本文我們將介紹如何使用JavaScript的一些特性來打破常規編程語言作用域的牢籠。1.JavaScript聲明提升很多人應該知道,

    JavaScript打破作用域的牢籠

    JavaScript作為一種松散型語言,有著很多令人瞠目結舌的特性(往往是一些令人捉摸不透的奇怪特性),本文我們將介紹如何使用JavaScript的一些特性來打破常規編程語言“作用域的牢籠”。

    1.JavaScript聲明提升

    很多人應該知道,js有變量聲明提升、函數聲明提升的特性。不管你之前是否了解,看下面的代碼運行的結果是否符合你的預期:

    var a=123;
    //可以運行
    abc();
    //報錯:def is not a function
    def();
    function abc(){
     //undefined
     console.log(a);
     var a="hello";
     //hello
     console.log(a);
    }
    var def=function(){
     console.log("def");
    }

    實際上js在運行時會對代碼進行兩輪掃描。第一輪,初始化變量;第二輪,執行代碼。第二輪執行代碼很好理解,不過第一輪過程比較模糊。具體來說,第一輪會做下面三件事:

    (1)聲明并初始化函數參數

    (2)聲明局部變量,包括將匿名函數賦給一個局部變量,但并不初始化他們

    (3)聲明并初始化函數

    明白了這些理論基礎之后,上面那段代碼在第一輪掃描之后實際上被js編譯器“翻譯”成了如下代碼:

    var a;
    a=123;
    function abc(){
     //局部變量,將會取代外部的a
     var a;
     //undefined
     console.log(a);
     var a="hello";
     //hello
     console.log(a);
    }
    var def;
    //可以運行
    abc();
    //報錯:def is not a function
    def();
    var def=function(){
     console.log("def");
    }

    現在再來看注釋里展示的程序運行時輸出,是不是覺得順理成章了。這就是js聲明提升在當中起到的作用。

    知道了js聲明提升的作用機制之后,我們來看下面這段代碼:

    var obj={};
    function start(){
     //undefined
     //This is obj.a
     console.log(obj.a);
     //undefined
     //This is a
     console.log(a);
     //成功
    輸出 //成功輸出 console.log("頁面執行完成"); } start(); var a="This is a"; obj.a="This is obj.a"; start();

    上述注釋第一行表示第一次執行start()方法時的輸出,第二行表示第二次執行start()方法的輸出。可以看到,由于js聲明提升的存在,兩次執行start()方法都沒有報錯。下面來看對這個例子進行小小的修改:

    var obj={};
    function start(){
     //undefined
     //This is obj.a
     console.log(obj.a);
     //報錯
     //This is a
     console.log(a);
     //因為上一行的報錯導致后續代碼不執行
     //成功
    輸出 console.log("頁面執行完成"); } start(); /*---------------另一個js文件----------------*/ var a="This is a"; obj.a="This is obj.a"; start();

    此時,由于將a變量的聲明推遲到另一個js文件中,導致第一次執行的時候console.log(a)代碼報錯,從而后續的js代碼不再執行。不過第二次執行start()方法仍然正常執行。這就是為什么幾乎所有地方都推薦大家使用“js命名空間”來部署不同的js文件。下面我們用一段代碼來總結聲明提升+命名空間如何巧妙的“打破作用于的牢籠”:

    /*-----------------第一個js文件----------------*/
    var App={};
    App.first=(function(){
     function a(){
     App.second.b();
     }
    
     return {
     a:a
     };
    })();
    
    /*-----------------另一個js文件----------------*/
    App.second=(function(){
     function b(){
     console.log("This is second.b");
     }
    
     return {
     b:b
     };
    })();
    
    //程序起點,
    輸出This is second.b App.first.a();

    這段程序將不會有任何報錯,我們可以在第一個js文件內訪問任何App命名空間后續的屬性,只要程序起點在所有必要的賦值工作之后執行,就不會有任何問題。這個例子成功的展示了如何通過合理的設計代碼結構來充分利用js語言的動態特性。

    看到這里讀者可能會覺得這文章有點標題黨,上面的技巧只是通過代碼布局來做出的一種“假象”:看上去前面的代碼在訪問不存在的屬性,實際上真正執行時的順序都是合理正確的。那下面本文將介紹真正的“跨作用于訪問”技巧。

    2.js執行時代碼

    大家都知道js語言有一個“eval()”方法,他就是一個典型的“真正打破作用于牢籠”的方法。看下面這段代碼:

    (function(){
     var code="console.log(a)";
     //This is a bird
     test(code);
    
     function test(code){
     console.log=function(arg){
     console.info("This is a "+arg);
     };
     var a="bird";
     eval(code);
     }
    })();

    看了這段代碼,相信很多人可能會不禁感嘆js的奇葩:“這也能行?!”。是的。test()方法由于聲明提升的機制,因此能夠被提前調用,正常執行。test()方法接受一個code參數,在test()方法內部我們重寫了console.log方法,修改了一下輸出格式,并且在test內部定義了一個私有變量var a=”bird”。在test方法最后我們使用eval來動態執行code的代碼,打印結果非常神奇:瀏覽器使用了我們重寫的console.log方法打印出了test方法內部的私有變量a。這是完全的作用域隔離。

    類似的方法在js中還有很多,例如:eval(),setTimeout(),setInterval()以及部分原生對象的構造方法。但是有兩點要提醒:

    (1)這種方式會大大降低程序的執行效率。大家都知道js本身是解釋性語言,其本身性能已經比編譯型語言慢了好多個級別。在這基礎之上如果我們再使用eval這樣的方法去“再編譯”一段字符串代碼,程序的性能將會慢很多。

    (2)使用這種方式編程會劇增代碼的復雜度,分分鐘你就會看不懂自己寫的代碼。本文介紹這種方法是希望能讓讀者全面的了解js語法特性從而能更好的修正、排錯。本文完全不推薦在生產級別的代碼中使用第二種方式。

    聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯系,我們將在第一時間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com

    文檔

    JavaScript打破作用域的牢籠

    JavaScript打破作用域的牢籠:JavaScript打破作用域的牢籠JavaScript作為一種松散型語言,有著很多令人瞠目結舌的特性(往往是一些令人捉摸不透的奇怪特性),本文我們將介紹如何使用JavaScript的一些特性來打破常規編程語言作用域的牢籠。1.JavaScript聲明提升很多人應該知道,
    推薦度:
    • 熱門焦點

    最新推薦

    猜你喜歡

    熱門推薦

    專題
    Top
    主站蜘蛛池模板: 精品国产一级在线观看| 四虎国产精品永久在线观看| 99久久成人国产精品免费| 无码国内精品人妻少妇| 久热这里只有精品12| 国产精品一区二区久久不卡 | 午夜DY888国产精品影院| 久久精品国产亚洲AV香蕉| 九九精品免视看国产成人| 天天视频国产精品| 国产伦精品一区二区三区女| 日韩精品人成在线播放| 精品久久久久久| 亚洲av永久无码精品古装片| 国产成人AV无码精品| 国产a精品视频| 欧美性videos高清精品| 日韩专区亚洲精品欧美专区| 99精品视频在线观看婷| 成人一区二区三区精品| 久久精品国产亚洲AV香蕉| 国产精品欧美一区二区三区| 国内精品人妻无码久久久影院| 免费人欧美日韩在线精品| 欧美精品免费专区在线观看| 中文字幕日韩精品有码视频| 国精品无码一区二区三区在线| 四虎永久在线精品国产免费| 亚欧洲精品在线视频免费观看 | 国产高清国内精品福利99久久| 久久99精品久久久久久秒播| 国产精品久久亚洲不卡动漫| 欲帝精品福利视频导航| 精品一区二区三区色花堂| 国产精品免费福利久久| 亚洲日韩精品无码专区网址| 精品乱子伦一区二区三区高清免费播放| 日产精品一线二线三线芒果| 日韩精品一区二三区中文| 国产这里有精品| 99久久夜色精品国产网站|