Loading

Socket介绍、Socket代码示例、粘包现象解释

socket的功能是把tcp/ip协议层的各种数据封装、数据发送、接受等功能封装。

服务器端先初始化Socket,然后与端口绑定(bind),对端口进行监听(listen),调用accept阻塞,等待客户端连接。在这时如果有个客户端初始化一个Socket,然后连接服务器(connect),如果连接成功,这时客户端与服务器端的连接就建立了。客户端发送数据请求,服务器端接收请求并处理请求,然后把回应数据发送给客户端,客户端读取数据,最后关闭连接,一次交互结束。

服务端套接字函数
s.bind()    绑定(主机,端口号)到套接字
s.listen()  开始TCP监听
s.accept()  被动接受TCP客户的连接,(阻塞式)等待连接的到来

客户端套接字函数
s.connect()     主动初始化TCP服务器连接
s.connect_ex()  connect()函数的扩展版本,出错时返回出错码,而不是抛出异常

公共用途的套接字函数
s.recv()            接收TCP数据
s.send()            发送TCP数据(send在待发送数据量大于己端缓存区剩余空间时,数据丢失,不会发完)
s.sendall()         发送完整的TCP数据(本质就是循环调用send,sendall在待发送数据量大于己端缓存区剩余空间时,数据不丢失,循环调用send直到发完)
s.recvfrom()        接收UDP数据
s.sendto()          发送UDP数据
s.getpeername()     连接到当前套接字的远端的地址
s.getsockname()     当前套接字的地址
s.getsockopt()      返回指定套接字的参数
s.setsockopt()      设置指定套接字的参数
s.close()           关闭套接字

面向锁的套接字方法
s.setblocking()     设置套接字的阻塞与非阻塞模式
s.settimeout()      设置阻塞套接字操作的超时时间
s.gettimeout()      得到阻塞套接字操作的超时时间

面向文件的套接字的函数
s.fileno()          套接字的文件描述符
s.makefile()        创建一个与该套接字相关的文件
# tcp服务器端
from socket import *

phone = socket(AF_INET, SOCK_STREAM)
phone.bind(('127.0.0.1', 8081))
phone.listen(5)

conn, addr = phone.accept()
while True:
    data = conn.recv(1024)
    print('server===>')
    print(data.decode(encoding="utf-8"))
    conn.send(data.upper())
conn.close()
phone.close()

 

#tcp客户端
from socket import *

phone = socket(AF_INET, SOCK_STREAM)
phone.connect(('127.0.0.1', 8081))

while True:
    data = input("输入要发送的数据:")
    data_bin = data.encode(encoding="utf-8")
    phone.send(data_bin)
    rececive = phone.recv(1024)

udp是无链接的,先启动哪一端都不会报错

# udp服务端
from socket import *
phone=socket(AF_INET,SOCK_DGRAM)
phone.bind(('127.0.0.1',8082))
while True:
    msg,addr=phone.recvfrom(1024)
    phone.sendto(msg.upper(),addr)
from socket import *
phone=socket(AF_INET,SOCK_DGRAM)
while True:
    msg=input('>>: ')
    phone.sendto(msg.encode('utf-8'),('127.0.0.1',8082))
    msg,addr=phone.recvfrom(1024)
    print(msg)

粘包现象:tcp协议的现象

缓冲区:

tcp是面向流的协议,发送端可以是1K、1K的发送数据,接收端可以2K、2K地接受数据,甚至一次3K或5K,当然一个接收一个字节也是可以。也就是说应用程序看到的是一个整体,并不知道一个消息是多长,这就容易出现粘包现象(一个消息长度超过了一次可以接收的最大长度,第一次接收只收到了一半,当第二次接收时本想拿到第二个消息,但拿到的是存在缓存区的第一个消息的后半部分。

而udp协议是面向消息的,每个UDP段都是一条消息,应用程序必须以消息为单位提取数据,不能一次提取任意字节的数据。但udp协议会丢数据(一次recvfrom没有接受完来自服务端的一个消息)。

struct模块 解决粘包现象

该模块可以把一个类型,如数字,转成固定长度的bytes。

# 客户端
import socket,time,struct

s=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
res=s.connect_ex(('127.0.0.1',8080))

while True:
    msg=input('>>: ').strip()
    if len(msg) == 0:continue
    if msg == 'quit':break

    s.send(msg.encode('utf-8'))

    l=s.recv(4)
    # 此时接收到的是服务端发送的要返回消息的长度
    # 这个长度消息被struct.pack编码成长度为4的bytes
    x=struct.unpack('i',l)[0]
    print(type(x),x)
    # print(struct.unpack('I',l))
    r_s=0
    data=b''
    while r_s < x:
        r_d=s.recv(1024)
        data+=r_d
        r_s+=len(r_d)

    # print(data.decode('utf-8'))
    print(data.decode('gbk')) #windows默认gbk编码
import socket,struct,json
import subprocess
phone=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
phone.setsockopt(socket.SOL_SOCKET,socket.SO_REUSEADDR,1) #就是它,在bind前加

phone.bind(('127.0.0.1',8080))

phone.listen(5)

while True:
    conn,addr=phone.accept()
    while True:
        cmd=conn.recv(1024)
        if not cmd:break
        print('cmd: %s' %cmd)

        res=subprocess.Popen(cmd.decode('utf-8'),
                             shell=True,
                             stdout=subprocess.PIPE,
                             stderr=subprocess.PIPE)
        err=res.stderr.read()
        print(err)
        if err:
            back_msg=err
        else:
            back_msg=res.stdout.read()


        conn.send(struct.pack('i',len(back_msg))) #先发back_msg的长度
        conn.sendall(back_msg) #在发真实的内容

    conn.close()
posted @ 2021-08-13 09:22  climber_dzw  阅读(131)  评论(0编辑  收藏  举报