python学习-13【网络编程】

1、Socket 网络模块

Socket 模块

在 Python 中,使用 socket 模块的 socket() 函数来创建一个 socket 对象:
socket.socket(family, type, proto)

  • family:套接字家族,该参数指定调用者期待返回的套接字接口地址结构的类型
    • AF_UNIX:同一台机器上的进程通信
    • AF_INET:使用 IPv4 通信,不会返回 IPv6 的信息
    • AF_INET6:使用 IPv6 通信,不返回 IPv4 的信息
    • AF_UNSPEC:函数返回的是适用于指定主机名和服务名,并且适合任何协议族的地址
  • type:套接字类型。根据是面向连接还是非连接分为 SOCK_STREAM(对应 TCP 协议)和 SOCK_DGRAM(对应 UDP 协议)
  • proto:默认为 0
import socket
socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 创建 TCP Socket
socket.socket(socket.AF_INET, socket.SOCK_DGRAM) # 创建 TCP Socket
Socket 方法
  • 服务器端套接字方法

    方法说明
    s.bind()以 (hostname, port) 的形式绑定地址到套接字,在 AF_INET 下,以元组的形式表示地址
    s.listen()进行 TCP 监听。backlog 指定在拒绝连接之前,操作系统可以挂起的最大连接数量(≥1)
    s.accept()被动接受 TCP 客户端连接,以阻塞式等待连接的到来
  • 客户端套接字方法

    方法说明
    s.connect()主动初始化 TCP 服务器连接。一般 address 的格式为元组形式。如果连接出错,返回 socket.error 错误
    s.connect_ex()是 connect() 的扩展版本,出错时返回出错码,不会抛出异常
  • 公共用途的套接字方法

    方法说明
    s.recv()用于接收 TCP 数据,数据以 字符串 形式返回
    s.send()发送 TCP 数据。返回值是要发送的字节数量
    s.sendall()完整发送 TCP 数据。成功返回 None,失败则抛出异常
    s.recvfrom()接受 UDP 数据,与 recv() 类似,但返回的是(data, address)
    s.sendto()发送 UDP 数据,address 是形式为(ipaddr, port)的元组,指定远程地址,返回的是发送的字节数
    s.close()关闭套接字
    s.getpeername()返回套接字的远程地址,返回值通常是元组
    s.getsockname()返回套接字自己的地址。通常是一个元组(ipaddr, port)
    s.setsockopt(level, optname, value)设置给定套接字选项的值
    s.getsockopt(level, optname[.buflen])返回套接字选项的值
    s.settimeout(timeout)设置套接字操作的超时期,timeout 是一个浮点数,单位是。值为 None 白哦是没有超时期
    s.gettimeout()返回当前超时期的值,单位是,没有则返回 None
    s.fileno()返回套接字的文件描述符
    s.setblocking(flag)如果 flag 为 0,则将套接字设置为非阻塞模式,否则设为阻塞模式(默认值)
    s.makefile()创建一个与该套接字相关联的文件

2、TCP 编程

TCP 客户端
import socket

s = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 创建 socket 对象
s.connect(('www.baidu.com', 80)) # 主动发起 TCP 连接
s.send(b'GET / HTTP/1.1\r\nHOST: www.baidu.com\r\nConnection: close\r\n\r\n') # 发送 HTTP 请求

buffer = []
while True:
    data = s.recv(1024) # 接收数据
    if data:
        buffer.append(data)
    else:
        break
web_data = b''.join(buffer)

http_header, http_content = web_data.split(b'\r\n\r\n', 1)
with open('web_info.html', 'wb') as f:
    f.write(http_content)
TCP 服务器

服务器端:

import socket
import threading

def tcp_server(client: socket.socket, address: tuple):
    print(f"The Client come from {address[0]}:{address[1]}")
    client.send(f"Welcone from {address[0]}:{address[1]}\r\n".encode("utf-8"))
    while True:
        content = client.recv(1024)
        if content == b"exit":
            break
        elif content:
            print(content.decode("utf-8"))
        else:
            break
    print("Client exits, connection closed!")
    client.close()

s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.bind(("127.0.0.1", 8888)) # 绑定 IP 地址和端口号
s.listen(5) # 进行监听
print('waiting connection...')
print('Server is launching the new connection from 127.0.0.1:8888')

while True:
    client, address = s.accept() # 接受客户端的连接
    t = threading.Thread(target=tcp_server, args=(client, address))
    t.start()

客户端:

import socket

client = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
client.connect(("127.0.0.1", 8888))
client.send("Hi, I am Client!".encode("utf-8"))
server_content = client.recv(1024)
print(server_content.decode("utf-8"))
client.send(b"exit")
client.close()

3、UDP 编程

UDP 服务器
import socket

s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
s.bind(('127.0.0.1', 8888))
while True:
    data, address = s.recvfrom(1024)
    print('Launching connection...')
    print(f'Server receive the message from {address[0]}:{address[1]}')
    print(data.decode("utf-8"))

不需要使用 listen() 方法监听,直接调用 recvfrom() 方法接收数据

UDP 客户端
import socket

s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
server = ("127.0.0.1", 8888)
s.sendto("Hi, I am Client!".encode("utf-8"), server)
print('connecting Server...')
s.close()
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值