Net

最后更新于:2022-04-01 01:58:45

### 稳定度: 2 - 稳定 `net`模块为你提供了异步的网络调用的包装。它同时包含了创建服务器和客户端的函数。你可以通过`require('net')`来引入这个模块。 #### net.createServer([options][, connectionListener]) 创建一个新的TCP服务器。`connectionListener`参数会被自动绑定为`connection`事件的监听器。 `options`是一个包含下列默认值的对象: ~~~ { allowHalfOpen: false, pauseOnConnect: false } ~~~ 如果`allowHalfOpen`是`true`,那么当另一端的`socket`发送一个`FIN`报文时`socket`并不会自动发送`FIN`报文。`socket`变得不可读,但是可写。你需要明确地调用`end()`方法。详见`end`事件。 如果`pauseOnConnect`是true,那么`socket`在每一次被连接时会暂停,并且不会读取数据。这允许在进程间被传递的连接不读取任何数据。如果要让一个被暂停的`socket`开始读取数据,调用`resume()`方法。 以下是一个应答服务器的例子,监听8124端口: ~~~ var net = require('net'); var server = net.createServer(function(c) { //'connection' listener console.log('client connected'); c.on('end', function() { console.log('client disconnected'); }); c.write('hello\r\n'); c.pipe(c); }); server.listen(8124, function() { //'listening' listener console.log('server bound'); }); ~~~ 使用`telnet`测试: ~~~ telnet localhost 8124 ~~~ 想要监听`socket``/tmp/echo.sock`,只需改变倒数第三行: ~~~ server.listen('/tmp/echo.sock', function() { //'listening' listener ~~~ 使用`nc`连接一个UNIX domain socket服务器: ~~~ nc -U /tmp/echo.sock ~~~ #### net.connect(options[, connectionListener]) #### net.createConnection(options[, connectionListener]) 工厂函数,返回一个新的`net.Socket`实例,并且自动使用提供的`options`进行连接。 `options`会被同时传递给`net.Socket`构造函数和`socket.connect`方法。 参数`connectListener`将会被立即添加为`connect`事件的监听器。 下面是一个上文应答服务器的客户端的例子: ~~~ var net = require('net'); var client = net.connect({port: 8124}, function() { //'connect' listener console.log('connected to server!'); client.write('world!\r\n'); }); client.on('data', function(data) { console.log(data.toString()); client.end(); }); client.on('end', function() { console.log('disconnected from server'); }); ~~~ 要连接`socket``/tmp/echo.sock`只需要改变第二行为: ~~~ var client = net.connect({path: '/tmp/echo.sock'}); ~~~ #### net.connect(port[, host][, connectListener]) #### net.createConnection(port[, host][, connectListener]) 工厂函数,返回一个新的`net.Socket`实例,并且自动使用指定的端口(port)和主机(host)进行连接。 如果`host`被省略,默认为`localhost`。 参数`connectListener`将会被立即添加为`connect`事件的监听器。 #### net.connect(path[, connectListener]) #### net.createConnection(path[, connectListener]) 工厂函数,返回一个新的unix`net.Socket`实例,并且自动使用提供的路径(path)进行连接。 参数`connectListener`将会被立即添加为`connect`事件的监听器。 #### Class: net.Server 这个类用于创建一个TCP或本地服务器。 #### server.listen(port[, hostname][, backlog][, callback]) 开始从指定端口和主机名接收连接。如果省略主机名,那么如果IPv6可用,服务器会接受从任何IPv6地址(::)来的链接,否则为任何IPv4地址(0.0.0.0)。如果端口为0那么将会为其设置一个随机端口。 积压量`backlog`是连接等待队列的最大长度。实际长度由你的操作系统的`sysctl`设置决定(如linux中的`tcp_max_syn_backlog`和`somaxconn`)。这个参数的默认值是511(不是512)。 这个函数式异步的。当服务器绑定了指定端口后,`listening`事件将会被触发。最后一个参数`callback`将会被添加为`listening`事件的监听器。 有些用户可能遇到的情况是收到`EADDRINUSE`错误。这意味着另一个服务器已经使用了该端口。一个解决的办法是等待一段时间后重试。 ~~~ server.on('error', function (e) { if (e.code == 'EADDRINUSE') { console.log('Address in use, retrying...'); setTimeout(function () { server.close(); server.listen(PORT, HOST); }, 1000); } }); ~~~ (注意,`io.js`中所有的`socket`都已经设置了`SO_REUSEADDR`) #### server.listen(path[, callback]) - path String - callback Function 启动一个本地`socket`服务器,监听指定路径(`path`)上的连接。 这个函数式异步的。当服务器监听了指定路径后,`listening`事件将会被触发。最后一个参数`callback`将会被添加为`listening`事件的监听器。 在UNIX中,`local domain`经常被称作`UNIX domain`。`path`是一个文件系统路径名。它在被创建时会受相同文件名约定(same naming conventions)的限制并且进行权限检查(permissions checks)。它在文件系统中可见,并且在被删除前持续存在。 在Windows中,`local doamin`使用一个命名管道(named pipe)实现。`path`必须指向`\\?\pipe\`或`\\.\pipe\.`中的一个条目,但是后者可能会做一些命名管道的处理,如处理`..`序列。除去表现,命名管道空间是平坦的(flat)。管道不会持续存在,它们将在最后一个它们的引用关闭后被删除。不要忘记,由于`JavaScript`的字符串转义,你必须在指定`path`时使用双反斜杠: ~~~ net.createServer().listen( path.join('\\\\?\\pipe', process.cwd(), 'myctl')) ~~~ #### server.listen(handle[, callback]) - handle Object - callback Function `handle`对象可以被设置为一个服务器或一个`socket`(或者任意以下划线开头的成员`_handle`),或者一个`{fd: <n>}`对象。 这将使得服务器使用指定句柄接受连接,但它假设文件描述符或句柄已经被绑定至指定的端口或域名`socket`。 在Windows下不支持监听一个文件描述符。 这个函数式异步的。当服务器已被绑定后,`listening`事件将会被触发。最后一个参数`callback`将会被添加为`listening`事件的监听器。 #### server.listen(options[, callback]) - **options Object** - port Number 可选 - host String 可选 - backlog Number 可选 - path String 可选 - exclusive Boolean 可选 - callback Function 可选 `port`,`host`和`backlog`属性,以及可选的`callback`函数,与`server.listen(port, [host], [backlog], [callback])`中表现一致。`path`可以被指定为一个UNIX `socket`。 如果`exclusive`是`false`(默认),那么工作集群(cluster workers)将会使用相同的底层句柄,处理的连接的职责将会被它们共享。如果`exclusive`是`true`,那么句柄是不被共享的,企图共享将得到一个报错的结果。下面是一个监听独有端口的例子: ~~~ server.listen({ host: 'localhost', port: 80, exclusive: true }); ~~~ #### server.close([callback]) 使服务器停止接收新的连接并且保持已存在的连接。这个函数式异步的,当所有的连接都结束时服务器会最终关闭,并处罚一个`close`事件。可选的,你可以传递一个回调函数来监听`close`事件。如果传递了,那么它的唯一的第一个参数将表示任何可能潜在发生的错误。 #### server.address() 返回服务器绑定的地址,协议族名和端口通过操作系统报告。对查找操作系统分配的地址哪个端口被分配非常有用。返回一个有三个属性的对象。如`{ port: 12346, family: 'IPv4', address: '127.0.0.1' }`。 例子: ~~~ var server = net.createServer(function (socket) { socket.end("goodbye\n"); }); // grab a random port. server.listen(function() { address = server.address(); console.log("opened server on %j", address); }); ~~~ 在`listening`事件触发前,不要调用`server.address()`方法。 #### server.unref() 调用一个`server`对象的`unref`方法将允许如果它是事件系统中唯一活跃的服务器,程序将会退出。如果服务器已经被调用过这个方法,那么再次调用这个方法将不会有任何效果。 返回`server`对象。 #### server.ref() 与`unref`相反,在一个已经被调用`unref`方法的`server`中调用`ref`方法,那么如果它是唯一活跃的服务器时,程序将不会退出(默认)。如果服务器已经被调用过这个方法,那么再次调用这个方法将不会有任何效果。 返回`server`对象。 #### server.maxConnections 设置了这个属性后,服务器的连接数达到时将会开始拒绝连接。 一旦`socket`被使用`child_process.fork()`传递给了子进程,这个属性就不被推荐去设置。 #### server.connections 这个函数已经被弃用。请使用`server.getConnections()`替代。 服务器的当前连接数。 当使用`child_process.fork()`传递一个`socket`给子进程时,这个属性将变成`null`。想要得到正确的结果请使用`server.getConnections`。 #### server.getConnections(callback) 异步地去获取服务器的当前连接数,在`socket`被传递给子进程时仍然可用。 回调函数的两个参数是`err`和`count`。 ### `net.Server`是一个具有以下事件的`EventEmitter`: #### Event: 'listening' 当调用`server.listen`后,服务器已被绑定时触发。 #### Event: 'connection' - Socket object 连接对象 当新的连接产生时触发。`socket`是一个`net.Socket`实例。 #### Event: 'close' 当服务器关闭时触发。注意如果服务器中仍有连接存在,那么这个事件会直到所有的连接都关闭后才触发。 #### Event: 'error' - Error Object 当发生错误时触发。`close`事件将会在它之后立即触发。参阅`server.listen`。 #### Class: net.Socket 这个对象是一个TCP或本地`socket`的抽象。`net.Socket`实例实现了双工流(duplex Stream)接口。它可以被使用者创建,并且被作为客户端(配合`connect()`)使用。或者也可以被`io.js`创建,并且通过服务器的`connection`事件传递给使用者。 #### new net.Socket([options]) 创建一个新的`socket`对象。 `options`是一个有以下默认值的对象: ~~~ { fd: null allowHalfOpen: false, readable: false, writable: false } ~~~ `fd`允许你使用一个指定的已存在的`socket`文件描述符。设置`readable` 和/或 `writable`为`true`将允许从这个`socket`中读 和/或 写(注意,仅在传递了`passed`时可用)。关于`allowHalfOpen`,参阅`createServer()`和`end`事件。 #### socket.connect(options[, connectListener]) 从给定的`socket`打开一个连接。 对于TCP`socket`,`options`参数需是一个包含以下属性的对象: - port: 客户端需要连接的端口(必选)。 - host: 客户端需要连接的主机(默认:'localhost') - localAddress: 将要绑定的本地接口,为了网络连接。 - localPort: 将要绑定的本地端口,为了网络连接。 - family : IP协议族版本,默认为`4`。 - lookup : 自定义查找函数。默认为`dns.lookup`。 对于本地domain `socket`,`options`参数需是一个包含以下属性的对象: - path: 客户端需要连接的路径(必选)。 通常这个方法是不需要的,因为通过`net.createConnection`打开`socket`。只有在你自定义了`socket`时才使用它。 这个函数式异步的,当`connect`事件触发时,这个`socket`就被建立了。如果在连接的过程有问题,那么`connect`事件将不会触发,`error`将会带着这个异常触发。 `connectListener`参数会被自动添加为`connect`事件的监听器。 #### socket.connect(port[, host][, connectListener]) #### socket.connect(path[, connectListener]) 参阅`socket.connect(options[, connectListener])`。 #### socket.bufferSize `net.Socket`的属性,用于`socket.write()`。它可以帮助用户获取更快的运行速度。计算机不能一直保持大量数据被写入`socket`的状态,网络连接可以很慢。`io.js`在内部会排队等候数据被写入`socekt`并确保传输连接上的数据完好。 (内部实现为:轮询`socekt`的文件描述符等待它为可写)。 内部缓存的可能结果是内存使用会增长。这个属性展示了缓存中还有多少待写入的字符(字符的数目约等于要被写入的字节数,但是缓冲区可能包含字符串,而字符串是惰性编码的,所以确切的字节数是未知的)。 遇到数值很大或增长很快的`bufferSize`时,应当尝试使用`pause()`和`resume()`来控制。 #### socket.setEncoding([encoding]) 设置`socket`的编码作为一个可读流。详情参阅`stream.setEncoding()`。 #### socket.write(data[, encoding][, callback]) 在套接字上发送数据。第二个参数指定了字符串的编码,默认为UTF8。 如果所有数据成功被刷新至了内核缓冲区,则返回`true`。如果所有或部分数据仍然在用户内存中排队,则返回`false`。`drain`事件将会被触发当`buffer`再次为空时。 当数据最终被写入时,`callback`回调函数将会被执行,但可能不会马上执行。 #### socket.end([data][, encoding]) 半关闭一个`socket`。比如,它发送一个`FIN`报文。可能服务器仍然在发送一些数据。 如果`data`参数被指定,那么等同于先调用`socket.write(data, encoding)`,再调用`socket.end()`。 #### socket.destroy() 确保这个`socket`上没有I/O活动发生。只在发生错误情况才需要(如处理错误)。 #### socket.pause() 暂停数据读取。`data`事件将不会再触发。对于控制上传非常有用。 #### socket.resume() 用于在调用`pause()`后,恢复数据读取。 #### socket.setTimeout(timeout[, callback]) 如果`socket`在`timeout`毫秒中没有活动后,设置其为超时。默认情况下,`net.Socket`没有超时。 当超时发生,`socket`会收到一个`timeout`事件,但是连接将不会被断开。用户必须手动地调用`end()`或`destroy()`方法。 如果`timeout`是`0`,那么现有的超时将会被禁用。 可选的`callback`参数就会被自动添加为`timeout`事件的监听器。 返回一个`socket`。 #### socket.setNoDelay([noDelay]) 警用纳格算法(Nagle algorithm)。默认情况下TCP连接使用纳格算法,它们的数据在被发送前会被缓存。设置`noDelay`为`true`将会在每次`socket.write()`时立刻发送数据。`noDelay`默认为`true`。 返回一个`socket`。 #### socket.setKeepAlive([enable][, initialDelay]) 启用/警用长连接功能,并且在第一个在闲置`socket`的长连接`probe`被发送前,可选得设置初始延时。`enable`默认为`false`。 设定`initialDelay`(毫秒),来设定在收到的最后一个数据包和第一个长连接`probe`之间的延时。将`initialDelay`设成`0`会让值保持不变(默认值或之前所设的值)。默认为`0`。 返回一个`socket`。 #### socket.address() 返回绑定的地址,协议族名和端口通过操作系统报告。对查找操作系统分配的地址哪个端口被分配非常有用。返回一个有三个属性的对象。如`{ port: 12346, family: 'IPv4', address: '127.0.0.1' }`。 #### socket.unref() 调用一个`socket`对象的`unref`方法将允许如果它是事件系统中唯一活跃的`socket`,程序将会退出。如果`socket`已经被调用过这个方法,那么再次调用这个方法将不会有任何效果。 返回`socket`对象。 #### socket.ref() 与`unref`相反,在一个已经被调用`unref`方法的`socket`中调用`ref`方法,那么如果它是唯一活跃的`socket`时,程序将不会退出(默认)。如果`socket`已经被调用过这个方法,那么再次调用这个方法将不会有任何效果。 返回`socket`对象。 #### socket.remoteAddress 远程IP地址字符串。例如,`'74.125.127.100'`或`'2001:4860:a005::68'`。 #### socket.remoteFamily 远程IP协议族字符串。例如,`'IPv4'`或`'IPv6'`。 #### socket.remotePort 远程端口数值。例如,`80`或`21`。 #### socket.localAddress 远程客户端正连接的本地IP地址字符串。例如,如果你正在监听`'0.0.0.0'`并且客户端连接在`'192.168.1.1'`,其值将为`'192.168.1.1'`。 #### socket.localPort 本地端口数值。例如,`80`或`21`。 #### socket.bytesRead 接受的字节数。 #### socket.bytesWritten 发送的字节数。 ### net.Socket `net.Socket`实例是一个包含以下事件的`EventEmitter`: #### Event: 'lookup' 在解析主机名后,连接主机前触发。对UNIX `socket`不适用。 - err {Error | Null} 错误对象,参阅 `dns.lookup()` - address {String} IP地址 - family {String | Null} 地址类型。参阅'dns.lookup()` #### Event: 'connect' 在`socket`连接成功建立后触发。参阅`connect()`。 #### Event: 'data' - Buffer object 在接受到数据后触发。参数将会是一个`Buffer`或一个字符串。数据的编码由`socket.setEncoding()`设置(更多详细信息请查看可读流章节)。 注意,当`socket`触发`data`事件时,如果没有监听器存在。那么数据将会丢失。 #### Event: 'end' 当另一端的`socket`发送一个`FIN`报文时触发。 默认情况(`allowHalfOpen == false`)下,一旦一个`socket`的文件描述符被从它的等待写队列(pending write queue)中写出,`socket`会销毁它。但是,当设定`allowHalfOpen == true`后,`socket`不会在它这边自动调用`end()`,允许用户写入任意数量的数据,需要注意的是用户需要在自己这边调用`end()`。 #### Event: 'timeout' 当`socket`因不活动而超时时触发。这只是来表示`socket`被限制。用户必须手动关闭连接。 参阅`socket.setTimeout()`。 #### Event: 'drain' 当写缓冲为空时触发。可以被用来控制上传流量。 参阅`socket.write()`的返回值。 #### Event: 'error' - Error object 当发生错误时触发。`close`事件会紧跟着这个事件触发。 #### Event: 'close' - had_error 如果`socket`有一个传输错误时为`true` 当`socket`完全关闭时触发。参数`had_error`是一个表示`socket`是否是因为传输错误而关闭的布尔值。 #### net.isIP(input) 测试`input`是否是一个IP地址。如果是不合法字符串时,会返回`0`。如果是IPv4地址则返回`4`,是IPv6地址则返回`6`。 #### net.isIPv4(input) 如果`input`是一个IPv4地址则返回`true`,否则返回`false`。 #### net.isIPv6(input) 如果`input`是一个IPv6地址则返回`true`,否则返回`false`。
';