insert的value里面怎么加select_socket网络编程(三)——select多路复用问题

本文探讨了在多客户端并发场景下,如何通过select避免单线程阻塞的问题。介绍了select的诞生原因,即为了解决多线程的复杂性和开销。文章详细展示了服务端和客户端的具体实现代码,并对select结构进行了剖析。同时,提出了在面临千万级并发时的新挑战。

1、select诞生的原因

在上文《socket网络编程(二)—— 实现持续发送 》我们提到了多客户端的时候,多台客户端发送数据到服务端的话,只能有一台客户端可以正常发送和接受数据,另外一台完全没有反应,那这个问题怎么解决呢?很多人可能第一反应想到利用多线程技术,线程多的话用线程池来维护。的确,多线程确实可以实现这个效果,但是,可能很多看见这个但是就不怎么开心了,却不知很多科学科技的进步都是这个但是引发的。但是一个多线程编程很麻烦又容易出错,二是如果连接有几千个的话,线程间切换的开销确实是很大。如果能够在一个线程里就实现这个效果的话,那该多好啊!

755ad6c34746ee0993da1f10e8121328.png

于是select就横空出世!

这个又叫做非阻塞IO多路复用,就是进程或线程执行此函数时不必非要等待事件的发生,一旦执行肯定返回,以返回值的不同来反映函数的执行情况,如果事件发生则与阻塞方式相同,若事件没有发生则返回一个代码来告知事件未发生,而进程或线程继续执行,所以效率较高可能很多人会说,现在都是用epoll,都不用select了,还讲这个干嘛?可我想说,因为我这个是socket网络编程的一系列教程,一定要一步步地推进,历史上是有诞生了select,然后epoll是为了完善select的缺陷的,做为学习,我们必须先了解select,然后才能知道epoll的特别之处。

2、具体实现

首先,还是先不扯其他的,我先扔出代码,然后结合代码讲解select,我本人是比较喜欢这种学习方式,带着疑问去学习,如果大家不习惯的话,可以先跳过以下的代码,先看代码下方的讲解部分。

2.1、服务端代码:

#include #include #include #include #include #include #include  #define PORT 39002#define MAX_FD_NUM 3#define BUF_SIZE 512#define ERR_EXIT(m)         \    do                      \    {                       \        perror(m);          \        exit(EXIT_FAILURE); \    } while (0) int main(){    //创建套接字    int m_sockfd = socket(AF_INET, SOCK_STREAM, 0);    if (m_sockfd < 0)    {        ERR_EXIT("create socket fail");    }     //初始化socket元素    struct sockaddr_in server_addr;    int server_len = sizeof(server_addr);    memset(&server_addr, 0, server_len);     server_addr.sin_family = AF_INET;    //server_addr.sin_addr.s_addr = inet_addr("0.0.0.0"); //用这个写法也可以    server_addr.sin_addr.s_addr = INADDR_ANY;    server_addr.sin_port = htons(PORT);     //绑定文件描述符和服务器的ip和端口号    int m_bindfd = bind(m_sockfd, (struct sockaddr *)&server_addr, server_len);    if (m_bindfd < 0)    {        ERR_EXIT("bind ip and port fail");    }     //进入监听状态,等待用户发起请求    int m_listenfd = listen(m_sockfd, MAX_FD_NUM);    if (m_listenfd < 0)    {        ERR_EXIT("listen client fail");    }     //定义客户端的套接字,这里返回一个新的套接字,后面通信时,就用这个m_connfd进行通信    //struct sockaddr_in client_addr;    //socklen_t client_len = sizeof(client_addr);    //int m_connfd = accept(m_sockfd, (struct sockaddr *)&client_addr, &client_len);     printf("client accept success\n");     struct sockaddr_in client_addr;    socklen_t client_len = sizeof(client_addr);     //接收客户端数据,并相应    char buffer[BUF_SIZE];    int array_fd[MAX_FD_NUM];    //客户端连接数量    int client_count = 0;     fd_set tmpfd;    int max_fd = m_sockfd;    struct timeval timeout;     for (int i = 0; i < MAX_FD_NUM; i++)    {        array_fd[i] = -1;    }    //array_fd[0] = m_sockfd;     while (1)    {        FD_ZERO(&tmpfd);        FD_SET(m_sockfd, &tmpfd);        int i;         //所有在线的客户端加入到fd中,并找出最大的socket        for (i = 0; i < MAX_FD_NUM; i++)        {            if (array_fd[i] > 0)            {                FD_SET(array_fd[i], &tmpfd); //set array_fd in red_set                if (max_fd < array_fd[i])                {                    max_fd = array_fd[i]; //get max_fd                }            }        }         int ret = select(max_fd + 1, &tmpfd, NULL, NULL, NULL);        if (ret < 0)        {            ERR_EXIT("select fail");        }        else if (ret == 0)        {            //ERR_EXIT("select timeout"); //超时不是错误,不可断掉连接            printf("select timeout\n");            continue;        }         //表示有客户端连接        if (FD_ISSET(m_sockfd, &tmpfd))        {            int m_connfd = accept(m_sockfd, (struct sockaddr *)&client_addr, &client_len);            if (m_connfd < 0)            {                ERR_EXIT("server accept fail");            }             //客户端连接数已满            if (client_count >= MAX_FD_NUM)            {                printf("max connections arrive!!!\n");                // char buff[]="max connections arrive!!!";                // send(m_connfd, buff, sizeof(buff) - 1, 0);                close(m_connfd);                continue;            }             //客户端数量加1            client_count++;            printf("we got a new connection, client_socket=%d, client_count=%d, ip=%s, port=%d\n", m_connfd, client_count, inet_ntoa(client_addr.sin_addr), ntohs(client_addr.sin_port));             for (i = 0; i < MAX_FD_NUM; i++)            {                if (array_fd[i] == -1)                {                    array_fd[i] = m_connfd;                    break;                }            }        }         //遍历所有的客户端连接,找到发送数据的那个客户端描述符        for (i = 0; i < MAX_FD_NUM; i++)        {            if (array_fd[i] < 0)            {                continue;            }            //有客户端发送过来的数据            else            {                if (FD_ISSET(array_fd[i], &tmpfd))                {                    memset(buffer, 0, sizeof(buffer)); //重置缓冲区                    int recv_len = recv(array_fd[i], buffer, sizeof(buffer) - 1, 0);                    if (recv_len < 0)                    {                        ERR_EXIT("recv data fail");                    }                    //客户端断开连接                    else if (recv_len == 0)                    {                        client_count--;                        //打印断开的客户端数据                        printf("client_socket=[%d] close, client_count=[%d], ip=%s, port=%d\n\n", array_fd[i], client_count, inet_ntoa(client_addr.sin_addr), ntohs(client_addr.sin_port));                        close(array_fd[i]);                        FD_CLR(array_fd[i], &tmpfd);                        array_fd[i] = -1;                    }                    else                    {                        printf("server recv:%s\n", buffer);                        strcat(buffer, "+ACK");                        send(array_fd[i], buffer, sizeof(buffer) - 1, 0);                    }                }            }        }    }     //关闭套接字    close(m_sockfd);     printf("server socket closed!!!\n");     return 0;}

2.1、客户端代码:

#include #include #include #include #include #include  #define BUF_SIZE 512#define ERR_EXIT(m)         \    do                      \    {                       \        perror(m);          \        exit(EXIT_FAILURE); \    } while (0) int main(){    //创建套接字    int m_sockfd = socket(AF_INET, SOCK_STREAM, 0);    if (m_sockfd < 0)    {        ERR_EXIT("create socket fail");    }     //服务器的ip为本地,端口号    struct sockaddr_in server_addr;    memset(&server_addr, 0, sizeof(server_addr));    server_addr.sin_family = AF_INET;    server_addr.sin_addr.s_addr = inet_addr("81.68.140.74");    server_addr.sin_port = htons(39002);     //向服务器发送连接请求    int m_connectfd = connect(m_sockfd, (struct sockaddr *)&server_addr, sizeof(server_addr));    if (m_connectfd < 0)    {        ERR_EXIT("connect server fail");    }     //发送并接收数据    char buffer[BUF_SIZE];    while (1)    {        memset(buffer, 0, sizeof(buffer)); //重置缓冲区        printf("client send:");        scanf("%s", buffer);        send(m_sockfd, buffer, sizeof(buffer) - 1, MSG_NOSIGNAL);        recv(m_sockfd, buffer, sizeof(buffer) - 1, 0);        printf("client recv:%s\n", buffer);    }     //断开连接    close(m_sockfd);     printf("client socket closed!!!\n");     return 0;}
以下是在Linux上运行的结果

155aa4f36f4e29ecbf4cbec5ced37e4b.png

服务端

0d0167fcbca97e24499990f4fe3a2168.png

客户端1

208c9b3f10e8c27634fe56fb2874c181.png

客户端2

3、select结构剖析

说到select的IO多路复用就不得不提fd_set这个变量类型,首先我们打开Linux的fd_set数据结构的源码我们可以看到,就是一个长度为32的long int类型的数组(要注意,windows的源码和Linux的不一样)。每一位可以代表一个文件描述符,所以fd_set最多表示1024个文件描述符!
这里为啥是1024个描述符呢?long int长度是32bit,数组长度是32,32*32=1024!

be970979572ae8a830a3bd421dd53ac8.png

如果知道epoll用法的童鞋,可能就会知道,最多只能表示1024个文件描述符恰恰也成为了select的缺陷!言归正传,fd_set中的每一bit可以对应一个文件描述符fd,则1字节长的fd_set最大可以对应8个fd。现在我们来看看fd_set定义的几个宏
#include int FD_ZERO(fd_set *fdset);    int FD_SET(int fd, fd_set *fd_set);   int FD_ISSET(int fd, fd_set *fdset);int FD_CLR(int fd, fd_set *fdset);
我们假设fdset就一个字节,就是8位,那么(1)执行FD_ZERO(&fdset),则set用位表示是00000000,就是所有位都清空成0,一般刚开始的时候就需要清空。
(2)执行FD_SET(fd,&fdset),若fd=5,后set变为00010000,第5位置为1,就是将客户端连接的描述字(一般就是一个整数啦)放入到set当中。
(3)执行FD_ISSET(fd,&fdset),若fd=5,则就是判断set的第5位是否是1,一般用来判断是否客户端的连接。
(4)执行FD_CLR(fd,&fdset),若fd=5,则就是将第5位置成0,在断开客户端连接的时候,一定要记得调用这个。以上的铺垫都做完之后,我们将要引出重量级的选手select,首先我们先来看下select的函数定义。
int select(int maxfdp, fd_set *readset, fd_set *writeset, fd_set *exceptset,struct timeval *timeout);
参数说明:maxfdp:被监听的文件描述符的总数,它比所有文件描述符集合中的文件描述符的最大值大1,因为文件描述符是从0开始计数的;fd_set *readset: 该参数是我们所关心的文件是否可读的文件描述符的集合,如果这个集合中有个文件可读了,那select返回一个大于0的数,表示有文件可读了,比如说服务端接收到客户端的数据,服务端都是读的状态,所以正常读的文件都放在这里。fd_set *writeset:那这个大家就比较好理解了,服务端发到客户端的数据,要写入到缓冲区,那么所有正常写的文件都放在这里。fd_set *exceptset:在所有正常读和正常写的时候,产生了异常情况,那么异常文件就放在这里。timeval *timeout:用于设置select函数的超时时间,即告诉内核select等待多长时间之后就放弃等待。这个参数使select处于三种状态:(1)timeout传入NULL,则select一直等到文件状态有变化时才返回,这段时间一直处于阻塞状态。(2):timeout传入0,则select会立即返回(非阻塞),如果文件状态有变化则返回一个大于0的值没有变化则返回0;(3)timeout传入一个大于0的数,则select在timeout时间内阻塞,一旦文件状态有变化就会返回,超时后不管怎样都会返回值同样是文件状态右边话就返回一个大于0的值,无变化则返回0;timeval结构体定义如下:
struct timeval{          long tv_sec;   /*秒 */    long tv_usec;  /*微秒 */   };
说完了用到的知识点,我来解释一下代码的部分实现。首先定义了一个array_fd的整形数据,数组的最长长度是MAX_FD_NUM,数组中存放的就是客户端连接的描述符,说白了就是客户端的连接,初始化的时候置成-1,只要客户端连接上了,就是描述符插入数组(实际上就是将其中的一个-1置成描述符)。最后遍历所有的客户端连接,找到发送数据的那个客户端描述符。刚开始看这些代码的时候可能有点难,但是只要掌握了以上知识点,那么再看这些代码就很简单了。

67fdc58a961b2642887b3aaf74354b60.png

4、新的问题,千万级的并发

我们刚才说到了select的IO多路复用最多只能支持1024个连接,超过了就不支持了,这个最大值用宏FD_SETSIZE定义的。可是我们实际的有些大的应用,连接数很容易超过这个数字,那如果还用这个就需要更改linux内核的select.h文件,然后重新编译内核了,这是一个问题。另一个是我们select的原理是遍历所有的连接,找到需要的那个,一旦连接数越多,就耗费资源,这是一对相互的矛盾体。最后假设我们的服务器需要支持100万的并发连接,则在FD_SETSIZE为1024的情况下,则我们至少需要开辟1k个进程才能实现100万的并发连接。除了进程间上下文切换的时间消耗外,从内核/用户空间大量的无脑内存拷贝、数组轮询等,是系统难以承受的。因此,基于select模型的服务器程序,要达到10万级别的并发访问,是一个很难完成的任务。那么在高并发下,我们又该怎么做呢?你能想得到解决方法吗?更多精彩内容,请关注同名公众号:一点笔记alittle

b693e2e25b70e9aac2b8ee4cc04b13f5.png

import struct import json import os import hashlib # 1. 打包协议头部:命令字(4B) + 载荷长度(4B, 小端) def pack_header(cmd, payload_len): cmd_bytes = cmd.encode('ascii')[:4].ljust(4, b'\x00') # 确保命令字4字节 len_bytes = struct.pack('<I', payload_len) # 小端无符号int return cmd_bytes + len_bytes # 2. 解包协议头部 def unpack_header(header): cmd = header[:4].decode('ascii').strip('\x00') # 去除填充的空字符 payload_len = struct.unpack('<I', header[4:])[0] return cmd, payload_len # 3. 生成不重复的目标文件名 def get_unique_filename(save_dir, class_name, stu_info, title, ext): base_name = f"{class_name}-{stu_info}-{title}{ext}" full_path = os.path.join(save_dir, base_name) # 若文件存在,添序号后缀(如xxx_1.docx) count = 1 while os.path.exists(full_path): full_path = os.path.join(save_dir, f"{class_name}-{stu_info}-{title}_{count}{ext}") count += 1 return full_path # 4. 计算文件MD5(用于校验完整性) def calculate_md5(file_path, chunk_size=1024*1024): md5 = hashlib.md5() with open(file_path, 'rb') as f: while chunk := f.read(chunk_size): md5.update(chunk) return md5.hexdigest() import socket import threading import json import os # 服务器配置 SAVE_DIR = "./received_files" # 文件保存目录 os.makedirs(SAVE_DIR, exist_ok=True) # 确保目录存在 DEFAULT_PORT = 6655 CHUNK_SIZE = 64 * 1024 # 64KB分块 # 处理单个客户端的交互逻辑(每个客户端一个线程) def handle_client(client_socket, client_addr): print(f"[新连接] {client_addr} 已连接") try: while True: # 1. 接收协议头部(8字节) header = client_socket.recv(8) if not header: print(f"[连接关闭] {client_addr} 主动断开") break cmd, payload_len = unpack_header(header) # 2. 处理不同命令 if cmd == "UPLD": # 上传文件命令 handle_upload(client_socket, payload_len) elif cmd == "LIST": # 列出已交作业(扩展功能) handle_list(client_socket) elif cmd == "QUIT": # 断开连接命令 print(f"[退出] {client_addr} 请求断开") client_socket.sendall(b'OK') break else: # 未知命令 client_socket.sendall(b'NO') break except Exception as e: print(f"[错误] {client_addr} 交互异常: {str(e)}") client_socket.sendall(b'NO') finally: client_socket.close() print(f"[连接关闭] {client_addr} 连接已释放") # 处理文件上传逻辑 def handle_upload(client_socket, payload_len): # 1. 接收载荷(JSON元数据 + 文件二进制流) payload = b'' while len(payload) < payload_len: recv_data = client_socket.recv(min(CHUNK_SIZE, payload_len - len(payload))) if not recv_data: client_socket.sendall(b'NO') return payload += recv_data # 2. 解析JSON元数据(找到JSON结束位置,区分元数据与文件流) json_end_idx = payload.find(b'}') + 1 if json_end_idx == 0: client_socket.sendall(b'NO') return meta_data = json.loads(payload[:json_end_idx].decode('utf-8')) file_data = payload[json_end_idx:] # 提取文件二进制流 # 3. 生成唯一保存路径 class_name = meta_data.get('class', '') stu_info = meta_data.get('stu', '') title = meta_data.get('title', '') ext = meta_data.get('ext', '') save_path = get_unique_filename(SAVE_DIR, class_name, stu_info, title, ext) # 4. 分块写入文件,实时反馈进度(每接收1块回传1次OK) try: with open(save_path, 'wb') as f: total_len = len(file_data) sent_len = 0 # 分块写入并反馈进度 for i in range(0, total_len, CHUNK_SIZE): chunk = file_data[i:i + CHUNK_SIZE] f.write(chunk) sent_len += len(chunk) client_socket.sendall(b'OK') # 回传进度确认 # 5. 上传完成,返回最终状态 client_socket.sendall(b'OK') print(f"[上传成功] 文件已保存至: {save_path}") except Exception as e: print(f"[上传失败] 写入文件异常: {str(e)}") client_socket.sendall(b'NO') # 处理列出已交作业(扩展功能) def handle_list(client_socket): try: # 获取保存目录下的所有作业文件 files = [f for f in os.listdir(SAVE_DIR) if f.endswith(('.docx', '.zip'))] file_list = json.dumps({"files": files}).encode('utf-8') # 打包并发送列表数据 header = pack_header("LIST", len(file_list)) client_socket.sendall(header + file_list) except Exception as e: client_socket.sendall(b'NO') # 服务器启动函数 def start_server(port): # 创建TCP socket server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 允许端口复用(避免重启时端口占用) server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) server_socket.bind(('0.0.0.0', port)) # 监听所有网卡 server_socket.listen(5) # 监听队列大小 print(f"[服务器启动] 监听端口 {port},等待客户端连接...") try: while True: # 接收客户端连接 client_socket, client_addr = server_socket.accept() # 创建新线程处理客户端 client_thread = threading.Thread(target=handle_client, args=(client_socket, client_addr)) client_thread.daemon = True # 守护线程,随主线程退出 client_thread.start() except KeyboardInterrupt: print("\n[服务器停止] 收到中断信号,正在关闭...") finally: server_socket.close() # 命令行参数解析与服务器启动 if __name__ == "__main__": import sys # 从命令行获取端口,默认6655 port = int(sys.argv[1]) if len(sys.argv) > 1 else DEFAULT_PORT start_server(port) import socket import json import os import tkinter as tk from tkinter import ttk, filedialog, messagebox # 客户端配置 DEFAULT_SERVER_IP = "127.0.0.1" DEFAULT_SERVER_PORT = 6655 CHUNK_SIZE = 64 * 1024 # 64KB分块 class FileUploadClient: def __init__(self, root): self.root = root self.root.title("作业上传客户端") self.root.geometry("500x350") # 初始化变量 self.server_ip = tk.StringVar(value=DEFAULT_SERVER_IP) self.server_port = tk.IntVar(value=DEFAULT_SERVER_PORT) self.class_name = tk.StringVar() self.stu_info = tk.StringVar() self.title = tk.StringVar() self.file_path = tk.StringVar() self.upload_status = tk.StringVar(value="等待上传...") # 构建界面 self.create_widgets() # 构建GUI界面 def create_widgets(self): # 1. 服务器配置区域 frame_server = ttk.LabelFrame(self.root, text="服务器配置") frame_server.pack(fill=tk.X, padx=10, pady=5) ttk.Label(frame_server, text="服务器IP:").grid(row=0, column=0, padx=5, pady=5, sticky=tk.W) ttk.Entry(frame_server, textvariable=self.server_ip).grid(row=0, column=1, padx=5, pady=5, sticky=tk.EW) ttk.Label(frame_server, text="服务器端口:").grid(row=0, column=2, padx=5, pady=5, sticky=tk.W) ttk.Entry(frame_server, textvariable=self.server_port).grid(row=0, column=3, padx=5, pady=5, sticky=tk.EW) # 2. 作业信息区域 frame_info = ttk.LabelFrame(self.root, text="作业信息") frame_info.pack(fill=tk.X, padx=10, pady=5) ttk.Label(frame_info, text="班级:").grid(row=0, column=0, padx=5, pady=5, sticky=tk.W) ttk.Entry(frame_info, textvariable=self.class_name).grid(row=0, column=1, padx=5, pady=5, sticky=tk.EW, columnspan=3) ttk.Label(frame_info, text="学号姓名:").grid(row=1, column=0, padx=5, pady=5, sticky=tk.W) ttk.Entry(frame_info, textvariable=self.stu_info).grid(row=1, column=1, padx=5, pady=5, sticky=tk.EW, columnspan=3) ttk.Label(frame_info, text="作业标题:").grid(row=2, column=0, padx=5, pady=5, sticky=tk.W) ttk.Entry(frame_info, textvariable=self.title).grid(row=2, column=1, padx=5, pady=5, sticky=tk.EW, columnspan=3) # 3. 文件选择区域 frame_file = ttk.LabelFrame(self.root, text="文件选择") frame_file.pack(fill=tk.X, padx=10, pady=5) ttk.Entry(frame_file, textvariable=self.file_path).grid(row=0, column=0, padx=5, pady=5, sticky=tk.EW, columnspan=3) ttk.Button(frame_file, text="选择文件", command=self.select_file).grid(row=0, column=3, padx=5, pady=5) # 4. 操作按钮区域 frame_btn = ttk.Frame(self.root) frame_btn.pack(fill=tk.X, padx=10, pady=5) ttk.Button(frame_btn, text="开始上传", command=self.start_upload).grid(row=0, column=0, padx=5, pady=5) ttk.Button(frame_btn, text="退出", command=self.root.quit).grid(row=0, column=1, padx=5, pady=5) # 5. 状态显示区域 frame_status = ttk.LabelFrame(self.root, text="上传状态") frame_status.pack(fill=tk.X, padx=10, pady=5) ttk.Label(frame_status, textvariable=self.upload_status).grid(row=0, column=0, padx=5, pady=5, sticky=tk.W) # 选择文件(docx/zip) def select_file(self): filetypes = [("作业文件", "*.docx *.zip"), ("所有文件", "*.*")] path = filedialog.askopenfilename(filetypes=filetypes) if path: self.file_path.set(path) # 开始上传逻辑 def start_upload(self): # 1. 校验输入合法性 if not all([self.class_name.get(), self.stu_info.get(), self.title.get(), self.file_path.get()]): messagebox.showerror("错误", "请填写完整信息并选择文件!") return if not os.path.exists(self.file_path.get()): messagebox.showerror("错误", "选择的文件不存在!") return # 2. 提取文件信息 file_path = self.file_path.get() ext = os.path.splitext(file_path)[1].lower() # 获取文件后缀(.docx/.zip) if ext not in ('.docx', '.zip'): messagebox.showerror("错误", "仅支持docx和zip格式文件!") return # 3. 封装JSON元数据 meta_data = { "class": self.class_name.get(), "stu": self.stu_info.get(), "title": self.title.get(), "ext": ext } meta_json = json.dumps(meta_data).encode('utf-8') # 4. 读取文件二进制数据 try: with open(file_path, 'rb') as f: file_data = f.read() except Exception as e: messagebox.showerror("错误", f"读取文件失败: {str(e)}") return # 5. 封装完整载荷(JSON元数据 + 文件流) payload = meta_json + file_data # 6. 封装协议头部(UPLD命令 + 载荷长度) header = pack_header("UPLD", len(payload)) # 7. 建立TCP连接并上传 self.upload_status.set("正在连接服务器...") self.root.update() # 刷新界面 try: # 创建客户端socket client_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) client_socket.connect((self.server_ip.get(), self.server_port.get())) # 8. 发送头部和载荷 self.upload_status.set("正在上传文件...") self.root.update() # 发送头部 client_socket.sendall(header) # 分块发送载荷 sent_len = 0 total_len = len(payload) while sent_len < total_len: chunk = payload[sent_len:sent_len + CHUNK_SIZE] client_socket.sendall(chunk) sent_len += len(chunk) # 接收服务器进度反馈(每块接收后回传OK) progress_resp = client_socket.recv(2) if progress_resp != b'OK': raise Exception("服务器接收进度异常") # 9. 接收最终上传结果 final_resp = client_socket.recv(2) if final_resp == b'OK': self.upload_status.set("上传成功!") messagebox.showinfo("成功", "文件上传完成!") 这段代码有什么问题,帮我修改
最新发布
10-27
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符  | 博主筛选后可见
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值