今天想要查下Node的類型什麼的知識,想要總結下,在Googol上看到一個文章,可是原始的連接不在了,在快照中把這篇文章拉出來,若是原做者有問題,請聯繫我!
該文章都是一些JS的基礎,高手自動跳過!我以前沒怎麼寫過js,這方面比較弱,因此在寫node的時候也遇到了麻煩,這裏給本身補充下知識!
文章名:一塊兒擼Node.JS(壹)——基本語法和類型javascript
Node.js 的基礎是 JavaScript 這門 腳本語言。而大多數的腳本語言一個共同的特色就是「弱類型」。java
不一樣於 PHP 的是,PHP 就是是有了新變量也無需申明,而 JavaScript 則仍是須要 var 來申明一下的。而這個 var 涵蓋了 C++ 中的int、string、char等一切類型的含義,甚至是 function。node
本篇以及後篇的全部內容都是在 Linux 或者 Cygwin 下用 vim 進行編輯(若不是則請自行轉變成你本身的方法),而後在命令行下進行查看結果的。shell
在 C/C++ 中,咱們這麼聲明變量的:vim
void foo() {} int a = 0; char b = 'a'; float c = 1.0f; void (*d)() = foo; ///< 忘了是否是這麼寫的了,總之是函數指針
而在 Node.js 中則是這樣的:數組
javascriptfunction foo() {} var a = 0; var b = 'a'; var c = 1.0; var d = foo;
因此,不管是什麼類型的變量,在 Node.js 中都是以一個 var 來解決的。函數
這個循環語句基本上跟 C/C++ 同樣,都是測試
for(int i = 0; i < foo; i++) { //... }
而鑑於 Node.js 是弱類型,因此只須要:this
javascriptfor(var i = 0; i < foo; i++) { //... }
這是一種後有型的循環語句,相似於 PHP 的 foreach。prototype
好比咱們有一個 JSON對象 以下:
javascriptvar foo = { "hello" : "world", "node" : "js", "blahblah" : "bar" };
這個時候咱們就能夠用 for...in 來循環遍歷了:
javascriptfor(var key in foo) { console.log(key + ": " + foo[key]); }
咱們若是在命令行中打入下面的命令:
$ node foo.js
屏幕上就會顯示下面的內容了:
javascripthello: world node: js blahblah: bar
提示:由上可知,for...in 語句是用來遍歷 JSON對象、數組、對象的鍵名的,而不提供鍵值的遍歷。若是要獲取鍵值,只能經過foo[<當前鍵名>]的形式來獲取。
這個跟 PHP 的 foreach 仍是有必定區別的。
這個就很少作解釋了,跟其它語言沒什麼大的區別,無非就是若是有變量聲明的話,須要用 var 就夠了。
這幾個運算符也就這樣,要注意的是 +。它既能夠做用於字符串,也能夠做用於數值運算。弱類型語言雖說類型是弱的,數字有時候能夠以字符串的形態出現,字符串有時候能夠用數值的形態出現,可是在必要的時候也仍是要說一下它是什麼類型的,咱們能夠用下面的代碼去看看結果:
javascriptvar a = "1"; var b = 2; console.log(a + b); console.log(parseInt(a) + b);
這裏的 parseInt 是 Node.js 的一個內置函數,做用是將一個字符串解析成 int 類型的變量。
上面的代碼執行結果是:
12 3
注:第一個 console.log 結果是 12,因爲 a 是字符串,因此 b 也被系統以字符串的姿態進行加操做,結果就是將兩個字符串黏連在一塊兒就變成了 12。而第二個 console.log 結果是 3,是由於咱們將第一個 a 轉變爲了 int 類型,兩個 int 型的變量相加即數值相加,結果固然就是 3 了。
這裏有一點要解釋,當這個邏輯運算符長度爲 2 的時候(==, !=),只是判斷外在的值是否是同樣的,而不會判斷類型。如
javascriptvar a = 1, b = "1"; console.log(a == b);
它輸出的結果就是 true。可是若是咱們在中間判斷的時候再加上一個等號,那麼就是嚴格判斷了,須要類型和值都同樣的時候纔會是 true,不然就是 false。也就是說
javascriptvar a = 1, b = "1"; console.log(a === b);
的時候,返回的結果就是 false 了,由於 a 是 int 型的,而 b 則是字符串。
順帶着就把條件語句講了吧,其實這裏的 if 跟別的語言沒什麼兩樣,就是幾個邏輯運算符兩個等號三個等號的問題。因此就很少作累述了。
這裏我姑且把它當成是一個運算符而不是函數了。
這個運算符的做用是判斷一個變量的類型,會返回一個字符串,即類型名,具體的執行下面的代碼就知道了:
javascriptfunction foo() {} var a = 0; var b = '噓~蛋花湯在睡覺。'; var c = 1.0; var d = foo; var e = { "a" : a }; var f = [ 1, 2, 3 ]; var g = null; var h = undefined; console.log(typeof a); console.log(typeof b); console.log(typeof c); console.log(typeof d); console.log(typeof e); console.log(typeof f); console.log(typeof g); console.log(typeof h);
這裏的執行結果就將會是:
shellnumber string number function object object object undefined
在 JavaScript 中,有三個特殊的值,如標題所示。其中第一個你們可能都比較熟悉吧,C/C++ 裏面也有,不過是大寫的,其本質就是一個
#define NULL 0
而在 JavaScript 中,這三個值所表明的意義都不一樣。
null 是一種特殊的 object,大體的意思就是空。好比說:
var a = null;
你們都能看懂,就很少作解釋了。可是跟 C/C++ 不一樣的是,這個 null 跟 0 不相等。
這個東西的意思就是說這個變量未聲明。爲了可以更好地區分 null,咱們的樣例代碼以下寫:
javascriptvar a = { "foo" : null }; console.log(a["foo"]); console.log(a["bar"]);
上面的代碼中,咱們讓 a["foo"] 的值爲空,即 null。而壓根沒有聲明 a["bar"] 這個東西,它連空都不是。輸出的結果你們都差很少應該猜到了:
null undefined
這是一個空的數值,是一個特殊的 number。它的全稱是 Not a Number。有點奇怪,你們能夠理解爲 不是數字形態,或者數值出錯的 number 類型變量。
多在浮點型數值運算錯誤(如被0除)的狀況下出現,甚至能夠是用戶本身讓一個變量等於 NaN 以便返回一個錯誤值讓你們知道這個函數運算出錯了云云。
其它剩餘的語句也跟已存在的其它語言差很少,好比說 break 啊、switch 啊、continue 啊等等等等。
這一節主要講的是 JavaScript 對象,其它類型差很少一帶而過吧。
Node.js 包含的基礎類型差很少有以下幾個:
number string boolean array
其中前三種類型能夠直接賦值,而 array 的賦值只是一個引用賦值而已,在新變量中改變某個值的話舊變量的值也會改變,直接能夠試試下面的代碼:
javascriptvar foo = [ 1, 2, 3 ]; var bar = foo; bar[0] = 3; console.log(foo);
它得出的結果是:
javascript[ 3, 2, 3 ]
也就是說 array 要是複製出一個新的數組的話,不能用直接賦值的方法,而必須「深拷貝」。
這裏有必要講一下 array 的三種建立方法。
第一種:
javascriptvar dog = new Array(); dog[0] = "噓~"; dog[1] = "蛋花湯"; dog[2] = "在睡覺";
第二種:
javascriptvar dog = new Array( "噓~", "蛋花湯", "在睡覺" );
第三種:
javascriptvar dog = [ "噓~", "蛋花湯", "在睡覺" ];
我我的比較喜歡第三種寫法,比較簡潔。
這裏我把 JSON對象 單獨拎出來而不是把它歸類爲 JavaScript對象,若是以爲我有點誤人子弟就能夠直接跳過這一節了。
本人對於 JSON對象 和 JavaScript 對象的區分放在 是否只用來存儲數據,而並不是是一個類的實例化。其實 JSON 的本質即是 JavaScript Object Notation。
更多有關 JSON 的信息請自行百科。
在 Node.js 中聲明一個 JSON對象 很是簡單:
javascriptvar dog = { "pre" : "噓~", "sub" : { "name" : "蛋花湯", "act" : "在睡覺", "time" : 12 }, "suf" : [ "我說了", "它在睡覺", "就是在睡覺" ] };
有兩種方式能獲得 JSON對象 中的某個鍵名的鍵值,第一種是用點鏈接,第二種是用中括號:
javascriptdog.pre; dog["pre"];
注意:上面在用點的時候,後面直接跟的是JSON中的key,若是把key當成是變量去當問,只能用dog[key]
試試看:如今你本身動手試試看,用 for...in 的形式遍歷一遍上面的 JSON對象。別忘了用上 typeof 喵~
嚴格意義上來說,Node.js 的類不能算是類,其實它只是一個函數的集合體,加一些成員變量。它的本質實際上是一個函數。
不過爲了通俗地講,咱們接下去以及之後都將其稱爲「類」,實例化的叫「對象」。
由於類有着不少 函數 的特性,或者說它的本質就是一個 函數,因此這裏面咱們可能一不留神就順帶着把函數基礎給講了。
聲明一個類很是簡單,你們不要笑:
javascriptfunction foo() { //... }
好了,咱們已經寫好了一個 foo 類了。
真的假的?!真的。
不信?不信你能夠接下去打一段代碼看看:
javascriptvar bar = new foo();
別看它是一個函數,若是以這樣的形式(new)寫出來,它就是這個類的實例化。
而這個所謂的 foo() 其實就是這個 foo() 類的構造函數。
成員變量有好兩種方法。
第一種就是在類的構造函數或者任何構造函數中使用 this.<變量名> 。你能夠在任什麼時候候聲明一個成員變量,在外部不影響使用,反正就算在還未聲明的時候使用它,也會有一個 undefined 來撐着。因此說這就是第一種方法:
javascriptfunction foo() { this.hello = "world"; }
注意:只有在加了 this 的時候纔是調用類的成員變量,不然只是函數內的一個局部變量而已。要分清楚有沒有 this 的時候變量的做用範圍。
第二種方法就是在構造函數或者任何成員函數外部聲明,其格式是 <類名>.prototype.<變量名>:
javascriptfunction foo() { //... } foo.prototype.hello = "world";
不管上面哪一種方法都是對成員變量的聲明,咱們能夠看看效果:
javascriptvar bar = new foo(); console.log(bar.hello);
甚至你能夠這麼修改這個類:
javascriptfunction foo() { this.hello = "world"; } foo.prototype.hello = "蛋花湯";
而後再用上面的代碼輸出。
想一想看爲何輸出的仍是 world 而不是 蛋花湯。
咱們以前說過了這個 foo() 其實是一個 構造函數。那麼顯然咱們能夠給構造函數傳參數,因此就有了下面的代碼:
javascript// 代碼2.1 function foo(hello) { if(hello === undefined) { this.hello = "world"; } else { this.hello = hello; } }
咱們看到上面有一個奇葩的判斷 if(hello === undefined),這個判斷有什麼用呢?第一種可能,就是開發者很蛋疼地特地傳進去一個 undefined 進去,這個時候它是 undefined 無可厚非。
還有一種狀況。咱們一開始就說了 JavaScript 是一門弱類型語言,其實不只僅是弱類型,它的傳參數也很是不嚴謹。你能夠多傳或者少傳(只要保證你多傳或者少傳的時候能夠保證程序不出錯,或者邏輯不出錯),原則上都是能夠的。多傳的參數會被自動忽略,而少傳的參數會以 undefined 補足。
看看下面的代碼就明白了:
javascript// 上接代碼2.1 var bar1 = new foo(); var bar2 = new foo("蛋花湯");
請自行輸出一下兩個 bar 的 hello 變量,會發現一個是 world 一個是 蛋花湯。顯而易見,咱們的第一個 bar1 在聲明的時候,被 Node.js 自動當作了:
javascriptvar bar1 = new foo(undefined);
因此就有了它是 world 一說。
還有就是在這個構造函數中,咱們看到了傳進去的參數是 hello 而這個類中原本就有個成員變量就是 this.hello。不過咱們以前說過了有 this 和沒 this 的時候做用域不一樣,那個參數只是做用於構造函數中,而加了 this 的那個則是成員變量。用一個 this 就立刻區分開來他們了,因此即便同名也不要緊。
成員函數的聲明跟成員變量的第二種聲明方法差很少,即 <類名>.prototype.<函數名> = <函數>;
javascript// 上接代碼2.1 function setHello(hello) { this.hello = hello; } foo.prototype.setHello = setHello; bar1.setHello("雞蛋餅");
上面這段代碼顯而易見,咱們實現了 foo 類的 setHello 函數,能經過它修改 foo.hello 的值。
可是這麼寫是否是有點麻煩?接下去我要講一個 JavaScript 函數重要的特性了。
不少時候咱們的某些函數只在一個地方被引用或者調用,那麼咱們爲這個函數起一個名字就太不值了,不必,因此咱們能夠臨時寫好這個函數,直接讓引用它的人引用它,調用它的人調用它。因此函數能夠省略函數名,如:
javascriptfunction(hello) { this.hello = hello; }
至於怎麼引用或者調用呢?若是是上面的那個類須要引用的話,就是寫成這樣的:
javascriptfoo.prototype.setHello = function(hello) { this.hello = hello; }
這樣的寫法跟 成員函數 聲明 是一個效果的,並且省了不少的代碼量。並且實際上,基本上的類成員函數的聲明都是採用這種匿名函數的方式來聲明的。
至於說怎麼樣讓匿名函數被調用呢?這一般用於傳入一個只被某個函數調用的函數時這樣寫。
好比咱們有一個函數的原型是:
javascript/** * 咱們將傳入a,b兩個變量, * 在算出a+b的值後,交由func(num) * 去進行輸出 */ function sumab(a, b, func) { var c = a + b; func(a, b, c); }
好比咱們有兩個版本的輸出函數,一個是中文輸出,一個是英文輸出,那麼若是不用匿名函數時候是這麼寫的:
javascriptfunction zh(a, b, sum) { console.log(a + " + " + b + " 的值是:" + sum); } function en(a, b, sum) { console.log(a + " plus " + b + " is " + sum); } sumab(1, 2, zh); sumab(3, 4, en);
執行一遍這段代碼,輸出的結果將會是:
1 + 2 的值是:3 3 plus 4 is 7
這樣的代碼若是採用匿名函數的形式則將會是:
javascriptsumab(1, 2, function(a, b, sum) { console.log(a + " + " + b + " 的值是:" + sum); }); sumab(3, 4, function(a, b, sum) { console.log(a + " plus " + b + " is " + sum); });
這種形式一般使用於回調函數。回調機制算是 Node.js 或者說 JavaScript 的精髓。在之後的篇章會作介紹。
雖然上一節講過了,不過仍是再講一遍吧。
一般咱們聲明類的成員函數時候都是用匿名函數來聲明的,由於反正那個函數也就是這個類的一個成員函數而已,不會在其它地方被單獨引用或者調用,因此就有了下面的代碼:
javascript// 上接代碼2.1 foo.prototype.setHello = function(hello) { this.hello = hello; }
這樣咱們就使得 foo 類有了 setHello 這個函數了。
這個又是我胡扯的。所謂類的隨意性即 JavaScript 中你能夠在任何地方修改你的類,這跟 Ruby 有着必定的類似之處。
好比說 string ,它其實也是一個類,有着諸如 length 這樣的成員變量,也有 indexOf、substr 等成員函數。可是萬一咱們以爲這個 string 有些地方不完善,想加本身的方法,那麼能夠在你想要的地方給它增長一個函數,好比:
javascriptString.prototype.sb = function() { var newstr = ""; for(var i = 0; i < this.length; i++) { if(i % 2 === 0) newstr += "s"; else newstr += "b"; } return newstr; };
這個函數的意思就是填充一個字符串,使其變成 sb 的化身。
咱們來測試一下:
var str = "噓~蛋花湯在睡覺。"; console.log(str.sb());
你將會獲得這樣的結果:
sbsbsbsbs
你跟你的電腦說「噓~蛋花湯在睡覺。」,你的電腦會罵你四次半傻逼。(趕快砸了它)
所謂深拷貝就是本身新建一個數組或者對象,把源數組或者對象中的基礎類型變量值一個個手動拷過去,而不是隻把源數組或者對象的引用拿過來。因此這就涉及到了一個遞歸的調用什麼的。
下面是我實現的一個深拷貝函數,你們能夠寫一個本身的而後加入到本身的 Node.js 知識庫中。
javascriptfunction cloneObject(src) { var dest = {}; for(var key in src) { if(typeof src === "object") dest[key] = cloneObject(src[key]); else dest[key] = src[key]; } return dest; }
字符串:http://www.w3school.com.cn/js/jsref_obj_string.asp
數字: http://www.w3school.com.cn/js/jsref_obj_number.asp
數組: http://www.w3school.com.cn/js/jsref_obj_array.asp
布爾: http://www.w3school.com.cn/js/jsref_obj_boolean.asp
日期: http://www.w3school.com.cn/js/jsref_obj_date.asp
數學庫:http://www.w3school.com.cn/js/jsref_obj_math.asp