zoukankan      html  css  js  c++  java
  • 【Day29 】Soket编程

    • 客户端/服务器架构

        1、什么是客户端服务器/服务器架构??

        服务器:意义--就是一系列软硬件的结合,为一个或多个客户端提供服务。目的:接受请求并响应,然后处理更多请求。

        客户端:发送请求,并接收信息,最后关闭他们之间的事务。

    • osi七层协议

         1、互联网层的协议分为osi七层tcp/ip五层或四层

      

     每层运行常见物理设备

    • 什么是网络?

        网络是底层的物理链接介质

    • 2.1 物理层

        物理层功能:主要是基于电器特性发送高低电压(电信号),高电压对应数字1,低电压对应数字0

    • 2.2数据链路层

        其功能:定义了电信号的分组方式

        以太网协议(ethernet)规定:

           每一组数据包含head/data ,

            head(18个字节):源地址6个,目标地址6个,数据类型6个

            data(最小46,最大1500):数据包的具体内容

          head + data 最大长度1518,最短长度64,超过长度分片发送

      mac地址:每个电脑都有唯一的mac地址且与ip地址绑定

      广播:有了mac地址,两台电脑可以通过arp协议进行通信,

      enternet 采用广播方式通信,即基本靠吼

    • 2.3 网络层

        1、网络层的意义:用来区分不同的广播域/子网,即网络地址

        2、规定网络地址的协议成为ip协议

        一、ip地址分为:网络部分-->标识子网,主机部分--->标识主机

          注意:单纯的ip地址段只是标识了ip地址的种类,从网络部分或主机部分都无法辨识一个ip所处的子网

          例:172.16.10.1与172.16.10.2并不能确定二者处于同一子网

        二、子网掩码(ip网络部分):

          所谓”子网掩码”就是表示子网络特征的一个参数。它在形式上等同于IP地址,也是一个32位二进制数字,它的网络部分全部为1,      主机部分全部为0。比如,IP地址172.16.10.1,如果已知网络部分是前24位,主机部分是后8位,那么子网络掩码就是            11111111.11111111.11111111.00000000,写成十进制就是255.255.255.0。

          知道”子网掩码”,我们就能判断,任意两个IP地址是否处在同一个子网络。方法是将两个IP地址与子网掩码分别进行AND运算(两个      数位都为1,运算结果为1,否则为0),然后比较结果是否相同,如果是的话,就表明它们在同一个子网络中,否则就不是。

          比如,已知IP地址172.16.10.1和172.16.10.2的子网掩码都是255.255.255.0,请问它们是否在同一个子网络?两者与子网掩码分别      进行AND运算,

          172.16.10.1:10101100.00010000.00001010.000000001

          255255.255.255.0:11111111.11111111.11111111.00000000

          AND运算得网络地址结果:10101100.00010000.00001010.000000001->172.16.10.0

          172.16.10.2:10101100.00010000.00001010.000000010

          255255.255.255.0:11111111.11111111.11111111.00000000

          AND运算得网络地址结果:10101100.00010000.00001010.000000001->172.16.10.0

          结果都是172.16.10.0,因此它们在同一个子网络。

          总结一下,IP协议的作用主要有两个,一个是为每一台计算机分配IP地址,另一个是确定哪些地址在同一个子网络。

       三、ip数据包

          ip数据包分为head,data部分,无须定义专门的栏目,直接放到以太网的data部分

          ip数据包大小:

              head:长度为20到60字节

              data:最长为65,515字节。

          以太网的数据包最大1500,如果ip数据包超过1500字节,就需要封装成几个以太网数据包分开发送

      四、ARP协议

          arp协议功能:广播的方式发送数据包,获取目标主机的mac地址

       流程:如果两台电脑互相访问(172.168.0.2访问172.168.0.3):首先通过ip地址和子网掩码判断是否在同一子网——>如果在同一子网会以广播的方式在局域网内传播,---->如果不在同一子网会通过 数据包中的目标ip地址 和ARP协议获取网关的mac地址---然后在再子网内进行广播的方式发送,主机拆开包后,发现ip地址是自己的,就响应返回自己的mac地址

    • 2.4 传输层  

        1.传输层的由来:网络层的ip帮我们区分子网,以太网层的mac帮我们找到主机,端口找到应用程序,端口即应用程序与网卡关联的编      号。

        2.传输层功能:建立端口到端口的通信

        3.ip 加mac + 端口 :标识唯一的软件

        4.可靠传输,TCP数据包没有长度限制,理论上可以无限长,但是为了保证网络的效率,通常TCP数据包的长度不会超过IP数据包的长      度,以确保单个TCP数据包不必再分割。

    • 2.5应用层

         应用层由来:使用的程序都是应用层,各种应用程序规定好数据的组织形式

        应用层的功能:规定应用层数据的格式

    • soket

        1.socket 是什么:soket 是应用层与Tcp/ip 通信的中间软件抽象层,他是一组接口

         socket:绑定地址的时候,网端必须是整数,最好大于1024 ,小于65535

        socket :进行传送时都是使用bytes类型

    #服务端
    import socket
    import time
    my_server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    my_server.bind(('127.0.0.1',8080))
    my_server.listen(5)
    tcp_sock, addr = my_server.accept()
    res1 = tcp_sock.recv(1)
    res2 = tcp_sock.recv(4)
    print(res1) print(res2)
    #客户端
    import socket
    import time
    
    
    client = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    client.connect(('127.0.0.1', 8080))
    client.send(bytes('hello','utf-8'))
    time.sleep(1)
    client.send(bytes('world','utf-8'))

    以上两段代码:服务端的输出结果,

    #服务端
    b'h'
    b'ello'

    原因为:客户端,发送完第一条信息(‘hello’)后,中间滞留一秒的时间。此时,这条信息已经通过tcp/ip协议传输到服务端电脑,服务端socket通过os.模块从内存中加读取刚刚传送过来的信息,由于服务端第一次只接收一个字节的信息,剩余的字节还停留在内存中,到了第二次接收的时候,才把内存中的信息读取完。

    • 如何解决粘包问题
    #客户端
    import socket
    import json
    import subprocess
    import struct
    
    
    sock_server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    sock_server.bind(('127.0.0.1',8080))
    sock_server.listen(5)
    while True:
        conn, adrr = sock_server.accept()
        while True:
            try:
                cmd_data = conn.recv(1024)
                print(cmd_data)
                if not cmd_data: break
                cmd_data = cmd_data.decode("utf-8")
                sub_data = subprocess.Popen(cmd_data, shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
                cmd_data_stdout = sub_data.stdout.read()
                cmd_data_stderr = sub_data.stderr.read()
                # heard_json = json.dumps(sub_data)
                # print(111)
                # heard_bytes = bytes(heard_json, 'utf-8')
                heard_len = struct.pack('i', len(cmd_data_stdout)+len(cmd_data_stderr))
                conn.send(heard_len)
                conn.send(cmd_data_stdout+cmd_data_stderr)
            except ConnectionError:
                break
        conn.close()
    sock_server.close()
    
    # import socket
    import json
    import subprocess
    import struct
    
    
    sock_server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    sock_server.bind(('127.0.0.1',8080))
    sock_server.listen(5)
    while True:
        conn, adrr = sock_server.accept()
        while True:
            try:
                cmd_data = conn.recv(1024)
                print(cmd_data)
                if not cmd_data: break
                cmd_data = cmd_data.decode("utf-8")
                sub_data = subprocess.Popen(cmd_data, shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
                cmd_data_stdout = sub_data.stdout.read()
                cmd_data_stderr = sub_data.stderr.read()
                # heard_json = json.dumps(sub_data)
                # print(111)
                # heard_bytes = bytes(heard_json, 'utf-8')
                heard_len = struct.pack('i', len(cmd_data_stdout)+len(cmd_data_stderr))
                conn.send(heard_len)
                conn.send(cmd_data_stdout+cmd_data_stderr)
            except ConnectionError:
                break
        conn.close()
    sock_server.close()
    
    #服务端
    import socket
    import json
    import struct
    
    
    cline = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    cline.connect(('127.0.0.1', 8080))
    while True:
        user_input = input(">>>").strip()
        if not user_input: continue
        cline.send(bytes(user_input,"utf-8"))
        server_total = cline.recv(4)
    
        head_total_len = struct.unpack('i', server_total)[0]
        print(head_total_len)
        ser_how = 0
        ser_data = b''
        while ser_how < head_total_len:
            data = cline.recv(1024)
            ser_how += len(data)
            ser_data += data
    
        print(ser_how)
    
        print(ser_data.decode("gbk"))
    远程操控 cmd 解决粘包
    #服务端
    import socket
    import json
    import struct
    import subprocess
    import os
    import hashlib
    
    
    servers = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    servers.bind(('127.0.0.1',8080))
    servers.listen(5)
    def get_func(filename):
        md5_hs = hashlib.md5()
        heard_total = {'filename':filename,
                       "len_size":os.path.getsize(r"%s"%filename),
                       "article_md5": md5_hs.hexdigest()
                       }
        heard_json = json.dumps(heard_total)
        heard_bytes = heard_json.encode('utf-8')
        heard_size = struct.pack('i',len(heard_bytes))
        conn.send(heard_size)
        conn.send(heard_bytes)
        with open(r"%s"% filename, 'rb')as f:
            for lien in f:
                md5_hs.update(lien)
                conn.send(lien)
    while True:
        conn, addr = servers.accept()
        while True:
            try:
                data = conn.recv(1024)
                if not data:break
                data_str = data.decode('utf-8')
                print(type(data_str))
                cmd, filename = data_str.split()
                print(filename)
                if cmd == 'get':
                    get_func(filename)
            except ConnectionError:
                break
    
    #客户端
    import socket
    import struct
    import json
    import hashlib
    import time
    
    
    client = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    client.connect(('127.0.0.1',8080))
    while True:
        user_input = input(">>>").strip()
        if not user_input:continue
        user_input = user_input.encode("utf-8")
        client.send(user_input)
        server_data = client.recv(4)
        heard_size = struct.unpack('i',server_data)[0]
        heard_bytes = client.recv(heard_size)
        heard_json = heard_bytes.decode("utf-8")
        heard_dic = json.loads(heard_json)
        print(heard_dic)
    
        file_size = heard_dic['len_size']
        file_name = heard_dic['filename']
        article_md5 = heard_dic['article_md5']
        ser_size = 0
        hash_obj = hashlib.md5()
        print(article_md5)
        print(hash_obj.hexdigest())
        # while ser_size < file_size:
        #     time.sleep(0.1)
        #     char_num = ser_size//file_size
        #     pre_str = '
    %s%%:%s'%(100,'|'*100)if ser_size == file_size else "
    %s%%:%s"%(char_num,"|"*char_num)
        #     print(pre_str, end='', flush=True)
        with open(r'E:资料day30sokect粘包	o.avi','wb')as f:
            while ser_size < file_size:
                data = client.recv(1024)
                # print(data)
                hash_obj.update(data)
                f.write(data)
                ser_size += len(data)
    远程下载
  • 相关阅读:
    8. Automatic Properties(自动属性)
    egg文件安装
    [翻译]深入理解Tornado——一个异步web服务器
    tornado模板的自动编码问题(autoescape )
    PIL的IOError: decoder jpeg not available错误的排除方法
    MongoDB的更新问题
    easy_install
    _imaging.c:75:20: 致命错误: Python.h:没有那个文件或目录
    各类情感的能量等级&大自然与人类能量级别的关系(转)
    python的装饰器(property)
  • 原文地址:https://www.cnblogs.com/huyangblog/p/7904803.html
Copyright © 2011-2022 走看看