多个客户端上传文件

TCP使用中的多线程问题

 1、问题:一般的TCP程序在处理多线程情况时会出现问题:

    如:有A、B两个客户端向服务器发送数据,当A客户端连接上以后被服务端获取到,服务端执行具体流程;这时B客户端连接,只有等待。

        因为服务端还没有处理完A客户端的请求,还有循环回来执行下次accept方法。所以暂时获取不到B客户端对象。

 

2、解决思路:
   服务端最好就是将每个客户端封装到一个单独的线程中,这样,就可以同时处理多个客户端请求。


程序代码:

  客户端:

 1 class PicSend{
 2 
 3 public static void main(String args[]){
 4         try {
 5             Socket s = new Socket(InetAddress.getLocalHost().getHostAddress(),11001    );
 6             FileInputStream ins = new FileInputStream("test/1.jpg");
 7             OutputStream out = s.getOutputStream();
 8             //BufferedReader bufr = new BufferedReader(new InputStreamReader(s.getInputStream()));
 9             byte[] buf = new byte[1024];
10             int len = 0;
11             while((len = ins.read(buf))!= -1){
12                 out.write(buf, 0, len);
13             }
14             
15             s.shutdownOutput();//结束图片上传,读取服务器返回信息
16             
17 //            String str = bufr.readLine();
18 //            System.out.println(str);
19             InputStream in = s.getInputStream();
20             byte[] bytes = new byte[1024];
21             int lens = in.read(bytes);
22             System.out.println("服务器返回:"+ new String(bytes,0,lens));
23             
24             ins.close();
25             s.close();
26                     
27         }  catch (Exception e) {
28             // TODO Auto-generated catch block
29             e.printStackTrace();
30         }
31         
32     }
33 
34 
35 }

服务器端:

 

 1 class ThreadServer{
 2     public static void main(String args[]){
 3         try {
 4             ServerSocket server = new ServerSocket(10001);
 5             
 6             while(true){  //注意:while中使用true,是因为accept方法为阻塞方法,不会导致程序无限循环
 7                 Socket s = server.accept();
 8                 new Thread(new PicThread(s)).start();
 9             }
10         } catch (IOException e) {
11             // TODO Auto-generated catch block
12             e.printStackTrace();
13         }        
14         
15     }
16 }
17 
18 class PicThread implements Runnable{//封装客户端,实现单线程
19     private Socket s;
20     PicThread(Socket s){
21         this.s = s;
22         
23     }
24     
25     public void run(){//将客户端处理代码写run方法中
26         try {
27             InputStream in = s.getInputStream();
28             FileOutputStream fout = new FileOutputStream("Thread.jpg");
29             byte[] buf = new byte[1024];
30             int len = 0;
31             while((len=in.read(buf))!=-1){
32                 fout.write(buf,0,len);
33             }
34             
35             OutputStream out = s.getOutputStream();
36             byte[] bytes = "图片上传成功".getBytes();
37             out.write(bytes);
38             
39             fout.close();
40             s.close();
41             
42         } catch (Exception e) {
43             // TODO Auto-generated catch block
44             e.printStackTrace();
45         }
46         
47     }
48     

这样就实现了多客户端向服务器发送文件。

 

3、设计亮点

  在这个TCP程序中十分巧妙的将服务器端将每一个客户端封装到一个单独的线程中,从而是服务器具备了处理多线程的能力;这样的设计思路在开发中非常有用。

  

    ServerSocket server = new ServerSocket(10001);
            
            while(true){  
                Socket s = server.accept();
                new Thread(new PicThread(s)).start();
            }    

 

转载于:https://www.cnblogs.com/chizhongyue/p/4610191.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要实现Python多客户端上传文件到服务端,可以使用socket和os模块。具体步骤如下: 1. 服务端创建socket并监听端口,等待客户端连接。 2. 客户端连接服务端,并发送要上传的文件名和大小等信息。 3. 服务端接收客户端发送的信息,并创建一个同名文件,准备接收文件内容。 4. 客户端读取文件内容并发送给服务端。 5. 服务端接收文件内容并写入文件,直到文件全部接收完毕。 6. 客户端发送文件传输结束的标志。 7. 服务端接收标志并关闭连接。 以下是Python示例代码: 服务端代码: ```python import socket import os SERVER_HOST = '0.0.0.0' SERVER_PORT = 5000 BUFFER_SIZE = 4096 def main(): # 创建socket并监听端口 server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) server_socket.bind((SERVER_HOST, SERVER_PORT)) server_socket.listen(1) print(f'Server listening on port {SERVER_PORT}') while True: # 等待客户端连接 client_socket, client_addr = server_socket.accept() print(f'Client {client_addr} connected') # 接收文件名和大小等信息 file_info = client_socket.recv(BUFFER_SIZE).decode() file_name, file_size = file_info.split(':') file_size = int(file_size) print(f'Receiving file "{file_name}" ({file_size} bytes)') # 创建同名文件并准备接收文件内容 with open(file_name, 'wb') as f: # 接收文件内容 received_size = 0 while received_size < file_size: data = client_socket.recv(BUFFER_SIZE) f.write(data) received_size += len(data) print(f'File "{file_name}" received') # 关闭连接 client_socket.close() if __name__ == '__main__': main() ``` 客户端代码: ```python import socket import os SERVER_HOST = '127.0.0.1' SERVER_PORT = 5000 BUFFER_SIZE = 4096 def send_file(file_path): # 获取文件名和大小等信息 file_name = os.path.basename(file_path) file_size = os.path.getsize(file_path) file_info = f'{file_name}:{file_size}' # 连接服务端并发送文件名和大小等信息 client_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) client_socket.connect((SERVER_HOST, SERVER_PORT)) client_socket.send(file_info.encode()) print(f'Sending file "{file_name}" ({file_size} bytes)') # 发送文件内容 with open(file_path, 'rb') as f: sent_size = 0 while sent_size < file_size: data = f.read(BUFFER_SIZE) client_socket.send(data) sent_size += len(data) print(f'File "{file_name}" sent') # 发送文件传输结束的标志 client_socket.send('EOF'.encode()) # 关闭连接 client_socket.close() def main(): file_path = 'test.txt' send_file(file_path) if __name__ == '__main__': main() ``` 在上面的示例代码中,服务端和客户端都使用了4096字节的缓冲区大小,可以根据实际情况进行调整。另外,在文件传输过程中,需要注意处理异常情况,如连接断开等。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值