zoukankan      html  css  js  c++  java
  • HTTP手记

    ---------------------tcp/ip模型和osi模型---------------------
    tcp/ip协议模型   osi模型
    应用层       应用层
            表示层
            会话层
    传输层       传输层
    网络层       网络层
    链路层       数据链路层
            物理层

    ---------------------交换机---------------------
    1、网络交换机介绍:
    网络交换机(又称“网络交换器”),是一个扩大网络的器材,能为子网络中提供更多的连接端口,以便连接更多的计算机 具有性能价格比高、高度灵活、相对简单、易于实现等特点 以太网技术已成为当今最重要的一种局域网组网技术,网络交换机也就成为了最普及的交换机

    2、交换机的作用:
    转发过滤:当一个数据帧的目的地址在MAC地址表中有映射时,它被转发到连接目的节点的端口而不是所有端口(如该数据帧为广播帧则转发至所有端口)
    学习功能:以太网交换机了解每一端口相连设备的MAC地址,并将地址同相应的端口映射起来存放在交换机缓存中的MAC地址表中

    ---------------------路由器---------------------
    1. 路由器简介
    路由器(Router)又称网关设备(Gateway)是用于连接多个逻辑上分开的网络

    所谓逻辑网络是代表一个单独的网络或者一个子网。当数据从一个子网传输到另一个子网时,可通过路由器的路由功能来完成

    具有判断网络地址和选择IP路径的功能

    不在同一网段的pc,需要设置默认网关才能把数据传送过去 通常情况下,都会把路由器默认网关
    当路由器收到一个其它网段的数据包时,会根据“路由表”来决定,把此数据包发送到哪个端口;路由表的设定有静态和动态方法

    每经过一次路由器,那么TTL值就会减一

    ---------------------模拟进行客户端、DNS服务器、HTTP服务器实现---------------------


    ---------------------HTTP请求的流程---------------------
    步骤1:浏览器首先向服务器发送HTTP请求

    方法:GET还是POST,GET仅请求资源,POST会附带用户数据;

    路径:/full/url/path;

    域名:由Host头指定:Host: www.xxx.com

    以及其他相关的Header;

    如果是POST,那么请求还包括一个Body,包含用户数据


    步骤2:服务器向浏览器返回HTTP响应

    响应代码:200表示成功,3xx表示重定向,4xx表示客户端发送的请求有错误,5xx表示服务器端处理时发生了错误;

    响应类型:由Content-Type指定;

    以及其他相关的Header;

    相应长度:由Content-Length:755

    通常服务器的HTTP响应会携带内容,也就是有一个Body,包含响应的内容,网页的HTML源码就在Body中。


    步骤3:如果浏览器还需要继续向服务器请求其他资源,比如图片,就再次发出HTTP请求
    Web采用的HTTP协议采用了非常简单的请求-响应模式,从而大大简化了开发。当我们编写一个页面时,我们只需要在HTTP请求中把HTML发送出去,不需要考虑如何附带图片、视频等,浏览器如果需要请求图片和视频,它会发送另一个HTTP请求,因此,一个HTTP请求只处理一个资源(此时就可以理解为TCP协议中的短连接,每个链接只获取一个资源,如需要多个就需要建立多个链接)

    ---------------------tcp协议三次握手和四次挥手---------------------

    三次握手

    四次握手

    ---------------------epoll和进程结合模拟实现一个简单的http服务器---------------------

      1 #coding=utf-8
      2 
      3 #引用对应的数据包
      4 from socket import *
      5 
      6 import sys
      7 
      8 import re
      9 
     10 import select
     11 
     12 import multiprocessing
     13 
     14 from time import sleep
     15 
     16 #创建全局变量
     17 #创建一个epoll对象
     18 epoll=select.epoll()
     19 
     20 #创建两个字典
     21 #字典1:根据套接字对应的文件标识符对应的套件字
     22 connection={}
     23 
     24 #字典2:根据套建字对应的文件表示符对应的ip和端口元祖信息
     25 address={}
     26 
     27 
     28 #运行进程
     29 def runProc(recvData,soc,addr,fd):
     30     #引用全局变量进行处理
     31     global epoll
     32     global connection
     33     global address
     34     
     35     #使用正则,获取到对应的要读取的文件名
     36     fileName=re.search("GET (.*) HTTP",recvData).group(1)
     37     
     38     print(fileName)
     39 
     40     #调用函数获取到处理后的地址
     41     fileName=getPath(fileName)
     42 
     43     #获取文件后缀名,进行相应头文件的准备
     44     fileExt=fileName[fileName.rindex('.')+1:]
     45 
     46     print ("02-文件的后缀名--:"+fileExt)
     47     
     48     #存储响应报文
     49     httpHead=''
     50 
     51     if fileExt.lower() in ['html','css']:
     52         #制定文件的读取操作
     53         fil=open(fileName,'r')
     54 
     55         #读取制定文件中的数据信息
     56         htmlText=fil.read()
     57 
     58         #关闭对应的文件对象
     59         fil.close()
     60 
     61         #创建变量:存储有效的响应头文件
     62         httpHead="HTTP/1.1 200 OK 
     Content-Type:text/"+fileExt.lower()+" 
    
    "
     63         #进行数据的回复操作
     64         soc.send(httpHead+htmlText)
     65 
     66         #将该套接字从epoll中注销
     67         epoll.unregister(fd)
     68 
     69         #从字典中进行排除
     70         del connection[fd]
     71         del address[fd]
     72 
     73         soc.close()
     74 
     75         print("%s-------end"%fileExt)
     76 
     77     elif fileExt.lower() in ['png','jpg','gif']:
     78         #制定文件的读取操作
     79         fil=open(fileName,'rb')
     80 
     81         #读取制定文件中的数据信息
     82         htmlText=fil.read()
     83 
     84         #关闭对应的文件对象
     85         fil.close()
     86         httpHead="HTTP/1.1 200 ok 
     Content-Type:image/"+fileExt.lower()+" 
    
    "
     87         
     88         #print(fileExt.lower())
     89         #print(htmlText)
     90         
     91         #sleep(2)
     92 
     93         #进行数据的回复操作
     94         soc.send(httpHead+htmlText)
     95 
     96         #将该套接字从epoll中注销
     97         epoll.unregister(fd)
     98 
     99         #从字典中进行排除
    100         del connection[fd]
    101         del address[fd]
    102 
    103         soc.close()
    104 
    105         print("%s------end"%fileExt.lower())
    106 
    107 
    108 #函数:文件路径
    109 def getPath(fileName):
    110     #变量存储文件地址;
    111     filePath=''
    112     
    113     print(fileName)
    114 
    115     #根目录,默认指向./html/Index.html
    116     if fileName[0:4]=='/html':
    117         filePath='.'+fileName                    
    118     else:
    119         filePath='./html'+fileName
    120     
    121     print("01-获取到对应的文件路径--:%s"%filePath)
    122 
    123     return filePath
    124 
    125 #主函数
    126 def main():
    127     #引用全局变量进行处理
    128     global epoll
    129     global connection
    130     global address
    131     
    132     #创建服务器套接字
    133     tcpSerSoc=socket(AF_INET,SOCK_STREAM)
    134 
    135     tcpSerSoc.setsockopt(SOL_SOCKET, SO_REUSEADDR,1)
    136 
    137     #进行对应数据的绑定
    138     tcpSerSoc.bind(("",int(sys.argv[1])))
    139 
    140     #tcp服务器开启被动监听状态
    141     tcpSerSoc.listen(10)
    142 
    143     #使用epoll对服务器套接字在操作系统中进行注册
    144     epoll.register(tcpSerSoc.fileno(),select.EPOLLIN|select.EPOLLET)
    145 
    146     #提示开启对应的服务
    147     print("-----开启对应的HTTP服务-----")
    148 
    149     #循环,对客户端传递过来的数据进行处理
    150     while True:
    151         #通过操作系统获取到要进行数据处理的套接字
    152         epollList=epoll.poll()
    153 
    154         #遍历所有的套接字进行相应的处理
    155         for fd,event in epollList:
    156             #判断是否为服务器的套接字
    157             if fd==tcpSerSoc.fileno():        
    158                 #接收客户端对应的数据信息
    159                 newSocket,destAddr= tcpSerSoc.accept()
    160     
    161                 print("客户端(%s)以接入HTTP服务器"%str(destAddr))
    162                 
    163                 #将对应的数据向字典中进行存储
    164                 connection[newSocket.fileno()]=newSocket
    165                 address[newSocket.fileno()]=destAddr
    166 
    167                 #将新的套接字通过epoll向操作系统中进行注册
    168                 epoll.register(newSocket.fileno(),select.EPOLLIN|select.EPOLLET)
    169 
    170             elif event==select.EPOLLIN:
    171                 #拿到对应的套接字和ip端口
    172                 soc=connection[fd]
    173                 addr=address[fd]
    174 
    175                 #进行客户端数据对应的接收操作
    176                 recvData=soc.recv(1024)
    177 
    178                 #判断接受的数据是否为空:如果为空表示客户下线
    179                 if len(recvData)>0:
    180                     #启动一个新的socked进程
    181                     p=multiprocessing.Process(target=runProc,args=(recvData,soc,addr,fd))
    182                     p.start()
    183                     p.join()
    184                     
    185                     soc.close()
    186                 else:
    187                     
    188                     print("客户(%s)以离开"%str(addr))
    189 
    190                     #将该套接字从epoll中注销
    191                     epoll.unregister(fd)
    192 
    193                     soc= connection[fd]
    194 
    195                     #从字典中进行排除
    196                     del connection[fd]
    197                     del address[fd]
    198 
    199                     #关闭该套接字
    200                     soc.close()
    201                                     
    202     #关闭服务器套接字
    203     tcpSerSoc.close()
    204 
    205 
    206 #程序入口
    207 if __name__=='__main__':
    208     main()
    epoll和进程实现简单的HTTP服务器
  • 相关阅读:
    学习进度笔记
    学习进度笔记
    学习进度笔记
    学习进度笔记
    博雅数据机器学习10
    学习进度笔记
    HDFS上文件权限操作
    HBase的安装与使用
    hadoop完全分布式安装教程
    python安装easygui
  • 原文地址:https://www.cnblogs.com/qingtianyu2015/p/5858197.html
Copyright © 2011-2022 走看看