文件上传的讲解:

 import subprocess

 res=subprocess.Popen("dir",
shell=True,
stderr=subprocess.PIPE,
stdout=subprocess.PIPE) print(res.stdout.read().decode("gbk"))

server

 import socket
import os
import json #序列化对象 : 把字典 列表变成字符串 sock=socket.socket()
sock.connect(("127.0.0.1",8800)) while 1 :
cmd=input("请输入命令:") # put 111.jpg action,filename=cmd.strip().split(" ")
filesize=os.path.getsize(filename) file_info={
"action": action,
"filename": filename,
"filesize": filesize,
}
file_info_json=json.dumps(file_info).encode("utf8")
sock.send(file_info_json) # 确认服务端接收到了文件信息
code=sock.recv(1024).decode("utf8")
if code=="":
# 发送文件数据
with open(filename,"rb") as f:
for line in f:
sock.send(line)
else:
print("服务器异常!") ''' '{.....}' 'xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx' '''

client

对上传的文件进行加密校验:
 import struct
import socket
import json
import hashlib sock=socket.socket()
sock.bind(('127.0.0.1',8800))
sock.listen(5) while 1:
print("server is working....")
conn,addr= sock.accept()
while 1: # 接收json的打包长度
file_info_length_pack=conn.recv(4)
file_info_length=struct.unpack("i",file_info_length_pack)[0] # 接收json字符串
file_info_json=conn.recv(file_info_length).decode("utf8")
file_info=json.loads(file_info_json) action=file_info.get("action")
filename=file_info.get("filename")
filesize=file_info.get("filesize") # 循环接收文件
md5=hashlib.md5()
with open("put/"+filename,"wb") as f:
recv_data_length=0
while recv_data_length<filesize:
data=conn.recv(1024)
recv_data_length+=len(data)
f.write(data)
# MD5摘要
md5.update(data)
print("文件总大小:%s,已成功接收%s"%(filesize,recv_data_length)) print("接收成功!")
conn.send(b"OK")
print(md5.hexdigest())
md5_val=md5.hexdigest()
client_md5=conn.recv(1024).decode("utf8")
if md5_val==client_md5:
conn.send(b"")
else:
conn.send(b"")

server

import socket
import os
import json
import struct
import hashlib sock=socket.socket()
sock.connect(("127.0.0.1",8800)) while 1 :
cmd=input("请输入命令:") # put 111.jpg action,filename=cmd.strip().split(" ")
filesize=os.path.getsize(filename) file_info={
"action": action,
"filename": filename,
"filesize": filesize,
}
file_info_json=json.dumps(file_info).encode("utf8") ret=struct.pack("i",len(file_info_json))
# 发送 file_info_json的打包长度
sock.send(ret)
# 发送 file_info_json字节串
sock.send(file_info_json)
# 发送 文件数据
md5=hashlib.md5()
with open(filename,"rb") as f:
for line in f:
sock.send(line)
md5.update(line) data=sock.recv(1024)
print(md5.hexdigest())
md5_val=md5.hexdigest()
sock.send(md5_val.encode("utf8"))
is_valid=sock.recv(1024).decode('utf8')
if is_valid=="":
print("文件完整!")
else:
print("文件上传失败!")

client

重要模块 socketserver

  内部使用IO多路复合,以及多线程、多进程,从而实现并发处理多个客户端请求的服务器

ThreadingTCPServer

ThreadingTCPServer实现的Soket服务器内部会为每个client创建一个 “线程”,该线程用来和客户端进行交互

1、ThreadingTCPServer基础

使用ThreadingTCPServer:

  • 创建一个继承自 SocketServer.BaseRequestHandler 的类
  • 类中必须定义一个名称为 handle 的方法
  • 启动ThreadingTCPServer
import SocketServer

class MyServer(SocketServer.BaseRequestHandler):

    def handle(self):
# print self.request,self.client_address,self.server
conn = self.request
conn.sendall('欢迎致电 10086,请输入1xxx,0转人工服务.')
Flag = True
while Flag:
data = conn.recv(1024)
if data == 'exit':
Flag = False
elif data == '':
conn.sendall('通过可能会被录音.balabala一大推')
else:
conn.sendall('请重新输入.') if __name__ == '__main__':
server = SocketServer.ThreadingTCPServer(('127.0.0.1',8009),MyServer)
server.serve_forever() SocketServer实现服务器
#!/usr/bin/env python
# -*- coding:utf-8 -*- import socket ip_port = ('127.0.0.1',8009)
sk = socket.socket()
sk.connect(ip_port)
sk.settimeout(5) while True:
data = sk.recv(1024)
print 'receive:',data
inp = raw_input('please input:')
sk.sendall(inp)
if inp == 'exit':
break sk.close() 客户端

客户端

SocketServer的ThreadingTCPServer之所以可以同时处理请求得益于 select 和 os.fork 两个东西,其实本质上就是在服务器端为每一个客户端创建一个进程,当前新创建的进程用来处理对应客户端的请求,所以,可以支持同时n个客户端链接(长连接)

最新文章

  1. 使用Javascript快速获取URL参数
  2. jQuery.zTree的跳坑记录
  3. VS快捷键大全(总结了一些记忆的口诀)
  4. 见招拆招:绕过WAF继续SQL注入常用方法
  5. Dungeon Master 分类: 搜索 POJ 2015-08-09 14:25 4人阅读 评论(0) 收藏
  6. libcurl安装
  7. Qt一个project调用还有一个project的类成员变量
  8. PHP第一课笔记
  9. Android中监听ScrollView滑动停止和滑动到底部
  10. vim插件和配置
  11. 数据库中操作XML(openXML)
  12. 十八、oracle 角色
  13. 【CNMP系列】CentOS7.0下安装PHP5.6.30服务
  14. dubbo负载均衡策略及对应源码分析
  15. ZOJ 1203 Swordfish
  16. LeetCode Javascript实现 344. Reverse String 292. Nim Game 371. Sum of Two Integers
  17. npm 使用 taobao 的镜像后,无法 login &amp; publish
  18. 动态SQL2
  19. 可视化神器--Plotly
  20. 一般删除网页数据和jquery下使用Ajax删除数据的区别

热门文章

  1. 设计模式(五)Singleton模式
  2. URL中文参数,JSON转换,PHP赋值JS
  3. zabbix清理监控历史mysql数据
  4. Swift UIViewController中的delegate方式传值
  5. SpringCloud番外篇-服务治理之Nacos
  6. thinkphp两表,多表联合查询及分页的连贯操作写法
  7. RocketMQ消息轨迹-设计篇
  8. Algorithm: GCD、EXGCD、Inverse Element
  9. 全网阅读过20k的Java集合框架常见面试题总结!
  10. 关于html与css的标签及属性(text文本属性、背景background属性、表格标签table、列表、)