基于tcp协议的登录,文件上传和下载
【1】先登录,登录不成功循环登录,直到成功。登录成功后可以选择上传或者下载,上传有对应的文件,可选择上传哪个;下载有对应的文件,可选择下载哪个
【2】登录,上传,下载时最好设置状态码,客户端和服务端约定好状态码,操作成功或者失败后,发送对应的状态码。注意,每一个操作要有唯一的状态码。
比如: 登录成功 100 登录失败 101
注册成功 102 注册失败 103
上传成功 104 上传失败 105
下载成功 106 下载失败 107
【3】也可以用字典: dic = {'operate': 'login', 'result': res} res= True 或 False 表明要进行的操作,以及状态。比较直观
【4】网络之间进行数据传递,最常用的就是json模块。传送一个字典
【5】编程尽量少使用面向过程编程,多使用 函数 + 面向对象 编程
【6】函数中都要用传参的方式,传入conn,sk等,不要用全局变量,便于文件拆分,进行进一步封装。不用全局变量 ,因为将来文件拆分后,全局变量就不能用了。
【7】输入时,容易出错的地方,要进行异常处理(进一步优化)
#server
import os
import sys
import json
import struct
import socket
import hashlib
def my_send(conn,dic): #用于发送要 被下载的文件的信息(字典{'filename': filename, 'filesize': filesize});可下载文件列表
str_dic = json.dumps(dic)
b_dic = str_dic.encode('utf-8')
mlen = struct.pack('i', len(b_dic))
conn.send(mlen) # 4个字节 表示字典/列表转成字节之后的长度
conn.send(b_dic) # 具体的字典数据
def my_recv(conn): #用于接收将要上传文件的信息(字典形式)
msg_len = conn.recv(4)
dic_len = struct.unpack('i', msg_len)[0]
msg = conn.recv(dic_len).decode('utf-8')
msg = json.loads(msg)
return msg
def download(conn):
g = os.walk(r'D:\python22\day30\下载文件')
file_lst = next(g)[2] # ['11.jpg', '23.mp4', 'userinfo']
my_send(conn,file_lst) #将文件列表发送到客户端
while 1:
num = conn.recv(1024).decode('utf-8') # 选择要下载文件的序号
if num.upper() == 'Q':break
else:
path = r'D:\python22\day30\下载文件'
abs_path = os.path.join(path,file_lst[int(num) - 1])
filename = os.path.basename(abs_path)
filesize = os.path.getsize(abs_path)
dic = {'filename': filename, 'filesize': filesize}
my_send(conn,dic) #发送用户选择要下载文件的信息,是一个字典
with open(abs_path, mode='rb') as f:
while filesize > 0:
content = f.read(1024)
filesize -= len(content)
conn.send(content)
def upload(conn):
while 1:
msg = my_recv(conn)
if msg['filename'] == None:break
else:
with open(msg['filename'],'wb') as f:
while msg['filesize'] > 0:
content = conn.recv(1024)
msg['filesize'] -= len(content)
f.write(content)
conn.send('上传成功'.encode('utf-8'))
def get_md5(username,password):
md5 = hashlib.md5(username.encode('utf-8'))
md5.update(password.encode('utf-8'))
return md5.hexdigest()
def login(conn):
flag = True#用于跳出外层循环
while flag: #注意跳出多级循环问题
msg = my_recv(conn)
with open('userinfo') as f:
for line in f:
name, pwd = line.strip().split('|')
if name == msg['username'] and pwd == get_md5(name, msg['password']):
res, flag = True, False
break #跳出本级循环
else:
res = False
dic = {'operate': 'login', 'result': res} #状态码
my_send(conn, dic)
#主逻辑
sk = socket.socket()
sk.bind(('127.0.0.1',9002))
sk.listen()
conn,_ =sk.accept()
login(conn)
opt_dic = my_recv(conn)
if hasattr(sys.modules[__name__],opt_dic['operate']): # __name__是变量名,以脚本运行时,对应的值为'__main__'。但是被当做模块导入时,名字为模块名。所以最好用__name__
getattr(sys.modules[__name__],opt_dic['operate'])(conn)
conn.close()
sk.close()
#client
import os
import sys
import json
import struct
import socket
def download(sk): # 下载
opt_dic = {'operate':'download'}
my_send(sk,opt_dic) #将要进行的操作发送到服务端
file_lst = my_recv(sk) #接收到来自服务端的文件列表
while 1:
for index, file in enumerate(file_lst, 1):
print(index,file)
num = input('请输入要下载文件的序号(退出请按Q):').strip()
if num.upper() == 'Q':
sk.send(num.encode('utf-8')) # 发送q或Q
break
elif num.isdigit() and 0 < int(num) <= len(file_lst):
sk.send(num.encode('utf-8')) # 发送要下载文件的序号
msg = my_recv(sk) #接收要下载文件的信息,是一个字典
with open(msg['filename'], 'wb') as f:
while msg['filesize'] > 0:
content = sk.recv(1024)
msg['filesize'] -= len(content)
f.write(content)
print('下载成功')
else:
print('输入有误')
def upload(sk):
opt_dic = {'operate': 'upload'}
my_send(sk, opt_dic)
g = os.walk(r'D:\python22\day30\上传文件')
file_lst = next(g)[2] # ['11.jpg', '23.mp4', 'userinfo']
while 1:
for index, file in enumerate(file_lst, 1):
print(index, file)
num = input('请输入要上传文件的序号(退出请按Q):').strip()
if num.upper() == 'Q':
dic = {'filename': None, 'filesize': None}
my_send(sk, dic)
break
elif num.isdigit() and 0 < int(num) <= len(file_lst):
path = r'D:\python22\day30\上传文件'
abs_path = os.path.join(path, file_lst[int(num) - 1])
filename = os.path.basename(abs_path)
filesize = os.path.getsize(abs_path)
dic = {'filename': filename, 'filesize': filesize}
my_send(sk,dic)
with open(abs_path, mode='rb') as f:
while filesize > 0:
content = f.read(1024)
filesize -= len(content)
sk.send(content)
msg = sk.recv(1024).decode('utf-8')
print(msg)
def my_recv(sk): # 接收
msg_len = sk.recv(4)
dic_len = struct.unpack('i', msg_len)[0]
msg = sk.recv(dic_len).decode('utf-8')
msg = json.loads(msg)
return msg
def my_send(sk,dic): # 发送
str_dic = json.dumps(dic)
b_dic = str_dic.encode('utf-8')
mlen = struct.pack('i', len(b_dic))
sk.send(mlen) # 4个字节 表示字典转成字节之后的长度
sk.send(b_dic) # 具体的字典数据
def login(sk):
while True: #alex|aee949757a2e698417463d47acac93df
usr = input('用户名:').strip() #alex
pwd = input('密 码 :').strip() #3714
dic = {'username': usr, 'password': pwd}
my_send(sk, dic)
ret = my_recv(sk) #dic = {'operate': 'login', 'result': res} res=True 或 False
if ret['operate'] == 'login' and ret['result']:
print('登录成功')
break
else:
print('登录失败')
#主逻辑
sk = socket.socket()
sk.connect(('127.0.0.1',9002))
login(sk) # 登录
opt_lst = ['upload','download']
for index,opt in enumerate(opt_lst,1):
print(index,opt)
num = int(input('请选择您要操作的序号 :'))
getattr(sys.modules[__name__],opt_lst[num-1])(sk)
sk.close()
基于tcp协议的登录,文件上传和下载的更多相关文章
- 网络编程应用:基于TCP协议【实现文件上传】--练习
要求: 基于TCP协议实现一个向服务器端上传文件的功能 客户端代码: package Homework2; import java.io.File; import java.io.FileInputS ...
- 转:【专题十一】实现一个基于FTP协议的程序——文件上传下载器
引言: 在这个专题将为大家揭开下FTP这个协议的面纱,其实学习知识和生活中的例子都是很相通的,就拿这个专题来说,要了解FTP协议然后根据FTP协议实现一个文件下载器,就和和追MM是差不多的过程的,相信 ...
- 专题十一:实现一个基于FTP协议的程序——文件上传下载器
引言: 在这个专题将为大家揭开下FTP这个协议的面纱,其实学习知识和生活中的例子都是很相通的,就拿这个专题来说,要了解FTP协议然后根据FTP协议实现一个文件下载器,就和和追MM是差不多的过程的,相信 ...
- 实现一个基于FTP协议的程序——文件上传下载器(十三)
此为一个系列,后续会把内容补上...
- koa2基于stream(流)进行文件上传和下载
阅读目录 一:上传文件(包括单个文件或多个文件上传) 二:下载文件 回到顶部 一:上传文件(包括单个文件或多个文件上传) 在之前一篇文章,我们了解到nodejs中的流的概念,也了解到了使用流的优点,具 ...
- 基于 Django的Ajax实现 文件上传
---------------------------------------------------------------遇到困难的时候,勇敢一点,找同学朋友帮忙,找导师求助. Ajax Ajax ...
- Resumable.js – 基于 HTML5 File API 的文件上传
Resumable.js 是一个 JavaScript 库,通过 HTML5 文件 API 提供,稳定和可恢复的批量上传功能.在上传大文件的时候通过每个文件分割成小块,每块在上传失败的时候,上传会不断 ...
- 基于jsp的文件上传和下载
参考: 一.JavaWeb学习总结(五十)--文件上传和下载 此文极好,不过有几点要注意: 1.直接按照作者的代码极有可能listfile.jsp文件中 <%@taglib prefix=&qu ...
- 基于hap的文件上传和下载
序言 现在,绝大部分的应用程序在很多的情况下都需要使用到文件上传与下载的功能,在本文中结合hap利用spirng mvc实现文件的上传和下载,包括上传下载图片.上传下载文档.前端所使用的技术不限,本文 ...
随机推荐
- 设置windows2008系统缓存大小限制,解决服务器运行久了因物理内存耗尽出僵死(提升权限后,使用SetSystemFileCacheSize API函数,并将此做成了一个Service)
声明: 找到服务器僵死的原因了,原因是虚拟内存设置小于物理内存. 只要虚拟内存设置为系统默认大小就不会出生僵死的现象了. 当时因为服务器内存48G,系统默认虚拟内存大小也是48G, 觉得太占硬盘空间, ...
- QT---Winsocket获取网关(Gateway) 主机IP等信息
基于WinPcap库做开发,需要利用到局域网的默认网关地址和Mac地址,但是WinPcap实现获取网关IP地址没有很好的思路,可以知道的是网关的接收和发出的数据包数量一般是比局域网内的各主机要多的 ...
- Delphi中,indy控件实现收发邮件的几点学习记录( 可以考虑加入多线程,用多个邮箱做一个邮箱群发器) 转
关于用Delphi中的Indy控件实现收发邮件的几点学习记录 这几天心里颇不宁静,不是因为项目延期,而是因为自己几个月前做的邮件发送程序至今无任何进展,虽然一向谦虚的人在网上发 ...
- foreach() 中用指针指向数组元素,循环结束后最好销毁指针
之前发过一次微博,今天又遇到这个问题,并且再次犯错,于是决定再加深一下. 就举php.net里的一个例子吧 $a = array('abe','ben','cam'); foreach ($a as ...
- Cloudera Impala需求
Cloudera Impala需求 为了达到预期的效果,Impala依赖于软件.硬件的可用性,以及下面章节描述的配置. 继续阅读: 支持的操作系统 支持的Hadoop发布 Hive Metastore ...
- 30411MySQL安装与配置_win10
1 下载 1.1下载地址 下载地址 https://downloads.mysql.com/archives/community/ 1.2 选择适合自己的版本并下载 1.3 将下载文件解压至自定义路 ...
- 使用fastjson读取超巨json文件引起的GC问题
项目中需要将巨量数据生成的json文件解析,并写入数据库,使用了 alibaba 的 fastjson,在实践过程中遇到了 GC 问题,记录如下: 数据大约为70万条,文件大小在3~4G左右,使用 f ...
- Java入门网络编程-使用UDP通信
程序说明: 以下代码,利用java的网络编程,使用UDP通信作为通信协议,描述了一个简易的多人聊天程序,此程序可以使用公网或者是局域网进行聊天,要求有一台服务器.程序一共分为2个包,第一个包:udp, ...
- idea提交代码到自己git账号的master branch
1.注册GitHub账号 2.创建本地密钥与远程仓库连接(使用idea的话,这第二步可能不是必须的,但是密钥我之前配置过.所以写下来) ①安装git 客户端sudo apt-get install g ...
- K Balanced Teams CodeForces - 1133E (Dp)
题意: 给出 n 个数,选取其中若干个数分别组成至多 k 组,要求每组内最大值与最小值的差值不超过5,求最后被选上的总人数. 题解: 将a[1∼n] 从小到大排序, f[i][j] 表示到第 i 个数 ...