模塊概覽
net模塊是同樣是nodejs的核心模塊。在http模塊概覽里提到,http.Server繼承了net.Server,此外,http客戶端與http服務(wù)端的通信均依賴于socket(net.Socket)。也就是說(shuō),做node服務(wù)端編程,net基本是繞不開(kāi)的一個(gè)模塊。
從組成來(lái)看,net模塊主要包含兩部分,了解socket編程的同學(xué)應(yīng)該比較熟悉了:
本文從一個(gè)簡(jiǎn)單的 tcp服務(wù)端/客戶端 的例子開(kāi)始講解,好讓讀者有個(gè)概要的認(rèn)識(shí)。接著再分別介紹 net.Server、net.Socket 比較重要的API、屬性、事件。
對(duì)于初學(xué)者,建議把文中的例子本地跑一遍加深理解。
簡(jiǎn)單的 server+client 例子
tcp服務(wù)端程序如下:
var net = require('net');var PORT = 3000;var HOST = '127.0.0.1';// tcp服務(wù)端var server = net.createServer(function(socket){ console.log('服務(wù)端:收到來(lái)自客戶端的請(qǐng)求'); socket.on('data', function(data){ console.log('服務(wù)端:收到客戶端數(shù)據(jù),內(nèi)容為{'+ data +'}'); // 給客戶端返回?cái)?shù)據(jù) socket.write('你好,我是服務(wù)端'); }); socket.on('close', function(){ console.log('服務(wù)端:客戶端連接斷開(kāi)'); });});server.listen(PORT, HOST, function(){ console.log('服務(wù)端:開(kāi)始監(jiān)聽(tīng)來(lái)自客戶端的請(qǐng)求');});tcp客戶端如下:
var net = require('net');var PORT = 3000;var HOST = '127.0.0.1';// tcp客戶端var client = net.createConnection(PORT, HOST);client.on('connect', function(){ console.log('客戶端:已經(jīng)與服務(wù)端建立連接');});client.on('data', function(data){ console.log('客戶端:收到服務(wù)端數(shù)據(jù),內(nèi)容為{'+ data +'}');});client.on('close', function(data){ console.log('客戶端:連接斷開(kāi)');});client.end('你好,我是客戶端');運(yùn)行服務(wù)端、客戶端代碼,控制臺(tái)分別輸出如下:
服務(wù)端:
服務(wù)端:開(kāi)始監(jiān)聽(tīng)來(lái)自客戶端的請(qǐng)求
服務(wù)端:收到來(lái)自客戶端的請(qǐng)求
服務(wù)端:收到客戶端數(shù)據(jù),內(nèi)容為{你好,我是客戶端}
服務(wù)端:客戶端連接斷開(kāi)
客戶端:
客戶端:已經(jīng)與服務(wù)端建立連接
客戶端:收到服務(wù)端數(shù)據(jù),內(nèi)容為{你好,我是服務(wù)端}
客戶端:連接斷開(kāi)
服務(wù)端 net.Server
server.address()
返回服務(wù)端的地址信息,比如綁定的ip地址、端口等。
console.log( server.address() );// 輸出如下 { port: 3000, family: 'IPv4', address: '127.0.0.1' }server.close(callback])
關(guān)閉服務(wù)器,停止接收新的客戶端請(qǐng)求。有幾點(diǎn)注意事項(xiàng):
下面會(huì)通過(guò)兩個(gè)具體的例子進(jìn)行對(duì)比,先把結(jié)論列出來(lái)
例子1:服務(wù)端正常關(guān)閉
var net = require('net');var PORT = 3000;var HOST = '127.0.0.1';var noop = function(){};// tcp服務(wù)端var server = net.createServer(noop);server.listen(PORT, HOST, function(){ server.close(function(error){ if(error){ console.log( 'close回調(diào):服務(wù)端異常:' + error.message ); }else{ console.log( 'close回調(diào):服務(wù)端正常關(guān)閉' ); } }); });server.on('close', function(){ console.log( 'close事件:服務(wù)端關(guān)閉' );});server.on('error', function(error){ console.log( 'error事件:服務(wù)端異常:' + error.message );});輸出為:
close事件:服務(wù)端關(guān)閉
close回調(diào):服務(wù)端正常關(guān)閉
例子2:服務(wù)端異常關(guān)閉
代碼如下
var net = require('net');var PORT = 3000;var HOST = '127.0.0.1';var noop = function(){};// tcp服務(wù)端var server = net.createServer(noop);// 沒(méi)有正式啟動(dòng)請(qǐng)求監(jiān)聽(tīng)// server.listen(PORT, HOST);server.on('close', function(){ console.log( 'close事件:服務(wù)端關(guān)閉' );});server.on('error', function(error){ console.log( 'error事件:服務(wù)端異常:' + error.message );});server.close(function(error){ if(error){ console.log( 'close回調(diào):服務(wù)端異常:' + error.message ); }else{ console.log( 'close回調(diào):服務(wù)端正常關(guān)閉' ); } });輸出為:
close事件:服務(wù)端關(guān)閉
close回調(diào):服務(wù)端異常:Not running
server.ref()/server.unref()
了解node事件循環(huán)的同學(xué)對(duì)這兩個(gè)API應(yīng)該不陌生,主要用于將server 加入事件循環(huán)/從事件循環(huán)里面剔除,影響就在于會(huì)不會(huì)影響進(jìn)程的退出。
對(duì)出學(xué)習(xí)net的同學(xué)來(lái)說(shuō),并不需要特別關(guān)注,感興趣的自己做下實(shí)驗(yàn)就好。
事件 listening/connection/close/error
幾個(gè)事件都比較簡(jiǎn)單,這里僅舉個(gè) connection 的例子。
從測(cè)試結(jié)果可以看出,有新的客戶端連接產(chǎn)生時(shí),net.createServer(callback) 中的callback回調(diào) 會(huì)被調(diào)用,同時(shí) connection 事件注冊(cè)的回調(diào)函數(shù)也會(huì)被調(diào)用。
事實(shí)上,net.createServer(callback) 中的 callback 在node內(nèi)部實(shí)現(xiàn)中 也是加入了做為 connection事件 的監(jiān)聽(tīng)函數(shù)。感興趣的可以看下node的源碼。
var net = require('net');var PORT = 3000;var HOST = '127.0.0.1';var noop = function(){};// tcp服務(wù)端var server = net.createServer(function(socket){ socket.write('1. connection 觸發(fā)/n');});server.on('connection', function(socket){ socket.end('2. connection 觸發(fā)/n');});server.listen(PORT, HOST);通過(guò)下面命令測(cè)試下效果
curl http://127.0.0.1:3000
輸出:
1. connection 觸發(fā)
2. connection 觸發(fā)
客戶端 net.Socket
在文章開(kāi)頭已經(jīng)舉過(guò)客戶端的例子,這里再把例子貼一下。(備注:嚴(yán)格來(lái)說(shuō)不應(yīng)該把 net.Socket 叫做客戶端,這里方便講解而已)
單從node官方文檔來(lái)看的話,感覺(jué) net.Socket 比 net.Server 要復(fù)雜很多,有更多的API、事件、屬性。但實(shí)際上,把 net.Socket 相關(guān)的API、事件、屬性 進(jìn)行歸類(lèi)下,會(huì)發(fā)現(xiàn),其實(shí)也不是特別復(fù)雜。
具體請(qǐng)看下一小節(jié)內(nèi)容。
var net = require('net');var PORT = 3000;var HOST = '127.0.0.1';// tcp客戶端var client = net.createConnection(PORT, HOST);client.on('connect', function(){ console.log('客戶端:已經(jīng)與服務(wù)端建立連接');});client.on('data', function(data){ console.log('客戶端:收到服務(wù)端數(shù)據(jù),內(nèi)容為{'+ data +'}');});client.on('close', function(data){ console.log('客戶端:連接斷開(kāi)');});client.end('你好,我是客戶端');API、屬性歸類(lèi)
以下對(duì)net.Socket的API跟屬性,按照用途進(jìn)行了大致的分類(lèi),方便讀者更好的理解。大部分API跟屬性都比較簡(jiǎn)單,看下文檔就知道做什么的,這里就先不展開(kāi)。
連接相關(guān)
數(shù)據(jù)讀、寫(xiě)相關(guān)
socket.write()、socket.end()、socket.pause()、socket.resume()、socket.setEncoding()、socket.setNoDelay()
數(shù)據(jù)屬性相關(guān)
socket.bufferSize、socket.bytesRead、socket.bytesWritten
事件循環(huán)相關(guān)
socket.ref()、socket.unref()
地址相關(guān)
事件簡(jiǎn)介
相關(guān)鏈接
官方文檔:https://nodejs.org/api/net.html#net_socket_destroy_exception
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持武林網(wǎng)。
新聞熱點(diǎn)
疑難解答
圖片精選
網(wǎng)友關(guān)注