socket浅析(二): 构建通过socket通信的服务端和客户端

socket服务端代码:

 

 1 import socket
 2 import os,subprocess
 3 
 4 
 5 server = socket.socket() #获得socket实例
 6 server.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
 7 
 8 ip_port = ('127.0.0.1', 10000) # 要绑定的IP及端口,元组数据形式
 9 
10 server.bind(ip_port) # 绑定ip port
11 server.listen()  # 开始监听
12 
13 while True: # 第一层loop
14 
15     print("server is waiting...")
16     conn,addr = server.accept() # 接受并建立与客户端的连接,程序在此处开始阻塞,只到有客户端连接进来...
17     print("Connected by :",addr ) # 打印出当前是谁链接了客户端
18 
19     while True:
20         data = conn.recv(1024)
21         if not data:
22             print("客户端断开了...")
23             break # 这里断开就会再次回到第一次外层的loop
24 
25         print("收到要执行的命令:",data)
26         #res = os.popen(data.decode()).read() #py3 里socket发送的只有bytes,os.popen又只能接受str,所以要decode一下
27         res = subprocess.Popen(data,shell=True,stdout=subprocess.PIPE).stdout.read() # 跟上面那条命令的效果是一样的
28 
29         if len(res) == 0:
30             res = "cmd exec success,has not output!".encode("utf-8")
31 
32         conn.send(str(len(res)).encode("utf-8")) # 发送数据之前,先告诉客户端要发多少数据给它
33         client_final_ack = conn.recv(1024) # 等待客户端响应
34         if client_final_ack.decode() == int(len(res)): # 如果收到客户端响应的标志位和发送的标志位相等,即代表客户端已经准备好接收数据了
35             
36             print(type(res))
37             conn.sendall(res) #发送端也有最大数据量限制,所以这里用sendall,相当于重复循环调用conn.send,直至数据发送完毕
38 
39 server.close()

 

 

socket客户端代码:

 

 1 import socket
 2 import sys
 3 
 4 client = socket.socket() # 实例化socket
 5 ip_port = ('127.0.0.1', 10000) # 要绑定的IP及端口,元组数据形式
 6 
 7 
 8 client.connect(ip_port)
 9 
10 while True:
11 
12     msg = input(">>:").strip() # 获取客户端的输入信息并去除空格
13 
14     if len(msg) == 0:continue # 如果输入为空则继续循环等待输入
15 
16     client.send(msg.encode("utf-8"))  # 转码并发送给服务端
17 
18     res_return_size  = client.recv(1024) #接收这条命令执行结果的大小
19     print("getting cmd result , ", res_return_size)
20 
21     total_rece_size = int(res_return_size)
22     print("total size:",res_return_size)
23 
24     client.send(res_return_size.encode("utf-8"))
25     received_size = 0 #已接收到的数据
26     cmd_res = b''
27     f = open("test_copy.html","wb")#把接收到的结果存下来,一会看看收到的数据 对不对
28     while received_size != total_rece_size: #代表还没收完
29         data = client.recv(1024)
30         received_size += len(data) #为什么不是直接1024,还判断len干嘛,注意,实际收到的data有可能比1024少
31         cmd_res += data
32     else:
33         print("数据收完了",received_size)
34         #print(cmd_res.decode())
35         f.write(cmd_res) #把接收到的结果存下来,一会看看收到的数据 对不对
36         #print(data.decode()) #命令执行结果
37 
38 client.close()

 

程序缺限:

这是一个简单的socket通信,里面存在一些bug

1.在客户端输入回车,会挂死。

2.服务端返回的数据大于1024,客户端显示不全。

3.单进程,如果多个客户端连接,要排队,前一个断开,后一个客户端才能通信。

先贴一个多线程的导图:

 

posted @ 2017-04-21 15:25  neightChen  阅读(253)  评论(0编辑  收藏  举报