成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Javascript動態(tài)作用域

DC_er / 2654人閱讀

摘要:獲取返回的匿名函數(shù)這個匿名函數(shù)會保留原本的作用域鏈有意思的是函數(shù)參數(shù)也是可以被我們捕獲到的這就給我們靈活的創(chuàng)造一些函數(shù)提供了便利,比如我們需要創(chuàng)造一個函數(shù)工廠,這個工廠可以根據(jù)我們提供的參數(shù)生產(chǎn)出不同的函數(shù)。

本文是在看《Javascript函數(shù)式》編程一書寫下的一些記錄。和大家分享。不足之處還望大家指正。

關于this的討論

首先來看這么幾段代碼

function globalThis(){return this;}

globalThis();
//=>window or global object

globalThis.call("haha");
//=>"haha"

globalThis.apply("abc",[]);
//=>"abc

可以看到,this的一般就是由調用他的對象決定的,如果進行綁定了的話,相當于說這個函數(shù)的調用對象只能夠是你綁定的那個對象,是不能夠更改的。

var bindThis = globalThis.bind("abc")

bindThis();
//=>"abc"

bindThis.call("x")
//=>"abc"

bindThis.apply("y",[]);
//=>"abc

當然,如果我看到這本書推薦大家使用underscore庫。用法如下

_.bind(globalThis,"abc")

這樣的操作也是可以的。如果說有很多個函數(shù)都需要綁定到同一個對象上去怎么辦呢?underscore提供了bindAll(obj,methondName)

var buttonView = {
  label  : "underscore",
  onClick: function(){ alert("clicked: " + this.label); },
  onHover: function(){ console.log("hovering: " + this.label); }
};
_.bindAll(buttonView, "onClick", "onHover");
函數(shù)的閉包

相信大家都或多或少踩過閉包這個坑吧,確實一開始接觸感覺很不能理解。我個人粗淺理解是閉包是一個函數(shù)執(zhí)行過后返回一個內(nèi)部函數(shù),這個內(nèi)部函數(shù)將保留包含這個內(nèi)部函數(shù)的函數(shù)的作用域鏈。也就是里面把外面包住了,簡稱閉包2333。

function captureOut(){
    var a = 123;
    return function(){
        console.log("a:"+a);
    }
}
var getA = captureOut();//獲取captureOut返回的匿名函數(shù)
getA();//這個匿名函數(shù)會保留原本的作用域鏈
//=>a:123

有意思的是函數(shù)參數(shù)也是可以被我們捕獲到的

function capturePara(PARA){
    return function(){
        console.log(PARA);
    }
}
var getP = capturePara("I"m the parameters");
getP();//I"m the parameters

這就給我們靈活的創(chuàng)造一些函數(shù)提供了便利,比如我們需要創(chuàng)造一個函數(shù)工廠,這個工廠可以根據(jù)我們提供的參數(shù)生產(chǎn)出不同的函數(shù)。見下面代碼

function createDivider(divFactor){
    return function(num){
        return num/divFactor
    }
}

var div9 = createDivider(9);//創(chuàng)造一個可以用來除以9的函數(shù)
var div3 = createDivider(3);//除以3的
div9(81);//=>9
_.map([9,18,27],div3);//=>[3,6,9]

既然可以訪問函數(shù)內(nèi)部變量,那么自然也可以訪問this咯,可是this是會隨著調用對象不同而變化的,我們可以通過其他名字來保存this

function captureThis(NAME){
    this.name = NAME;
    var that = this;
    return function(){
        return that.name;
    }
}
var getThis = captureThis("小花");
getThis.call({});
//=>小花

可以看到,雖然我們重新把getThis綁定到其他地方去了,還是能夠得到我們的“小花”。如果我們再一次利用captureThis()函數(shù)來創(chuàng)建一個新的函數(shù),綁定新的值不會影響到原來的“小花”

var getHong = captureThis("小紅");
getHong.call({});
//=>小紅

剛剛我們討論的都是內(nèi)部變量和外部變量名字不同的情況,如果相同會出現(xiàn)什么現(xiàn)象呢?繼續(xù)往下看吧

var name = "大黃";
function captureName(name){
    return function(){
        return name;
    }
}
var getName = captureName("阿狗");
getName();//?

會領養(yǎng)到阿狗還是大黃呢?其實這個還算簡單,返回的閉包就是返回原來的作用域鏈,首先訪問到的當然是最近的name,因此正確答案是“阿狗”(直接拷貝代碼到console就可以測試)
再來看下面的例子

function captureName(name){
    return function(name){
        return name;
    }
}
var getName = captureName("阿黃");
getName("大狗");//=>?

這一次其實也差不多,相同的變量同時存在于外包函數(shù)參數(shù)和內(nèi)部匿名函數(shù)的參數(shù)中,我們還是按照就近原則,最近的當然是內(nèi)部匿名函數(shù)的參數(shù),因此這次拿到的是“大狗”。

注意,只要拿到了閉包的返回函數(shù),即便是修改原來外部的函數(shù)也不會對現(xiàn)有接收到的返回函數(shù)造成影響。比如說把captureName改為null,那么照樣可以使用getName。

不過下面的代碼可能讓你有些困惑

function showObj(obj){
    return function(){
        return obj
    }
}
var a = 10;
var showA = showObj(a);
showA();//=>10;
a=20;
showA();//=>10;


var b = {name:"daming"}
var showB = showObj(b);
showB();//{name:"daming"}
b.age =12;
showB();//{name:"daming",age:12}
b=null;
showB();//{name:"daming",age:12}

是不是覺得有點暈,我也有點暈。書上是說,“由于引用對象同時存在于閉包內(nèi)部和閉包外部,它的變化可以跨越看似私有的界限,很容易導致混亂,所以通常都盡量減少暴露捕獲變量的風險,把捕獲的對象作為私有數(shù)據(jù)?!?/p>

var pingpong = (function(){
    var private=0;
    
    return{
        inc:function(n){
            return private+=n;
        },
        dec:function(n){
            return private-=n;
        }
    }
})();

pingpong.inc(3)//=>3

這樣對象是很安全的,甚至可以禁止往閉包里添加函數(shù)!

pingpong.showP = function(){return PRIVATE;}
pingpong.showP();//notdefined

總結:靈活利用this以及閉包是實現(xiàn)函數(shù)式編程的基礎,而正如我們看到的,函數(shù)式編程是一種安全而優(yōu)美的編程方式~

文章版權歸作者所有,未經(jīng)允許請勿轉載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉載請注明本文地址:http://systransis.cn/yun/87968.html

相關文章

  • JavaScript深入之詞法作用動態(tài)作用

    摘要:作用域作用域是指程序源代碼中定義變量的區(qū)域。采用詞法作用域,也就是靜態(tài)作用域。而與詞法作用域相對的是動態(tài)作用域,函數(shù)的作用域是在函數(shù)調用的時候才決定的。前面我們已經(jīng)說了,采用的是靜態(tài)作用域,所以這個例子的結果是。 JavaScript深入系列的第二篇,JavaScript采用詞法作用域,什么語言采用了動態(tài)作用域?兩者的區(qū)別又是什么?還有一個略難的思考題,快來看看吧。 作用域 作用域是指...

    gclove 評論0 收藏0
  • 深入學習js之——詞法作用動態(tài)作用

    摘要:在中的應用采用詞法作用域,也就是靜態(tài)作用域。那什么又是詞法作用域或者靜態(tài)作用域呢請繼續(xù)往下看靜態(tài)作用域與動態(tài)作用域因為采用的是詞法作用域函數(shù)的作用域在函數(shù)定義的時候就決定了。 開篇 當我們在開始學習任何一門語言的時候,都會接觸到變量的概念,變量的出現(xiàn)其實是為了解決一個問題,為的是存儲某些值,進而,存儲某些值的目的是為了在之后對這個值進行訪問或者修改,正是這種存儲和訪問變量的能力將狀態(tài)給...

    shiweifu 評論0 收藏0
  • javascript靜態(tài)作用動態(tài)作用

    摘要:靜態(tài)作用域指的是一段代碼,在它執(zhí)行之前就已經(jīng)確定了它的作用域,簡單來說就是在執(zhí)行之前就確定了它可以應用哪些地方的作用域變量。 靜態(tài)作用域指的是一段代碼,在它執(zhí)行之前就已經(jīng)確定了它的作用域,簡單來說就是在執(zhí)行之前就確定了它可以應用哪些地方的作用域(變量)。 動態(tài)作用域–函數(shù)的作用域是在函數(shù)調用的時候才決定的 JavaScript采用的是詞法作用域即靜態(tài)作用域; // 靜態(tài)作用域: va...

    jimhs 評論0 收藏0
  • javascript中的作用(詞法and動態(tài))

    摘要:中作用域的問題可以說是老生常談,個人認為的作用域中存在著兩種作用域,一種是詞法作用域,一種是動態(tài)作用域。但是自動有了箭頭函數(shù)后,箭頭函數(shù)中的并不是動態(tài)作用域,而是屬于詞法作用域,再其定義時就已經(jīng)確定好了,相當于。 js中作用域的問題可以說是老生常談,個人認為js的作用域中存在著兩種作用域,一種是詞法作用域,一種是動態(tài)作用域。 詞法作用域 詞法作用域就是定義在詞法階段的作用域,也就是說由...

    Taonce 評論0 收藏0
  • You-Dont-Know-JS - 詞法作用

    摘要:原文原文原文詞法作用域作用域有兩種常見的模型,一種叫做詞法作用域,一種叫做動態(tài)作用域。其中詞法作用域更常見,被大多數(shù)語言采用,包括。值得注意的是,一個函數(shù)作用域只有可能存在于一個父級作用域中,不會同時存在兩個父級作用域。 原文: 原文1 | 原文2 Lexical Scope - 詞法作用域 作用域有兩種常見的模型,一種叫做 詞法作用域 Lexical Scope,一種叫做...

    bang590 評論0 收藏0
  • 先有蛋還是先有雞?JavaScript 作用與閉包探析

    摘要:而閉包的神奇之處正是可以阻止事情的發(fā)生。拜所聲明的位置所賜,它擁有涵蓋內(nèi)部作用域的閉包,使得該作用域能夠一直存活,以供在之后任何時間進行引用。依然持有對該作用域的引用,而這個引用就叫閉包。 引子 先看一個問題,下面兩個代碼片段會輸出什么? // Snippet 1 a = 2; var a; console.log(a); // Snippet 2 console.log(a); v...

    elisa.yang 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<