电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術文章
文章詳情頁

JavaScript中的執行環境和作用域鏈

瀏覽:85日期:2023-10-15 10:40:23

前言

JS 中的執行環境和作用域鏈是非常重要的概念,它們是 JS 引擎在處理 JS 代碼的時候對變量和函數的處理方式,這兩個概念的正確理解能夠幫助我們更好地理解和預測代碼的行為。

執行環境

執行環境定義了變量或者函數有權訪問的數據集合,每一個執行環境都有一個與之關聯的變量對象,該執行環境中定義的所有變量和函數都保存在這個對象中。我們無法直接訪問這個對象,這個對象只是在解析器處理數據的時候使用。

我們平時說的全局變量就是在最外圍的一個執行環境中定義的變量,全局執行環境根據 ECMAScript 的不同實現而有不同的表示,在 Web 瀏覽器中,全局執行環境就是 window 對象,所有的全局變量和函數就是作為 window 對象的屬性和方法創建的。在 nodejs 的實現中,全局執行環境就是global 對象。除了全局執行環境,每個函數都有自己的執行環境,當執行流進入一個函數時,函數的環境就會被推入一個環境棧中,而函數執行之后,棧將其環境彈出,把控制權返回給之前的執行環境。也就是說某個執行環境中的代碼全部執行完畢之后,該環境就被銷毀,保存在其中的所有變量和函數定義也隨之銷毀,全局執行環境直到應用程序額推出——例如網頁或瀏覽器被關閉時才被銷毀。

作用域鏈

前面說到每個執行環境都有一個變量對象來保存環境中定義的變量和函數,環境是層層嵌套的,所以當代碼進入到一個新的環境開始執行時,會創建變量對象的一個作用域鏈,把嵌套的執行環境之間的變量對象做一個有序的聯系。作用域鏈最主要的作用是確保當前執行環境有權訪問的變量和函數,并且有序地查找。在作用域鏈的最前端始終是當前正在執行的代碼所處的執行環境的變量對象,如果這個環境是一個函數,就把函數的活動對象作為其變量對象,在函數中沒有定義新的變量時,這個活動獨享就是函數的 arguments 對象。作用域鏈的下一個變量都西昂來自于當前執行環境的包含環境,依次類推,逐層嵌套,知道全局執行環境;全局執行環境的變量對象始終都是作用域鏈中的最后一個都對象。

當我們的代碼在執行的時候,遇到的每一個標識符解析都會沿著作用域鏈一級一級地進行搜索,從作用域鏈的前端(當前執行環境的變量對象)逐級向后回溯,知道找到標識符為止,如果在作用域鏈上沒有找到這個標識符,通常會導致錯誤。我們經常遇到的 Uncaught ReferenceError: x is not defined 就是這個錯誤在瀏覽器中的表現。

JS解釋器在執行時會將變量和函數進行聲明提前,在聲明函數的時候,會給函數一個 [[scope]] 屬性,這個屬性中包含了當前函數所有包含環境的變量對象,也就是我們的函數在聲明提前的時候就已經生成了他的包含環境的作用域鏈了,然后當函數執行的時候會把自己的 arguments 和內部定義的函數和變量打包成一個變量對象加到 scope chain 的最后。

函數參數也被當做變量來對待,因此起訪問規則與執行環境中的其他變量相同。

作用域鏈的這種特性理解起來其實也是比較直觀的,但是在實際的代碼中由于情況非常多,有時候有些行為還是比較反直覺或者說容易產生誤解的。比如下面的情況:

作用域鏈看的是函數定義的位置而不是執行的位置

var x = 10bar()function foo() { console.log(x)}function bar(){ var x = 30 foo()}

在這個例子里面,可能會有人誤以為 bar() 會輸出 30,我們只要理解函數其實是保存在堆中,我們給函數命名只是一個指向函數堆中地址的一個引用,當我們執行函數的時候根據這個引用去堆中找對應的函數執行。所以無論我們在哪里執行函數,函數的位置都是不變的,我們看作用域鏈也是,我們確定作用域鏈不是看函數是在哪里執行,而是要看函數是在哪里定義,作用域鏈可以認為是函數聲明時就已經生成了。

個人認為 ECMAScript 這樣處理作用域鏈是為了作用域鏈能夠保持不變而不用一直維護,并且根據環境的嵌套保持一致性。

閉包

除了全局執行環境的變量對象是始終存在的,其他局部函數的變量對象都只在函數的執行過程中存在,一般來講,函數執行完畢之后,局部活動對象就被銷毀了,內存中僅僅保存全局執行環境的變量對象,但是閉包的情況是不同的。

閉包指的是有權訪問另一個函數作用域中的變量的函數,比如下面這樣:

function outer(){ var scope = 'outer'; return function (){ return scope; } }var fn = outer();fn();

在一個函數內部定義的函數會將包含函數(即外部函數)的活動對象添加到他的作用域鏈中,因此在 outer 函數內部定義的匿名函數(我們下面把這個匿名函數稱為 inner 函數)的作用域鏈中,實際上會包含外部函數 outer() 的活動對象,下圖可以看書當代碼執行時,outer 和 inner 函數的作用域鏈。

JavaScript中的執行環境和作用域鏈

當匿名函數從 outer() 中被返回后,inner() 函數仍然可以訪問在 outer() 中定義的所有變量,也就是說,當 outer() 函數執行完畢后,其活動對象也不會被銷毀,因為匿名函數的作用域鏈依然在引用這個活動對象。換句話說,當 outer() 函數執行完畢返回后,其執行環境和作用域鏈都被銷毀,但它的活動對象依然保存在內存中,如果匿名函數不銷毀,則這個活動對象會一直存在于內存中。

js中的對象都是保存在堆中,我們在代碼中寫的都是對對象的引用,作用域鏈中也是,所以上面說的 outer() 函數執行完畢后作用域鏈被銷毀但是對象還存在,其實銷毀的只是引用, js 中的垃圾處理機制的一種策略是引用計數,當某個變量或對象的引用次數為 0 的時候內存會被收回。outer 函數的變量對象的引用有兩個一個是 outer 的作用域鏈和匿名函數的作用域鏈,所以只要匿名函數不被銷毀,這個引用就一直存在,outer() 的活動對象也會一直存在。

輪子哥在知乎給過一個比較容易理解的說法:“閉”的意思不是封閉內部狀態,而是封閉外部狀態,一個函數如何能夠封閉外部狀態呢,當外部狀態的 scope 失效的時候,它自己還保留了一份。

JavaScript中的執行環境和作用域鏈

由于閉包會攜帶包含它的函數的作用域,因此回避其他函數占用更多的內存。過度使用閉包可能會導致內存占用過多,只在必要的時候使用閉包。

總結

任何一種編程語言都有作用域的概念,我們的程序是圍繞著變量操作的,那么在設計語言的時候,變量如何儲存,儲存到哪里,我們的程序如何找到對應的變量就是一個首先要解決的問題。而作用域就是語言設計者針對這個問題編寫的一套設計良好的規則來存儲并搜索對象,這就是作用域的概念。而 JS 中的這個規則就是作用域鏈,我們在編寫程序的時候也需要知道我們的變量(以及函數)是如何儲存,以及 JS 引擎在遇到標識符解析的時候是按照什么規則來搜索變量或者函數的,只有這樣我們才能寫出更可靠的代碼。

以上就是JavaScript中的執行環境和作用域鏈的詳細內容,更多關于JavaScript執行環境和作用域鏈的資料請關注好吧啦網其它相關文章!

標簽: JavaScript
相關文章:
主站蜘蛛池模板: 【中联邦】增稠剂_增稠粉_水性增稠剂_涂料增稠剂_工业增稠剂生产厂家 | 3dmax渲染-效果图渲染-影视动画渲染-北京快渲科技有限公司 | 长沙一级消防工程公司_智能化弱电_机电安装_亮化工程专业施工承包_湖南公共安全工程有限公司 | 飞行者联盟-飞机模拟机_无人机_低空经济_航空技术交流平台 | 真丝围巾|真丝丝巾|羊绒围巾|围巾品牌|浙江越缇围巾厂家定制 | 酒精检测棒,数显温湿度计,酒安酒精测试仪,酒精检测仪,呼气式酒精检测仪-郑州欧诺仪器有限公司 | 代办建筑资质升级-建筑资质延期就找上海国信启航 | 新能源汽车电池软连接,铜铝复合膜柔性连接,电力母排-容发智能科技(无锡)有限公司 | 浙江宝泉阀门有限公司| 内窥镜-工业内窥镜厂家【上海修远仪器仪表有限公司】 | 软装设计-提供软装装饰和软装配饰及软装陈设的软装设计公司 | 一点车讯-汽车网站,每天一点最新车讯! | 硫化罐-电加热蒸汽硫化罐生产厂家-山东鑫泰鑫智能装备有限公司 | 气密性检测仪_气密性检测设备_防水测试仪_密封测试仪-岳信仪器 | 吉林污水处理公司,长春工业污水处理设备,净水设备-长春易洁环保科技有限公司 | 手持式线材张力计-套帽式风量罩-深圳市欧亚精密仪器有限公司 | 热熔胶网膜|pes热熔网膜价格|eva热熔胶膜|热熔胶膜|tpu热熔胶膜厂家-苏州惠洋胶粘制品有限公司 | 户外环保不锈钢垃圾桶_标识标牌制作_园林公园椅厂家_花箱定制-北京汇众环艺 | 沥青灌缝机_路面灌缝机_道路灌缝机_沥青灌缝机厂家_济宁萨奥机械有限公司 | POS机办理_个人POS机免费领取 - 银联POS机申请首页 | 蚂蚁分类信息系统 - PHP同城分类信息系统 - MayiCMS | 进口便携式天平,外校_十万分之一分析天平,奥豪斯工业台秤,V2000防水秤-重庆珂偌德科技有限公司(www.crdkj.com) | 27PR跨境电商导航 | 专注外贸跨境电商 | 户外环保不锈钢垃圾桶_标识标牌制作_园林公园椅厂家_花箱定制-北京汇众环艺 | 软膜天花_软膜灯箱_首选乐创品牌_一站式天花软膜材料供应商! | 粉末包装机-给袋式包装机-全自动包装机-颗粒-液体-食品-酱腌菜包装机生产线【润立机械】 | 复盛空压机配件-空气压缩机-复盛空压机(华北)总代理 | 隧道烘箱_隧道烘箱生产厂家-上海冠顶专业生产烘道设备 | 国际线缆连接网 - 连接器_线缆线束加工行业门户网站 | 软瓷_柔性面砖_软瓷砖_柔性石材_MCM软瓷厂家_湖北博悦佳软瓷 | sus630/303cu不锈钢棒,440C/430F/17-4ph不锈钢研磨棒-江苏德镍金属科技有限公司 | 上海深蓝_缠绕机_缠膜机-上海深蓝机械装备有限公司 | 大型低温冷却液循环泵-低温水槽冷阱「厂家品牌」京华仪器_京华仪器 | 千斤顶,液压千斤顶-力良企业,专业的液压千斤顶制造商,shliliang.com | 贴板式电磁阀-不锈钢-气动上展式放料阀-上海弗雷西阀门有限公司 工业机械三维动画制作 环保设备原理三维演示动画 自动化装配产线三维动画制作公司-南京燃动数字 | 真空粉体取样阀,电动楔式闸阀,电动针型阀-耐苛尔(上海)自动化仪表有限公司 | ALC墙板_ALC轻质隔墙板_隔音防火墙板_轻质隔墙材料-湖北博悦佳 | 合肥废气治理设备_安徽除尘设备_工业废气处理设备厂家-盈凯环保 合肥防火门窗/隔断_合肥防火卷帘门厂家_安徽耐火窗_良万消防设备有限公司 | 深圳公司注册-工商注册公司-千百顺代理记账公司 | 橡胶膜片,夹布膜片,橡胶隔膜密封,泵阀设备密封膜片-衡水汉丰橡塑科技公司网站 | EDLC超级法拉电容器_LIC锂离子超级电容_超级电容模组_软包单体电容电池_轴向薄膜电力电容器_深圳佳名兴电容有限公司_JMX专注中高端品牌电容生产厂家 |