网络编程TCP
TCP要点
- TCP服务器一般情况下都需要绑定,否则客户端找不到这个服务器。
- TCP客户端一般不绑定,因为是主动链接服务器,所以只要确定好服务器的ip、port等信息就好,本地客户端可以随机。
- TCP服务器中通过listen可以将socket创建出来的主动套接字变为被动的,这是做TCP服务器时必须要做的。
- 当客户端需要链接服务器时,就需要使用connect进行链接,UDP是不需要链接的而是直接发送,但是TCP必须先链接,只有链接成功才能通信。
- 当一个TCP客户端连接服务器时,服务器端会有1个新的套接字,这个套接字用来标记这个客户端,单独为这个客户端服务。
- listen后的套接字是被动套接字,用来接收新的客户端的链接请求的,而accept返回的新套接字是标记这个新客户端的。
- 关闭listen后的套接字意味着被动套接字关闭了,会导致新的客户端不能够链接服务器,但是之前已经链接成功的客户端正常通信。
- 关闭accept返回的套接字意味着这个客户端已经服务完毕。
- 当客户端的套接字调用close后,服务器端会recv解堵塞,并且返回的长度为0,因此服务器可以通过返回数据的长度来区别客户端是否已经下线。
客户端
需要服务的一方,例如我们用浏览器上个,浏览器帮助我们向远程的一台电脑发送请求接收数据,以及展示效果等,此时浏览器就是客户端。
当收到数据后需要给对方发送确认消息,但那时操作系统做的事情,编写的代码不需要确认,只管收发数据即可。
- 客户端发送请求:
import socket
# 1初始化套接字
tcp_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 2建立链接 要传入链接的服务器ip和port
tcp_socket.connect(('xxx.xxx.xxx.x', xxxxx))
while True:
text = input('发送的数据')
# 3发数据
tcp_socket.send(text.encode('gbk'))
# 4接收数据
data = tcp_socket.recv(1024)
print(data.decode('gbk'))
# 5断开
tcp_socket.close()
逻辑顺序:
- 创建一个TCP套接字(套接字不区分客户端、服务器)
- 链接服务器
- 向服务器发送数据、接收来自服务器的响应数据
- 断开链接
服务器
为其他的TCP程序提供服务的程序,称之为TCP服务器。
- 接收多次数据
import socket
# 1 初始化套接字
tcp_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 设置地址可以复用,解决了 端口没有及时释放地问题
tcp_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, True)
# 2 服务器半丁ip和port
tcp_socket.bind(('xxx.xxx.xxx.x', xxxxx))
# 3 设置为被动监听模式 最大并发接收的数量是 128
tcp_socket.listen(128)
while True:
# 4 等待接收链接请求 接收到一个元组(客户端的socket对象,(客户端的地址, 端口))
client, addr = tcp_socket.accept()
print('client:', client)
print('addr:', addr)
while True:
data = client.recv(1024)
print(data.decode('gbk'))
if not data:
# 如果数据为空 说明客户端关闭了链接 这里的client也就可以关闭
client.close()
break
# 5 断开
tcp_socket.close()
逻辑顺序:
- 子主题 1socket创建一个套接字
- bind绑定ip和port
- listen使套接字变为被动链接
- accept等待客户端的链接
- recv/send接收发送数据
- close关闭套接字
案例: 模拟在线客户
import socket
# 1. 创建TCP套接字
server_s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 2. 绑定本地信息
server_s.bind(("", 7788))
# 3. 设置为被动的
server_s.listen(128)
while True:
print("等待新的顾客到来...")
# 4. 等待客户端链接
new_s, client_info = server_s.accept()
print("一个新的顾客链接成功,ta是:%s" % str(client_info))
# 5. 用新的套接字为已经连接好的客户端服务器
while True:
recv_content = new_s.recv(1024)
print("%s>>>%s" % (str(client_info), recv_content.decode("gbk")))
if not recv_content:
# 当客户端调用了close后,recv返回值为空,此时服务套接字就可以close了
# 6. 关闭服务套接字
new_s.close()
break
print("顾客:%s已离开\n" % str(client_info))
# 7. 关闭监听套接字
server_s.close()
- 当一个客户端没有断开链接之前,新的客户端可以链接成功,但是服务器不能接收到数据。
- 此程序是一对一服务,不能同时为多个顾客同时服务。
- accept的新的返回值代表着刚刚链接成功的那个客户端。