目录
  • 前言
  • 决上述问题
    • 目前想到两个方法
    • 实验结果
  • 总结

    前言

    在使用aiohttp并发访问多个页面时效率,明显比串行requests快很多,

    但是也存在一个问题,就是网站检测到短时间内请求的数量过多会导致页面请求不成成功,

    页面返回429 (too many requests)。

    解决上述问题

    目前想到两个方法

    1、控制请求的时间,用sleep延时,来消耗每一次访问的时间,减少单位时间内的访问量,这样肯定是可以,但效率太低

    2、控制并发数量,控制并发数量,普遍推荐用信号量来控制使用方法也比较简单如下:

    from asyncio import tasks
    from aiohttp.client import ClientSession
    from lxml import etree
    from time import sleep
    import time
    import asyncio
    import aiohttp
    
    async def read_page_list(page_num,sem):
        params = {
            'page':page_num,
        }
        #通过连接池控制并发数量 limit 默认为100  0 为无限制
        async with sem:
            try:
                async with aiohttp.ClientSession() as session:
                    async with session.get(url=url,params=params,headers=headers) as response:
                        text = await response.text()
            except Exception as e:
                print('exception:',e)
            
            tree = etree.HTML(text)
            page_list = tree.xpath('//*[@id="thumbs"]/section[1]/ul/li')
            # break
            for li in page_list:
                pic_small_url = li.xpath('.//img/@data-src')[0]
                # print(pic_small_url,type(pic_small_url))
                # pic_small_url = str(pic_small_url)
                if 'small' in pic_small_url:
                    temp_url = pic_small_url.replace('small','full')
                    a = temp_url.rfind('/')
                    temp_url1= temp_url[:a]
                    pic_full_url = temp_url1+'/wallhaven-'+temp_url.split('/')[-1]
                    pic_full_url = pic_full_url.replace('th','w')
                    # print(page_num,pic_full_url)
                    pic_list.append(pic_full_url)
                else:
                    print(page_num,'find small error',pic_small_url)
                
            print(page_num,len(page_list),response.status)
            # await asyncio.sleep(1)
            #这里可以用硬延时来控制程序的访问速度,进而控制单位时间内并发的数量
            # sleep(0.5)
    
    #定义信号量
    sem = asyncio.Semaphore(2)
    
    start = time.time()
    #建立任务列表
    tasks = [loop.create_task(read_page_list(i,sem)) for i in range(1,20)]
    loop.run_until_complete(asyncio.wait(tasks))
    print('get page list use time:',time.time()-start)
    
    
    

    实验结果

    如下:

    • 经试验只有当请求页面20个 sem=1时才不会出现服务器返回429.
    • 当把请求页面数量改为10 sem=5是就不会出现服务返回429的情况

    总结

    以上为个人经验,希望能给大家一个参考,也希望大家多多支持小闻网。

    声明:本站所有文章,如无特殊说明或标注,均为本站原创发布。任何个人或组织,在未征得本站同意时,禁止复制、盗用、采集、发布本站内容到任何网站、书籍等各类媒体平台。如若本站内容侵犯了原著者的合法权益,可联系我们进行处理。