python网络编程----传输图片

1,客户端

import socket
import os
import struct
import sys


def sock_client_image():
    while True:
        try:
            s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
            # s.connect(('服务器的ip',6666)) # 服务器和客户端在不同的系统或者不同的主机下时使用的ip和端口,首先要查看服务器所在的系统网卡ip
            s.connect(('192.168.8.177', 6677))  # 服务器和客户端都在一个系统下使用的ip和端口
        except socket.error as msg:
            print(msg)
            print(sys.exit(1))
        filepath = input('input the file')  # 输入当前目录下的图片名   xxx.png
        fhead = struct.pack(b'128sq', bytes(os.path.basename(filepath), encoding='utf-8'),
                            os.stat(filepath).st_size)  # 将 xxx.jpg以128sq的格式打包
        s.send(fhead)
        fp = open(filepath, 'rb')
        while True:
            data = fp.read(1024)  # 读入图片数据
            if not data:
                print('{0} send over ....'.format(filepath))
                break
            s.send(data)
        s.close()
        # break        # 循环结束


if __name__ == '__main__':
    sock_client_image()

2,服务端建立过程

        1,利用while实现死循环(可以多次传输数据)

        2,socket创建套接字

        3,利用connect函数创建ip和端口

         4,sys.error:该属性标识的是标准错误,通常也是定向到屏幕的,可以粗糙地认为是一个输出错误信息的特殊的标准输出流。

        5,sys.exit():最终“仅”引发异常,因此只有在主线程调用时才会退出进程,并且不会截获异常,直接退出程序执行。

        6,struct.pack(fmt,v1,v2,.....)

  将v1,v2等参数的值进行一层包装,包装的方法由fmt指定。被包装的参数必须严格符合fmt。最后返回一个包装后的字符串。

        7,对文件进行操作(‘rb’:表示以二进制方式读取文件。该文件必须已存在)

        8,利用死循环对文件进行每次1024字节读取,知道所有数据读取完为止

3,服务端

import socket
import os
import sys
import struct


def socket_service_image():
    try:
        s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
        s.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
        s.bind(("192.168.9.169", 6668))
        s.listen(10)
    except socket.error as msg:
        print(msg)
        sys.exit(1)

    print("Wait for Connection.........................")

    while True:
        sock, addr = s.accept()
        deal_image(sock, addr)


def deal_image(sock, addr):
    print("Accept connection from {0}".format(addr))

    while True:
        fileinfo_size = struct.calcsize('128sq')
        buf = sock.recv(fileinfo_size)
        if buf:
            filename, filesize = struct.unpack('128sq', buf)
            fn = filename.decode().strip('\x00')
            new_filename = os.path.join('../pass--photo_file/',
                                        'new_' + fn)
            recvd_size = 0
            fp = open(new_filename, 'wb')
            while not recvd_size == filesize:
                if filesize - recvd_size > 1024:
                    data = sock.recv(1024)
                    recvd_size += len(data)
                else:
                    data = sock.recv(1024)
                    recvd_size = filesize
                fp.write(data)
            fp.close()
        sock.close()
        break


if __name__ == '__main__':
    socket_service_image()
4,服务器的创建过程

        1,cocket:创建套接字

        2,setsockopt

                有三个参数:

                level:选项定义的层次。支持SOL_SOCKET、IPPROTO_TCP、IPPROTO_IP和IPPROTO_IPV6。

                optname:需设置的选项。

                value:设置选项的值。

        3,bind:建立ip地址和接口

        4,listen:

                socket.listen(n) 简单来说,这里的nt表示socket的”排队个数“

                一般情况下,一个进程只有一个主线程(也就是单线程),那么socket允许的最大连接数为: n + 1
                如果服务器是多线程,比如上面的代码例子是开了2个线程,那么socket允许的最大连接数就是: n + 2

        5,异常同客户端

        6,死循环 + accept()函数循环获取所接受的数据

        7,struct.calcsize('128sq'):

             truct模块提供了一种机制来处理C中的字节级别数据,包括各种数据类型的打包和解包操作。其中,struct.calcsize()函数用于计算给定格式字符串所代表的C结构体的大小(以字节为单位)。

                “1”:表示一个字节大小的无符号字符。

                “2”:表示两个字节大小的无符号短整数。

                “8s”:表示一个长度为8字节的字符串。

        8,strip('内容'):去除指定的内容

        

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值