python实现tcp服务端和客户端互相传输文件并保存为exe文件

python实现tcp服务端和客户端互相传输文件

一、环境
首先判断主机是否在同一网络下,如果是同一网络的不同主机查看本机的ip,在cmd窗口输入ipconfig,查看以太网ipv4地址,之后ping对方的ip,如果ping不通,查看对方主机是否开启防火墙以及杀毒软件。
在这里插入图片描述

正常ping通如上图所示,关闭防火墙的方式如下:
打开控制面板–>系统和安全–>Windows Defender 防火墙–>启用或关闭Windows Defender 防火墙(在左侧栏目)将专有网络以及公用网络中的防火墙进行关闭。如下图:
在这里插入图片描述

测试完成之后在此开启即可。
这个地方需要注意的是同一网络的同一主机的ip+端口和同一网络的不同主机的ip+端口是不一样的。
二、服务端

#!coding=utf-8
import threading
import socket
import struct
import os
def socket_service():
    try:
        s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
        s.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
        # 绑定端口为9001
        # 同一网络的不同主机
        # s.bind(('192.168.0.16', 9001))
        # 同一网络同一主机
        s.bind(('127.0.0.1', 9001))
        # 设置监听数
        s.listen(10)
    except socket.error as msg:
        print(msg)
        os.sys.exit(1)
    print('等待连接...')

    while 1:
        # 等待请求并接受(程序会停留在这一旦收到连接请求即开启接受数据的线程)
        conn, addr = s.accept()
        # 接收数据
        t = threading.Thread(target=deal_data, args=(conn, addr))
        t.start()


def deal_data(conn, addr):
    print('客户端地址为: {0}'.format(addr))
    # conn.settimeout(500)
    # 收到请求后的回复
    conn.send('hi, Welcome to the server!'.encode('utf-8'))

    while 1:
        # 申请相同大小的空间存放发送过来的文件名与文件大小信息
        fileinfo_size = struct.calcsize('128sl')
        # 接收文件名与文件大小信息
        buf = conn.recv(fileinfo_size)
        # 判断是否接收到文件头信息
        if buf:
            # 获取文件名和文件大小
            filename, filesize = struct.unpack('128sl', buf)
            fn = filename.strip(b'\00')
            fn = fn.decode()
            print('数据名为 {0}, 数据大小为 {1}'.format(str(fn), filesize))

            recvd_size = 0  # 定义已接收文件的大小
            # 存储在该脚本所在目录下面
            fp = open('./' + str(fn), 'wb')

            # print(str(fn))
            print('开始接收客户端数据...')

            # 将分批次传输的二进制流依次写入到文件
            while not recvd_size == filesize:
                if filesize - recvd_size > 1024:
                    data = conn.recv(1024)
                    recvd_size += len(data)
                else:
                    data = conn.recv(filesize - recvd_size)
                    recvd_size = filesize
                fp.write(data)
            fp.close()
            print('客户端数据接收完成...')
            print('向客户端发送数据...')
            path = './'
            file_path = os.path.join(path, str(fn)).replace('\\', '/')
            fp = open(file_path, 'rb')
            fhead = struct.pack('128sl', os.path.basename(file_path).encode('utf-8'), os.stat(file_path).st_size)
            conn.send(fhead)
            while 1:
                data = fp.read(1024)
                if not data:
                    print('{0} 数据传输完成...'.format(os.path.basename(file_path)))
                    break
                conn.send(data)
        # 传输结束断开连接
        conn.close()
        break

socket_service()

三、客户端

#!coding=utf-8
import socket
import os
import sys
import struct
import cv2
def socket_client():
    try:
        s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
        s.connect(('127.0.0.1', 9001))
    except socket.error as msg:
        print(msg)
        sys.exit(1)
    print(s.recv(1024))    # while循环是为了保证能持续进行传输文件
    while True:
    # 需要传输的文件路径
        filepath = input("请输入文件路径:\n")
    #如果客户端输入的是exit,则停止传输数据并且退出程序
        if filepath == 'exit':
            break
    # 判断是否为文件
        if os.path.isfile(filepath):
        # 定义定义文件信息。128s表示文件名为128bytes长,l表示一个int或log文件类型,在此为文件大小
            fileinfo_size = struct.calcsize('128sl')
        # 定义文件头信息,包含文件名和文件大小
            fhead = struct.pack('128sl', os.path.basename(filepath).encode('utf-8'), os.stat(filepath).st_size)
        # 发送文件名称与文件大小
            s.send(fhead)

        # 将传输文件以二进制的形式分多次上传至服务器
            fp = open(filepath, 'rb')
            while 1:
                data = fp.read(1024)
                if not data:
                    print('{0} 数据传输完成...'.format(os.path.basename(filepath)))
                    break
                s.send(data)
        # 接收处理之后的数据
        print("开始接收服务端返回的数据...")
        fileinfo_size = struct.calcsize('128sl')
        buf = s.recv(fileinfo_size)

        if buf:
            filename, filesize = struct.unpack('128sl', buf)

            fn = filename.strip(b'\00')
            fn = fn.decode()
            print('数据名为 {0}, 数据大小为 {1}'.format(str(fn), filesize))
            recvd_size = 0  # 定义已接收文件的大小

            # 存储在该脚本所在目录下面
            fp = open('./' + str(fn), 'wb')
            print('保存服务器返回的数据...')
            while not recvd_size == filesize:
                if filesize - recvd_size > 1024:
                    data = s.recv(1024)
                    recvd_size += len(data)
                else:
                    data = s.recv(filesize - recvd_size)
                    recvd_size = filesize
                fp.write(data)
            print("展示服务器传输的数据...:")
            path = './'
            file_path = os.path.join(path, str(fn)).replace('\\', '/')
            img = cv2.imread(file_path)
            cv2.imshow('picture', img)
            cv2.waitKey(0)
            fp.close()
if __name__ == '__main__':
    socket_client()

四、运行结果显示
1、服务端

在这里插入图片描述
2、客户端
在这里插入图片描述
五、服务端以及客户端编译成exe 文件
具体编译过程请参考:
https://blog.csdn.net/weixin_43218120/article/details/107556151
我已经编译好的获取连接:
链接:https://pan.baidu.com/s/1y34KYaRZPPPmOAHMuDK7oA
提取码:8el3
注意:因为我在客户端使用可展示从服务器传输回来的图片的库,这就导致编译的client.exe文件较大,如果不需要可以删除,如果需要,则需要在自己的环境中包含库。
具体运行结果:
1、服务端
在这里插入图片描述
2、客户端
在这里插入图片描述

  • 3
    点赞
  • 31
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
好的,我可以为您提供一个基于 Python 的简单的 TCP 邮件传输系统的实现服务端实现: ```python import socket SERVER_HOST = '127.0.0.1' SERVER_PORT = 5000 # 创建一个 TCP 套接字 server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 将套接字绑定到指定的 IP 地址和端口号 server_socket.bind((SERVER_HOST, SERVER_PORT)) # 开始监听传入连接 server_socket.listen(1) print(f"[*] Listening on {SERVER_HOST}:{SERVER_PORT}") while True: # 等待客户端连接 client_socket, client_address = server_socket.accept() print(f"[*] Accepted connection from {client_address[0]}:{client_address[1]}") # 接收客户端发送的数据 data = client_socket.recv(1024) print(f"[*] Received data from client: {data.decode('utf-8')}") # 发送邮件数据给客户端 message = "From: you@example.com\nTo: recipient@example.com\nSubject: Test Message\n\nThis is a test message." client_socket.sendall(message.encode('utf-8')) # 关闭客户端连接套接字 client_socket.close() ``` 客户端实现: ```python import socket SERVER_HOST = '127.0.0.1' SERVER_PORT = 5000 # 创建一个 TCP 套接字 client_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 连接到服务端 client_socket.connect((SERVER_HOST, SERVER_PORT)) print(f"[*] Connected to {SERVER_HOST}:{SERVER_PORT}") # 发送数据到服务端 data = "Hello, server!" client_socket.sendall(data.encode('utf-8')) # 接收服务端发送的邮件数据 mail_data = client_socket.recv(1024) print(f"[*] Received email data from server: {mail_data.decode('utf-8')}") # 关闭客户端套接字 client_socket.close() ``` 以上代码实现了一个简单的 TCP 邮件传输系统,服务端接收客户端发送的数据,然后发送一封测试邮件给客户端客户端连接到服务端后,发送一个简单的消息,然后接收服务端发送的邮件数据。您可以根据需要修改此示例代码以满足您的实际需求。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值