前言
在JavaScript中,作用域、上下文、閉包、函數等算是精華中的精華了。對於初級JSer來說,是進階必備。對前端攻城師來說,只有靜下心來,了解這些精華,才能寫出優雅的程式碼。
本文旨在總結容易忘記的重要知識,不會講基本的概念。如果對基本知識不太熟悉,就去翻下《 JavaScript權威指南》吧~
語言特性函數表達式
先看程式碼段:
[javascript] view plaincopyprint return typeof foo; // foo是在內部作用域內有效
};
// foo在外部用於是不可見的
f(); // "function"
var f = function foo(){
return typeof foo; // foo在內部作用域內有效
};
// foo在外部用於是不可見的
typeof foo; // "undefined"
f(); // "function"
json
很多JavaScript開發人員都錯誤地把JavaScript物件字面量(Object Literals)稱為JSON物件(JSON Objects)。 JSON是設計成描述資料交換格式的,它也有自己的語法,這個語法是JavaScript的子集。
{ “prop”: “val” } 這樣的聲明有可能是JavaScript物件字面量,也有可能是JSON字串,取決於什麼上下文使用它。如果是用在string上下文(用單引號或雙引 號引住,或者從text檔讀取)的話,那它就是JSON字串,如果是用在物件字面量上下文中,那它就是物件字面量。
// 這是物件字面量
var bar = { "prop": "val" };
// 這是JSON字串
var foo = '{ "prop": "val" }';
// 這是物件字面量
var bar = { "prop": "val" };
還有一點要知道的是,JSON.parse用來將JSON字串反序列化成對象,JSON.stringify用來將物件序列化成JSON字串。舊版的瀏覽器不支援這個對象,但你可以透過json2.js來實現同樣的功能。
// ...
}
cat.prototype = new Animal();//這種方式會繼承建構子裡面的。
cat.prototype = Animal.prototype;//這種方式不會繼承建構子裡面的。
//還有一個重要的細節要注意的就是一定要維護自己的原型鏈,新手總是會忘記這個!
cat.prototype.constructor = cat;
如果我們徹底改變函數的prototype屬性(透過分配一個新的物件),那麼原始建構函式的參考就是遺失,這是因為我們建立的物件不包括constructor屬性:
複製程式碼
A.prototype = {
x: 10
};
var a = new A();
alert(a.x); // 10
alert(a.constructor === A); / / false!
function A() {}
A.prototype = {
constructor: A,
x: 10
};
var a = new A();
alert(a.x); // 10
然而,提交prototype屬性不會影響已經創建物件的原型(只有在建構函式的prototype屬性改變的時候才會影響到),就是說新建立的物件才有新的原型,而已建立物件還是引用到原來的舊原型(這個原型已經不能再被修改了)。
function A() {}
function A() {}
A.prototype .x = 10;
var a = new A();
alert(a.x); // 10
A.prototype = {
constructor: A,
x: 20
constructor: A,
x: 20
constructor: A,
x: 20
y: 30
};
// 物件a是透過隱式的[[Prototype]]引用從原油的prototype上取得的值
var b = new A();
// 但新物件是從新原型上取得的值alert(b.x); // 20
alert(b.y) // 30複製程式碼
程式碼如下:
function foo(x, z) {
// 宣告的函數參數數arguments (x, y, z)
alert(arguments.length); // 2
// 參數的callee是函數本身 alert(arguments.callee === foo); // true
}
複製程式碼
程式碼如下:
var x = 10;
複製程式碼
程式碼如下:
a = 10;
alert(.a. ); // 10
alert(delete a); // true
alert(window.a); // undefined
var b = 20;
alert(window.b); // still 20。 b is variable,not property!
var a = 10; // 全域上下文中的變數(function () {
(function () {
alert(this); // null => global
})();
在這個例子中,我們有一個函數物件但不是引用型別的物件(它不是標示符,也不是屬性存取器),對應地,this值最終設為全域物件。
var foo = {
{
alert(this);
}
};
foo.bar(); // Reference, OK => foo
(foo.bar)(); // Reference, OK => foo
(foo.bar)(); // Reference, OK => foo
(foo.bar = foo.bar)(); // global
(false || foo.bar)(); // global
(foo.bar, foo.bar )(); // global
問題在於後面的三個調用,在應用一定的運算操作之後,在調用括號的左邊的值不在是引用類型。
•第一個例子很明顯—-明顯的引用類型,結果是,this為base對象,即foo。
•在第二個例子中,群組運算子並不適用,想想上面提到的,從引用類型中獲得一個物件真正的值的方法,如GetValue。對應的,在群組運算的返回中———我們得到仍是一個引用類型。這就是this值為什麼要再次設為base對象,也就是foo。
•第三個例子中,與群組運算子不同,賦值運算子呼叫了GetValue方法。傳回的結果是函數物件(但不是引用型別),這表示this設為null,結果是global物件。
•第四個和第五個也是一樣-逗號運算子和邏輯運算子(OR)呼叫了GetValue 方法,相應地,我們失去了引用而得到了函數。並再次設為global。
程式碼如下:
function foo() {
function foo() {
function {
alert(this); // global
}
活動物件總是作為this返回,值為null——(即偽代碼的AO.bar()相當於null.bar())。這裡我們再次回到上面描述的例子,this設定為全域物件。
透過函式建構函式所建立的函數的scope屬性總是唯一的全域物件。
複製程式碼程式碼如下:
var x = 10;
var x = 10;
function f🎜> {
var y = 20;
function barFD() { // 函數宣告
alert(x);
alert(x); alert(y);');
}foo ();
程式碼如下:
var x = 10, y = 10; with ({x: 20}) {
var x = 30, y = 30;
//這裡的x = 30 覆蓋了x = 20;
alert(x); // 30
alert(y); // 30
在進入上下文時發生了什麼事?標識符“x”和“y”已被加入到變數物件中。此外,在程式碼運行階段作如下修改:
•x = 10, y = 10;
•物件{x:20}加入作用域的前端;
•在with內部,遇到了var聲明,當然什麼也沒創建,因為在進入上下文時,所有變數已被解析添加;
•在第二步驟中,僅修改變數“x”,實際上物件中的“x”現在被解析,並添加到作用域鏈的最前端, 「x」為20,變成30;
•同樣也有變數物件「y」的修改,被解析後其值也對應的由10變成30;
•此外,在with聲明完成後,它的特定物件從作用域鏈中移除(已改變的變數「x」-30也從那個物件移除),即作用域鏈的結構恢復到with會加強先前的狀態。
•在最後兩個alert中,目前變數物件的「x」保持相同,「y」的值現在等於30,在with聲明運行中已改變。
函數
關於圓括號的問題
讓我們看下這個問題:『 為何在函數創建後的立即呼叫中必須用圓括號來包圍它? ',答案就是:表達式句子的限制就是這樣的。
依照標準,表達式語句不能以一個大括號 { 開始是因為他很難與程式碼區塊區分,同樣,他也不能以函數關鍵字開始,因為很難與函數宣告區分。即,所以,如果我們定義一個立即執行的函數,在其建立後立即以以下方式呼叫:
function () {
>}();
// 即使有名稱
function foo() {
...
}();
alert(foo); // 函數
function foo(x) {
alert(x);
}(1); // 這只是一個分組運算符,不是函數呼叫!
foo(10); // 這是一個真正的函數調用,結果是10
x);
})(1); // 這就是調用,不是分組運算子
return x % 2 != 0 ? 'yes' : 'no';
}(1)
};
alert(foo.bar); // 'yes'
alert(foo.bar); // 'yes'
就像我們看到的,foo.bar是一個字串而不是一個函數,這裡的函數僅僅用來根據條件參數初始化這個屬性——它創建後並立即調用。
1.因此,」關於圓括號」問題完整的答案如下:
2.當函數不在表達式的位置的時候,分組運算子圓括號是必須的-也就是手工將函數轉化成FE。
3.如果解析器知道它處理的是FE,就沒必要用圓括號。
自由變數:
function testFn() {
function testFn() {
;//對innerFn函數來說,localVar就屬於自由變數。
function innerFn(innerParam) {
alert(innerParam localVar);
}
return innerFn;
}
程式碼如下:
var = 10;
function foo() {
alert(z);
}
foo(); // 10 – 使用靜態和動態作用域的時候
(function () {
var z = 20;
foo(); // 10 – 使用靜態作用域, 20 – 使用動態作用域
})();
// 將foo當作參數的時候是相同的
(function (funArg) {
})(foo);
理論:因為作用域鏈,使得所有的函數都是閉包(與函數型別無關: 匿名函數,FE,NFE,FD都是閉包)。從實踐角度:以下函數才算是閉包:* 即使創建它的上下文已經銷毀,它仍然存在(例如,內部函數從父函數中返回)
* 在程式碼中引用了自由變數

去掉重复并排序的方法:1、使用“Array.from(new Set(arr))”或者“[…new Set(arr)]”语句,去掉数组中的重复元素,返回去重后的新数组;2、利用sort()对去重数组进行排序,语法“去重数组.sort()”。

本篇文章给大家带来了关于JavaScript的相关知识,其中主要介绍了关于Symbol类型、隐藏属性及全局注册表的相关问题,包括了Symbol类型的描述、Symbol不会隐式转字符串等问题,下面一起来看一下,希望对大家有帮助。

怎么制作文字轮播与图片轮播?大家第一想到的是不是利用js,其实利用纯CSS也能实现文字轮播与图片轮播,下面来看看实现方法,希望对大家有所帮助!

本篇文章给大家带来了关于JavaScript的相关知识,其中主要介绍了关于对象的构造函数和new操作符,构造函数是所有对象的成员方法中,最早被调用的那个,下面一起来看一下吧,希望对大家有帮助。

本篇文章给大家带来了关于JavaScript的相关知识,其中主要介绍了关于面向对象的相关问题,包括了属性描述符、数据描述符、存取描述符等等内容,下面一起来看一下,希望对大家有帮助。

方法:1、利用“点击元素对象.unbind("click");”方法,该方法可以移除被选元素的事件处理程序;2、利用“点击元素对象.off("click");”方法,该方法可以移除通过on()方法添加的事件处理程序。

foreach不是es6的方法。foreach是es3中一个遍历数组的方法,可以调用数组的每个元素,并将元素传给回调函数进行处理,语法“array.forEach(function(当前元素,索引,数组){...})”;该方法不处理空数组。

本篇文章给大家带来了关于JavaScript的相关知识,其中主要介绍了关于BOM操作的相关问题,包括了window对象的常见事件、JavaScript执行机制等等相关内容,下面一起来看一下,希望对大家有帮助。


熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

AI Hentai Generator
免費產生 AI 無盡。

熱門文章

熱工具

SublimeText3漢化版
中文版,非常好用

PhpStorm Mac 版本
最新(2018.2.1 )專業的PHP整合開發工具

EditPlus 中文破解版
體積小,語法高亮,不支援程式碼提示功能

MantisBT
Mantis是一個易於部署的基於Web的缺陷追蹤工具,用於幫助產品缺陷追蹤。它需要PHP、MySQL和一個Web伺服器。請查看我們的演示和託管服務。

SAP NetWeaver Server Adapter for Eclipse
將Eclipse與SAP NetWeaver應用伺服器整合。