這一年中零零散散看過幾本javascript的書,回過頭看以前寫過的javascript學習筆記,未免有點汗顏,突出「膚淺」二字,然越深刻越以爲javascript的博大精深,有種只緣身在此山中的感受,茫茫然而不得其要領,索性在一邊寫博文中,求得突破,乃至更上一層樓。javascript
看過的書籍推薦:java
《javascript語言精粹》 Douglas Crockford編程
《javascript設計模式》 Addy Osmanijson
《javascript設計模式》 Ross Harmes設計模式
《悟透javascript》 李戰數組
《javascript語言精髓與編程實踐》 周愛民瀏覽器
這些書是在絕大部分javascript書籍中較好的幾本了,市面上的javascript,不得不說,水分太多,營養太少。不少書翻幾頁也就困了。反卻是網上有些不錯的博客,博客園裏面,博客園的知識庫裏面,還有一些其餘的我的博客,都是挺不錯的,可是大多都比較分散,知識點也比較散落。另外因爲隨着javascript自己的更新,還有jQuery,YUI等一系列插件的出現,NodeJS興起等等的影響,不一樣瀏覽器的更新以及兼容性問題,致使了原生JS的逐漸冷落,可是無論怎麼說,也遮不住JS這門語言自己的魅力。閉包
JS有不少深邃的地方值得學習,接下來着重介紹一些比較好玩的東西,這些也是我看了不少的書和博文累積起來的,可是最好的讀者是稍微有點JS基礎了,由於我講的東西可能會有點跳躍,不是基礎的教程。app
不講解那些複雜拗口的語法,或者那些深坑,或者那些晦澀難懂的設計模式,經過簡單的實例來學習實踐區分靜態語言和動態語言的區別於共同點函數
樓主是一個記性不好,特別怕麻煩的人,樓主忠於且堅信用最簡單的技術實現用戶需求,能抓老鼠的貓纔是貓的原則,反對過度追求優化,追求複雜的設計模式。
只求能幹淨的實現個人功能。主要分如下幾點來說解:
1. var 變量 聲明 函數聲明 提早
2. call apply的使用
3. new 實例化,複製,引用的區別
4. 原型鏈的使用
5. 閉包保持用戶狀態
6. 面向對象的實現,繼承
7. this指針
8. 數組,對象,json
9. AMD,CMD簡介
10. js中的深坑
11. 繼續深刻方向
1. var 變量 聲明 函數聲明提早
提到變量/函數聲明,理解一個核心,JS的解析和執行時分開的,瞭解了這個,不少問題和怪事就都明白了。
JS代碼在執行的時候,會通過兩個步驟,首先是解析(預編譯),而後再執行。解析就是聲明變量,尚未賦值,因此是undefined的,到了具體執行的時候纔會給予具體的值。要把一條賦值語句分紅兩部分看,例如var a = 5; 應該是 var a; a = 5;這樣兩部分看,複雜的狀況以下幾個例子:
/** * 變量聲明和函數表達式(我更以爲應該叫定義,可是你們好像默認都是聲明瞭) * 都是用var來聲明,因此若是將函數具體內容看作是一個值得話,跟變量賦值實際上是一回事 * 下面語句執行的順序以下: * 首先進行變量聲明,即var f;因此第一個輸出時undefined * 由於有兩個var f的定義,以後的定義覆蓋以前的定義,因此第二個輸出時25 * 第三個輸出時f函數的具體內容 * 若是將var f = 0; 和var f = function(x){return x*x;}順序倒過來,那第二個輸出就要報錯了, * 第三個輸出是0 * */ //console.log(f); //var f = 0; //var f = function(x){ // return x*x; //} //console.log(f(5)); //console.log(f); /** * 變量聲明和函數聲明 * 下面語句的執行輸出結果以下: * 第一個輸出是函數的具體內容 * 第二個輸出是16, * 第三個輸出是0 * 問題在於從技術上說,function語句並不是是一個語句. * JS中語句會引起動態的行爲, * 可是函數定義描述的倒是靜態的程序結構。 * 語句是在運行時執行的,而函數則是實際運行以前, * 當JS代碼被解析或被編譯時定義的。 * 當JS解析程序遇到一個函數定義時,它就解析並存儲構成函數主體的語句, * 而後定義一個和該函數同名的屬性 * (若是函數定義在其它的函數中,那麼就在調用對象中定義這個屬性, * 不然在全局對象中定義這個屬性)以保存它。這會產生一些奇怪的行爲。 * 總結來講就是函數聲明優於比變量聲明。 */ console.log(f); console.log(f(4)); function f(x){ return x*x; } var f = 0; console.log(f);
事實上,js的解析器對函數聲明與函數表達式並非一視同仁地對待的。對於函數聲明,js解析器會優先讀取,確保在全部代碼執行以前聲明已經被解析,而函數表達式,如同定義其它基本類型的變量同樣,只在執行到某一句時也會對其進行解析,因此在實際中,它們仍是會有差別的,具體表如今,當使用函數聲明的形式來定義函數時,可將調用語句寫在函數聲明以前,然後者,這樣作的話會報錯。
2. call apply的使用
一、方法定義
call方法:
語法:call([thisObj[,arg1[, arg2[, [,.argN]]]]])
定義:調用一個對象的一個方法,以另外一個對象替換當前對象。
說明:
call 方法能夠用來代替另外一個對象調用一個方法。call 方法可將一個函數的對象上下文從初始的上下文改變爲由 thisObj 指定的新對象。
若是沒有提供 thisObj 參數,那麼 Global 對象被用做 thisObj。
apply方法:
語法:apply([thisObj[,argArray]])
定義:應用某一對象的一個方法,用另外一個對象替換當前對象。
說明:
若是 argArray 不是一個有效的數組或者不是 arguments 對象,那麼將致使一個 TypeError。
若是沒有提供 argArray 和 thisObj 任何一個參數,那麼 Global 對象將被用做 thisObj, 而且沒法被傳遞任何參數。
簡單來講,區別就是apply方法後面傳遞的參數是數組,而call方法是add.call(doubleAdd,a,b,c)形式的. 具體執行的方法看call、apply前面的函數,
若是沒有提供thisObj,至關於Global。
具體來講,看幾個例子:
function add(a,b) { alert(a+b); } function sub(a,b) { alert(a-b); } add.call(sub,3,1); //== add.call(null,3,1); //輸出4
sub.call(null,3,1);
//輸出2
//這裏稍微涉及到this指針,this指向的是運行時的上下文 function Class1() { this.name = "class1"; this.showNam = function() { alert(this.name); } this.detail = "hello-1"; this.showDetail = function(){ alert(this.detail); } } function Class2() { this.name = "class2"; this.detail = "hello-2"; this.showDetail = function(){ alert(this.detail); } } var c1 = new Class1(); var c2 = new Class2(); c1.showNam.call(c2); c1.showNam(); c1.showDetail(); c2.showDetail(); // class2 // class1 //hello-1 //hello-2
還能夠用call實現繼承,具體例子就不給出了,參考文章JS中的call()和apply()方法 ...裏面寫的很詳細,該說的也都說了。
3. new 實例化,複製,引用的區別
記住function有對象化能力,體如今使用new上。若是沒有使用new新建立一個對象(即複製一份相同內容), 即仍是引用,對函數內部變量的改變會致使其餘引用值的改變。
這裏涉及到對象的概念,統一在對象的時候再說吧。。今天就寫到這裏了。
以上所有都屬我的原創,請你們轉載的時候附上原創連接: http://www.cnblogs.com/tonylp