zoukankan      html  css  js  c++  java
  • python I/O多路复用 使用http完成http请求

    1. 使用类实现比较方便我们使用里面的参数

    2. 我们使用selector,不适用select

    from selectors import DefaultSelector

    3. I/O多路复用是指使用 回调+事件循环+select(pollepoll)

      a. 使用selector注册,并注册回调函数

      b. 使用事件循环一直循环,查询状态

      c. 使用select调用相应的回调函数

    import socket
    from urllib.parse import urlparse
    from selectors import DefaultSelector, EVENT_READ, EVENT_WRITE
    
    
    selector = DefaultSelector()
    #使用select完成http请求
    urls = []
    stop = False
    
    
    class Fetcher:
        def connected(self, key):
            selector.unregister(key.fd)
            self.client.send("GET {} HTTP/1.1
    Host:{}
    Connection:close
    
    ".format(self.path, self.host).encode("utf8"))
            selector.register(self.client.fileno(), EVENT_READ, self.readable)
    
        def readable(self, key):
            d = self.client.recv(1024)
            if d:
                self.data += d
            else:
                selector.unregister(key.fd)
                data = self.data.decode("utf8")
                html_data = data.split("
    
    ")[1]
                print(html_data)
                self.client.close()
                urls.remove(self.spider_url)
                if not urls:
                    global stop
                    stop = True
    
        def get_url(self, url):
            self.spider_url = url
            url = urlparse(url)
            self.host = url.netloc
            self.path = url.path
            self.data = b""
            if self.path == "":
                self.path = "/"
    
            # 建立socket连接
            self.client = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
            self.client.setblocking(False)
    
            try:
                self.client.connect((self.host, 80))  # 阻塞不会消耗cpu,不阻塞,使其一直运行下去,因为这里不使用socket.blocking(True)会抛出异常。
            except BlockingIOError as e:
                pass
    
            #注册
            selector.register(self.client.fileno(), EVENT_WRITE, self.connected)
    
    
    def loop():
        #事件循环,不停的请求socket的状态并调用对应的回调函数
        #1. select本身是不支持register模式
        #2. socket状态变化以后的回调是由程序员完成的
        while not stop:
            ready = selector.select()
            for key, mask in ready:
                call_back = key.data
                call_back(key)
        #回调+事件循环+select(pollepoll)
    
    if __name__ == "__main__":
        fetcher = Fetcher()
        import time
        start_time = time.time()
        for url in range(20):
            url = "http://shop.projectsedu.com/goods/{}/".format(url)
            urls.append(url)
            fetcher = Fetcher()
            fetcher.get_url(url)
        loop()
        print(time.time()-start_time)

    回调之痛

    1. 代码可读性差,因为嵌套了多层回调

    2. 共享状态困难,这里指的是共享变量,如socket的变量

    3. 异常处理困难,如果嵌套多层,异常难以处理

  • 相关阅读:
    margin 碰到过的重叠问题
    等宽字体延伸到的 ch 长度单位和动画 animation-timing-function
    三栏布局 左右固定 中间自适应
    左侧固定,右侧自适应的五种布局方式
    jmeter环境配置
    python3 + selenum 环境搭建
    win10系统安装loadrunner11提示“为了对电脑进行保护 已经阻止此应用”的解决方案
    关于loadrunner运行场景时提示“初始化失败,通信错误”的解决方案
    安装phpwind报错
    关于XAMPP Apache无法启动问题解决方案
  • 原文地址:https://www.cnblogs.com/callyblog/p/11183331.html
Copyright © 2011-2022 走看看