创建TCP服务器
1. 伪代码
s = socket.socket() # 创建服务器套接字
s.bind() # 套接字与地址绑定
s.listen() # 监听连接
inf_loop: # 服务器无限循环
cs = s.accept() # 接受客户端连接
comm_loop: # 通信循环
cs.recv()/cs.send() # 对话(接收/发送)
cs.close() # 关闭客户端套接字
s.close() # 关闭服务器套接字#(可选)
所有套接字都是通过使用 socket.socket()函数来创建的。因为服务器需要占用一个端口并等待客户端的请求,所以它们必须绑定到一个本地地址。因为 TCP 是一种面向连接的通信系 统,所以在 TCP 服务器开始操作之前,必须安装一些基础设施。特别地,TCP 服务器必须监 听(传入)的连接。一旦这个安装过程完成后,服务器就可以开始它的无限循环。
调用 accept()函数之后,就开启了一个简单的(单线程)服务器,它会等待客户端的连接。 默认情况下,accept()是阻塞的,这意味着执行将被暂停,直到一个连接到达。另外,套接字 确实也支持非阻塞模式,可以参考文档或操作系统教材,以了解有关为什么以及如何使用非阻塞套接字的更多细节。
一旦服务器接受了一个连接,就会返回(利用 accept())一个独立的客户端套接字,用来与即将到来的消息进行交换。使用新的客户端套接字类似于将客户的电话切换给客服代表。 当一个客户电话最后接进来时,主要的总机接线员会接到这个电话,并使用另一条线路将这个电话转接给合适的人来处理客户的需求。
这将能够空出主线(原始服务器套接字),以便接线员可以继续等待新的电话(客户请求), 而此时客户及其连接的客服代表能够进行他们自己的谈话。同样地,当一个传入的请求到达 时,服务器会创建一个新的通信端口来直接与客户端进行通信,再次空出主要的端口,以使 其能够接受新的客户端连接。
一旦创建了临时套接字,通信就可以开始,通过使用这个新的套接字,客户端与服务器 就可以开始参与发送和接收的对话中,直到连接终止。当一方关闭连接或者向对方发送一个 空字符串时,通常就会关闭连接。
在代码中,一个客户端连接关闭之后,服务器就会等待另一个客户端连接。最后一行代 码是可选的,在这里关闭了服务器套接字。其实,这种情况永远也不会碰到,因为服务器应该在一个无限循环中运行。
2. 创建TCP服务器
import socket
from time import strftime, localtime
HOST = '192.168.0.104'
PORT = 8080
BUFSIZE = 1024
ADDR = (HOST, PORT)
tcp_s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
tcp_s.bind(ADDR)
tcp_s.listen(5)
while True:
print("waiting for connection...")
tcp_c, addr = tcp_s.accept()
print("...connect from: ", addr)
while True:
data = tcp_c.recv(BUFSIZE)
if not data:
break
tcp_c.send(
str(
'[%s] %s\n' % (strftime("%Y-%m-%d %H:%M:%S", localtime()), str(data))
).encode('gbk'))
tcp_c.close()
tcp_s.close()
代码解释
1-11: TCP服务器一般需要有固定的端口信息,以便于客户端的访问,因此这里要提前设定好套接字信息并与服务器进行绑定(bind),然后通过listen来监听端口,检查是否有请求访问。
13-28 首先设置一个等待连接的循环,一旦有客户端连接,就用一个新的套接字对象来和这个用户进行通讯,而使监听套接字空闲可以继续监听请求连接,第二个循环用于处理和客户端的通讯环节,当客户端不在发送信息的时候就关闭这个服务套接字。28行虽然是关闭监听套接字,但是他理论上是不会执行了,因为外循环不会中断,这也符合服务器应该一直运行等待访问的实施。
3. 运行TCP客户端和TCP服务器
下面,我们开启这个服务器,然后用自己编写的TCP客户端访问它,看看能否连接成功并且正确的传输消息。
首先用TCP客户端,发起一系列问候,可以看到服务器都将我们的问候打上了时间戳并进行了返回。
这里是服务器监听的状态,表明已经和192.168.0.104
主机建立了连接