网络编程 UDP协议 TCP局域网客户端与服务端上传下载电影示例

UDP协议 (了解)
称之为数据包协议,又称不可靠协议。

特点:
  1) 不需要建立链接。
  2) 不需要知道对方是否收到。
  3) 数据不安全
  4) 传输速度快
  5)能支持并发
  6) 不会粘包
  7) 无需先启动服务端再启动客户端

优点:
  - 传输速度快
  - 能支持并发
  - 不会粘包

缺点:
  - 数据不安全, 容易丢失

应用场景: 早期的QQ聊天室。

# server端
import socket

# socket.SOCK_DGRAM ---> UPD协议
server = socket.socket(type=socket.SOCK_DGRAM)
# 服务端需要绑定一个地址,让别人知道你在哪里
server.bind(
    ('127.0.0.1', 9002)
)

while True:

    # 发送数据给服务端的用户地址
    data, addr = server.recvfrom(1024)
    print(addr)
    print(data)
    # msg = input('Server ---> Client:').encode('utf-8')

    # 无论服务端还是客户端,发送消息时,都必须知道对方是谁
    # server.sendto(msg, addr)
# client端
import socket

client = socket.socket(type=socket.SOCK_DGRAM)

address = ('127.0.0.1', 9002)

while True:
    msg = input('Client ---> Server:').encode('utf-8')

    client.sendto(msg, address)

    # data, addr = client.recvfrom(1024)
    #
    # print(data)

 

- TCP协议(称为流式协议):

  优点:
    - 数据安全

  缺点:
    - 传输速度慢
    - 粘包

面试: TCP与UDP的区别, 简述优缺点即可。(*******)

 

TCP局域网客户端与服务端上传下载电影示例

文件夹位置:

'''  server
6.实现需求
    - 准备一堆电影,存放在一个文件夹中。

    - 需求:
        - 1)客户端上传到服务端
            1.让用户选择的上传的电影
            2.服务端接收电影并保存

        - 2)客户端下载服务端中的电影
            1.客户端让服务端返回可下载电影
            2.客户端选择下载的电影并下载
'''
import socket
import struct
import json
import os

movie_path = r'D:\oldboy_edu\python课堂内容\day29\1 作业讲解\server_movie'

def recv(conn):
    headers = conn.recv(4)
    bytes_len = struct.unpack('i', headers)[0]
    json_data = conn.recv(bytes_len).decode('utf8')
    back_dic = json.loads(json_data)
    print('收到服务端数据: ', back_dic)
    return back_dic

def send(msg,conn):
    json_bytes = json.dumps(msg).encode('utf8')
    headers = struct.pack('i', len(json_bytes))
    conn.send(headers)
    conn.send(json_bytes)

server = socket.socket()
server.bind(('127.0.0.1',9000))
server.listen(5)

while True:
    conn,addr = server.accept()
    while True:
        try:
            # 1、接收客户端发送的消息
            back_dic=recv(conn)
            _type = back_dic.get('type')

            if _type == 'upload':
                # 4.组织服务端存放上传电影的目录
                movie_size = back_dic.get('movie_size')
                movie_name = back_dic.get('movie_name')
                download_path = r'D:\oldboy_edu\python课堂内容\day29\1 作业讲解\server_save_movie'
                # 保存电影文件路径
                movie_download_path = os.path.join(download_path,movie_name)

                recv_data = 0
                with open(movie_download_path,'wb')as f:
                    # 一次性接收电影所有数据,有可能撑爆内存
                    # movie_data = conn.recv(movie_size)
                    # f.write(movie_data)

                    # 一点一点接收
                    while recv_data < movie_size:
                        data = conn.recv(1024)
                        f.write(data)
                        recv_data+=len(data)

            elif _type == 'download':
                movie_list = os.listdir(movie_path)
                send_dic = {'movie_list':movie_list}
                # 服务端发送给客户端的字典
                send(send_dic, conn)

                # 服务端接收客户端发送过来的字典
                back_dic=recv(conn)

                movie_name = back_dic.get('movie_name')
                # 拼接需要下载的电影文件路径
                movie_choose_path = os.path.join(movie_path,movie_name)
                movie_size= os.path.getsize(movie_choose_path)
                # 组织需要发送给客户端的电影数据
                send_dic = {'movie_size':movie_size}

                send(send_dic,conn)
                # json数据发送完毕后,紧接着发送真实电影数据
                with open(movie_choose_path,'rb')as f:
                    for line in f:
                        conn.send(line)

        except Exception as e:
            print(e)
            break
    conn.close()

 

# client
import socket
import struct
import json
import os

client = socket.socket()
client.connect(('127.0.0.1',9000))

movie_path = r'D:\oldboy_edu\python课堂内容\day29\1 作业讲解\client_movie'

def send(msg,client):
    json_bytes = json.dumps(msg).encode('utf8')
    headers = struct.pack('i', len(json_bytes))
    client.send(headers)
    client.send(json_bytes)

def recv(client):
    headers = client.recv(4)
    bytes_len = struct.unpack('i', headers)[0]
    json_data = client.recv(bytes_len).decode('utf8')
    back_dic = json.loads(json_data)
    print('收到服务端数据: ', back_dic)
    return back_dic

while True:
    # 打印功能编号,让用户选择功能
    print('''
    1.上传
    2.下载
    q.退出
    ''')
    choice = input('请输入功能编号:').strip()
    if choice == 'q':
        break
    elif choice == '1':
        # 1) 组织报头数据,并打包
        # 上传电影功能
        movie_list = os.listdir(movie_path)
        while True:
            for index,movie_name in enumerate(movie_list):
                print(index,movie_name)
            choice = input('输入需要上传的电影编号:').strip()
            if not choice.isdigit():
                continue
            choice = int(choice)
            if not choice in range(len(movie_list)):
                continue
            movie_name = movie_list[choice]
            # 拼接上传电影的路径
            upload_movie_path= os.path.join(movie_path,movie_name)
            # os.path.getsize(): 获取文件大小
            movie_size = os.path.getsize(upload_movie_path)
            # 客户端发送给服务端的字典
            send_dic = {
                'type':'upload',
                'movie_name':movie_name,
                'movie_size':movie_size
            }
            send(send_dic,client)
            # 开始发送上传的电影
            with open(upload_movie_path,'rb')as f:
                # 一次性发送: 内存资源占用过大
                # data = f.read()
                # client.send(data)

                # 一点一点发送
                for line in f:
                    client.send(line)

    elif choice == '2':
        # 下载电影功能
        # 1.客户端发送消息给服务端
        send_dic = {'type':'download'}
        send(send_dic,client)

        # 2.客户端接收服务端返回的消息
        back_dic=recv(client)
        movie_list = back_dic.get('movie_list')
        # 4.循环打印需要下载的电影列表
        while True:
            for index,movie_name in enumerate(movie_list):
                print(index,movie_name)
            choice = input('请输入上传电影编号:').strip()
            if not choice.isdigit():
                continue
            choice = int(choice)
            if choice not in range(len(movie_list)):
                continue
            movie_name = movie_list[choice]
            # 发送需要下载电影的字典给服务端
            send_dic = {'movie_name':movie_name}
            send(send_dic,client)

            # 客户端接收服务单发送过来的电影字典
            back_dic = recv(client)
            # 接收电影的数据大小
            movie_size = back_dic.get('movie_size')

            movie_save_path = os.path.join(movie_path,movie_name)

            recv_data = 0
            # 开始接收真实电影数据
            with open (movie_save_path,'wb')as f:
                while recv_data < movie_size:
                    data = client.recv(1024)
                    f.write(data)
                    recv_data += len(data)

client.close()

 

posted @ 2019-12-06 20:27  战斗小人  阅读(517)  评论(0编辑  收藏  举报