Python-套接字编程


标准库中的套接字模块包含服务器和客户端之间在硬件级别进行通信所需的功能。

该模块提供对 BSD 套接字接口的访问。它适用于所有操作系统,例如 Linux、Windows、MacOS。

什么是套接字?

套接字是双向通信通道的端点。套接字可以在进程内、同一台机器上的进程之间或不同大陆的进程之间进行通信。

套接字由 IP 地址和端口号的组合来标识。两端都应正确配置才能开始通信。

联系 IP地址

套接字可以通过多种不同的通道类型来实现:Unix 域套接字、TCP、UDP 等。套接字库提供了用于处理公共传输的特定类以及用于处理其余传输的通用接口。

术语“套接字编程”意味着以编程方式设置套接字以能够发送和接收数据。

有两种类型的通信协议 -

  • 面向连接的协议

  • 无连接协议

TCP 或传输控制协议是一种面向连接的协议。数据由服务器以数据包的形式传输,并由接收器按照相同的传输顺序进行组装。由于通信两端的套接字需要在开始前设置,因此这种方法更加可靠。

UDP 或用户数据报协议是无连接的。该方法并不可靠,因为套接字不需要建立任何连接和终止过程来传输数据。

Python 套接字模块

该模块包括 Socket 类。套接字对象代表主机名和端口号的 psir。构造函数方法具有以下签名 -

句法

socket.socket (socket_family, socket_type, protocol=0)

参数

  • family - 默认情况下为 AF_INET。其他值 - AF_INET6(八组四个十六进制数字)、AF_UNIX、AF_CAN(控制器局域网)或 AF_RDS(可靠数据报套接字)。

  • socket_type - 应该是 SOCK_STREAM (默认)、SOCK_DGRAM、SOCK_RAW 或其他 SOCK_ 常量之一。

  • 协议- 数字通常为零并且可以省略。

返回类型

该方法返回一个套接字对象。

一旦获得了套接字对象,就可以使用所需的方法来创建客户端或服务器程序。

服务器套接字方法

在服务器上实例化的套接字称为服务器套接字。服务器上的套接字对象可以使用以下方法 -

  • bind() 方法- 此方法将套接字绑定到指定的 IP 地址和端口号。

  • Listen() 方法- 此方法启动服务器并运行到侦听循环中寻找来自客户端的连接请求。

  • Accept() 方法- 当连接请求被服务器拦截时,此方法接受它并用其地址识别客户端套接字。

要在服务器上创建套接字,请使用以下代码片段 -

import socket
server = socket.socket()
server.bind(('localhost',12345))
server.listen()
client, addr = server.accept()
print ("connection request from: " + str(addr))

默认情况下,服务器绑定到本地计算机的 IP 地址“localhost”,监听任意空端口号。

客户端套接字方法

在客户端也建立类似的socket。主要向监听其IP地址和端口号的服务器socket发送连接请求

连接()方法

此方法采用两项元组对象作为参数。这两项是服务器的IP地址和端口号。

obj=socket.socket()
obj.connect((host,port))

一旦服务器接受连接,两个套接字对象就可以发送和/或接收数据。

发送()方法

服务器使用其截获的地址向客户端发送数据。

client.send(bytes)

客户端套接字将数据发送到与其建立连接的套接字。

sendall() 方法

与send()类似。但是,与 send() 不同,此方法继续从字节发送数据,直到所有数据已发送或发生错误。成功时不会返回任何内容。

sendto() 方法

此方法仅在 UDP 协议的情况下使用。

接收()方法

该方法用于检索发送到客户端的数据。对于服务器,它使用请求已被接受的远程套接字。

client.recv(bytes)

recvfrom() 方法

该方法用于UDP协议的情况。

Python-套接字服务器

为了编写互联网服务器,我们使用套接字模块中提供的套接字函数来创建套接字对象。然后使用套接字对象调用其他函数来设置套接字服务器。

现在调用bind(hostname, port) 函数来为给定主机上的服务指定端口。

接下来,调用返回对象的accept方法。此方法等待客户端连接到您指定的端口,然后返回表示与该客户端的连接的连接对象。

import socket
host = "127.0.0.1"
port = 5001
server = socket.socket()
server.bind((host,port))
server.listen()
conn, addr = server.accept()
print ("Connection from: " + str(addr))
while True:
   data = conn.recv(1024).decode()
   if not data:
      break
   data = str(data).upper()
   print (" from client: " + str(data))
   data = input("type message: ")
   conn.send(data.encode())
conn.close()

Python-Socket 客户端

让我们编写一个非常简单的客户端程序,它打开到给定端口 5001 和给定本地主机的连接。使用Python的socket模块功能创建一个socket客户端非常简单。

socket.connect (hosname, port)在端口上打开到主机名的 TCP 连接。一旦打开了套接字,您就可以像任何 IO 对象一样从中读取数据。完成后,请记住关闭它,就像关闭文件一样。

以下代码是一个非常简单的客户端,它连接到给定的主机和端口,从套接字读取任何可用数据,然后在输入“q”时退出。

import socket
host = '127.0.0.1'
port = 5001
obj = socket.socket()
obj.connect((host,port))
message = input("type message: ")
while message != 'q':
   obj.send(message.encode())
   data = obj.recv(1024).decode()
   print ('Received from server: ' + data)
   message = input("type message: ")
obj.close()
  • 首先运行服务器代码。它开始聆听。

  • 然后启动客户端代码。它发送请求。

  • 请求已接受。已识别客户地址

  • 输入一些文本并按 Enter。

  • 打印收到的数据。发送数据给客户端。

  • 接收到来自服务器的数据。

  • 当输入“q”时循环终止。

服务器-客户端交互如下所示 -

服务器客户端交互

我们已经在本地计算机上使用套接字模块实现了客户端-服务器通信。要将服务器和客户端代码放置在网络上的两台不同计算机上,我们需要找到服务器计算机的 IP 地址。

在 Windows 上,您可以通过运行 ipconfig 命令查找 IP 地址。ifconfig 命令是 Ubuntu 上的等效命令。

ipv4_地址

使用 IPv4 地址值更改服务器和客户端代码中的主机字符串,并像以前一样运行它们。

Python 使用 Socket 模块进行文件传输

以下程序演示了如何使用套接字通信将文件从服务器传输到客户端

服务器代码

建立连接的代码与之前相同。接受连接请求后,以二进制方式打开服务器上的文件进行读取,并依次读取字节并将其发送到客户端流,直到到达文件末尾。

import socket
host = "127.0.0.1"
port = 5001
server = socket.socket()
server.bind((host, port))
server.listen()
conn, addr = server.accept()
data = conn.recv(1024).decode()
filename='test.txt'
f = open(filename,'rb')
while True:
   l = f.read(1024)
   if not l:
      break
   conn.send(l)
   print('Sent ',repr(l))
f.close()
print('File transferred')
conn.close()

客户端代码

在客户端,以wb模式打开一个新文件。从服务器接收到的数据流被写入文件。当流结束时,输出文件将关闭。将在客户端计算机上创建一个新文件。

import socket

s = socket.socket()
host = "127.0.0.1"
port = 5001

s.connect((host, port))
s.send("Hello server!".encode())

with open('recv.txt', 'wb') as f:
   while True:
      print('receiving data...')
      data = s.recv(1024)
      if not data:
         break
      f.write(data)
      
f.close()
print('Successfully received')
s.close()
print('connection closed')

Python 套接字服务器模块

Python标准库中的socketserver模块是一个用于简化编写网络服务器任务的框架。模块中有以下类,它们代表同步服务器 -

套接字服务器模块

这些类与相应的 RequestHandler 类一起使用来实现服务。BaseServer 是模块中所有 Server 对象的超类。

TCPServer类使用互联网 TCP 协议,在客户端和服务器之间提供连续的数据流。构造函数自动尝试调用 server_bind() 和 server_activate()。其他参数传递给 BaseServer 基类。

您还必须创建StreamRequestHandler类的子类。IT提供self.rfile和self.wfile属性来读取或写入以获取请求数据或将数据返回给客户端。

  • UDPServerDatagramRequestHandler - 这些类用于 UDP 协议。

  • DatagramRequestHandlerUnixDatagramServer - 这些类使用 Unix 域套接字;它们在非 Unix 平台上不可用。

服务器代码

您必须编写一个RequestHandler 类。它在每次连接到服务器时实例化一次,并且必须重写handle() 方法来实现与客户端的通信。

import socketserver
class MyTCPHandler(socketserver.BaseRequestHandler):
   def handle(self):
      self.data = self.request.recv(1024).strip()
      host,port=self.client_address
      print("{}:{} wrote:".format(host,port))
      print(self.data.decode())
      msg=input("enter text .. ")
      self.request.sendall(msg.encode())

在服务器分配的端口号上,TCPServer 类的对象调用forever() 方法将服务器置于侦听模式并接受来自客户端的传入请求。

if __name__ == "__main__":
   HOST, PORT = "localhost", 9999
   with socketserver.TCPServer((HOST, PORT), MyTCPHandler) as server:
      server.serve_forever()

客户端代码

使用套接字服务器时,客户端代码或多或少与套接字客户端应用程序类似。

import socket
import sys

HOST, PORT = "localhost", 9999

while True:
   with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as sock:
      # Connect to server and send data
      sock.connect((HOST, PORT))
      data = input("enter text .. .")
      sock.sendall(bytes(data + "\n", "utf-8"))
      
      # Receive data from the server and shut down
      received = str(sock.recv(1024), "utf-8")
      print("Sent: {}".format(data))
      print("Received: {}".format(received))

在一个命令提示符终端中运行服务器代码。为客户端实例打开多个终端。您可以模拟服务器与多个客户端之间的并发通信。

服务器 客户端1 客户端2

D:\socketsrvr>python myserver.py

127.0.0.1:54518 写道:

来自客户端1

输入文字..

你好

127.0.0.1:54522 写道:

你好吗

输入文字..

美好的

127.0.0.1:54523 写道:

来自客户端2

输入文字..

嗨,客户 2

127.0.0.1:54526 写道:

再见

输入文字..

再见

127.0.0.1:54530 写道:

谢谢

输入文字..

再见客户2

D:\socketsrvr>python myclient.py

输入文字...

来自客户端1

发送:

来自客户端1

收到: 你好

输入文字...

你好吗

发送:

你好吗

收到: 很好

输入文字...

再见

已发送:再见

收到:再见

输入文字...

D:\socketsrvr>python myclient.py

输入文字...

来自客户端2

发送:

来自客户端2

收到:嗨客户 2

输入文字...

谢谢

已发送: 谢谢

已收到:

再见客户2

输入文字...