python装饰器和偏函数联合运用碰到的问题

2024-06-12 17:38

本文主要是介绍python装饰器和偏函数联合运用碰到的问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在写一个爬虫系统,用来爬取学校各个新闻网站的新闻并推送到微信上。

开始,我每个网站写一个爬虫函数,后来发现爬取的过程有许多相同的地方,代码冗余度太大,就想到了用装饰器:

def newsResultProcess(fun):  # 每个新闻爬虫按要求返回5个参数即可,爬取元素的顺序必须是:id、标题、时间@functools.wraps(fun)def wrapper(*args, **kwargs):html, remod, fullurl, imageurl, timestamp = fun(*args, **kwargs)html = u''.join(html.split())result = re.findall(remod, html)result = filter(lambda x: timestamp == x[2], result)return map(lambda x: (fullurl % x[0], x[1], imageurl), result)return wrapper

每个爬虫函数就可以写成这样:

@newsResultProcess
def deanNews():  #教务新闻deannewsurl = 'http://dean.swjtu.edu.cn/servlet/NewsAction?Action=NewsMore'deanfullnewsurl = 'http://dean.swjtu.edu.cn/servlet/NewsView?NewsID=%s'deannewsimage = siteurl + '/static/img/dean.jpg'timestamp = time.strftime(u'%Y-%m-%d')response = urllib2.urlopen(deannewsurl)html = response.read().decode('utf-8')remod = re.compile(ur'<ahref=.*?ID=(.*?)"class.*?title="(.*?)".*?hidden;">(.*?)</div></li>')return html, remod, deanfullnewsurl, deannewsimage, timestamp

这样,每个爬虫函数只要写入每个网页不同的内容就可以了,即链接,时间戳格式,正则表达式,网页编码方式。代码基本上没有什么冗余度了

后来为了提高响应速度,又写了一个装饰器,用来进行缓存:

def newsCache(fun):@functools.wraps(fun)def wrapper(*args, **kwargs):ret = cache.get(fun.__name__)if ret == None:ret = fun(*args, **kwargs)cache.set(fun.__name__, ret, newsexpire)return retreturn wrapper

缓存系统根据函数名来区分各个函数返回的内容


再后来,我发现一个网站可能要爬取多个网页的新闻,这些网页只是链接不同,其他都一样,只用装饰器,仍然会有很高的冗余度。于是,我想到了偏函数,将链接作为参数提供给爬虫函数,用偏函数来固定不同爬虫的链接。就像这样:

@newsCache
@newsResultProcess
def yanghuaCollege( url,image):fullnewsurl = 'http://www.yanghua.net/WebSite/Head/Show.aspx?ID=%s'image = siteurl + imagetimestamp = time.strftime(u'%Y-%m-%d')response = urllib2.urlopen(url)html = response.read().decode('gbk')remod = re.compile(ur'<ahref.*?ID=(\d+).*?>(.*?)\((.*?)\)')return html, remod, fullnewsurl, image, timestampyanghuaCivil = functools.partial(yanghuaCollege, url='', image='/static/img/civil.jpg')

可是这里问题出现了,这时微信端所有新闻页返回的结果都是一样的,又思考了一会我才发现了问题所在:

首先要说装饰器,添加了装饰器后,原来的那个单纯的用def 定义的函数已经不存在了,函数名字__name__也变成了装饰器的名字,不过这里有个补救措施,使用functools里的wraps装饰器,这样外部装饰器返回的函数名仍然不变,这样再根据函数名识别不同的缓存就没有问题。

然后再说偏函数,偏函数就是相当于把一个函数的参数固定,再用一个引用指向这个固定了参数的函数,并不影响原来的函数。但是运用偏函数后,返回的这个函数对象,并不是只在参数上存在区别,实际的情况可以自己打印dir测试。在这里,偏函数返回的函数对象没有了__name__属性(实际上,少了很多属性)。这里我还要仔细研究一下偏函数的工作方式。

最后,当两者放在一起的时候,在这里问题就更严重了,由于装饰器作用之后,函数的一切数性都已确定,所以,缓存装饰器里的函数名,还是内部装饰器返回的,即原函数的名字。于是再加上偏函数的时候,所有返回的偏函数内部用来区分的函数名都是一样的,而且这里不会因为偏函数没有__name__属性发生AttributeError异常,因为缓存里的函数名字在装饰器发生作用的时候是存在的。

最后,我想到的解决办法是,既然函数名已经无法在定义偏函数的时候确定,就只能动态的添加__name__属性了

def newsCache(fun):@functools.wraps(fun)def wrapper(*args, **kwargs):if 'name' in kwargs:fun.__name__ = kwargs['name']  # 由于先进行装饰再使用偏函数,函数名字只能这样动态确定ret = cache.get(fun.__name__)if ret == None:ret = fun(*args, **kwargs)cache.set(fun.__name__, ret, newsexpire)return retreturn wrapperdef newsResultProcess(fun):  # 每个新闻爬虫按要求返回5个参数即可,爬取元素的顺序必须是:id、标题、时间@functools.wraps(fun)def wrapper(*args, **kwargs):html, remod, fullurl, imageurl, timestamp = fun(*args, **kwargs)html = u''.join(html.split())result = re.findall(remod, html)result = filter(lambda x: timestamp == x[2], result)return map(lambda x: (fullurl % x[0], x[1], imageurl), result)return wrapper
<pre name="code" class="python">@newsCache
@newsResultProcess
def yanghuaCollege(urlNo, image, name):url = 'http://www.yanghua.net/WebSite/Head/AcademyBulletin.aspx?CollegeNo=' + urlNofullnewsurl = 'http://www.yanghua.net/WebSite/Head/Show.aspx?ID=%s'image = siteurl + imagetimestamp = time.strftime(u'%Y-%m-%d')response = urllib2.urlopen(url)html = response.read().decode('gbk')remod = re.compile(ur'<ahref.*?ID=(\d+).*?>(.*?)\((.*?)\)')return html, remod, fullnewsurl, image, timestampyanghuaCivil = functools.partial(yanghuaCollege, urlNo='001', image='/static/img/civil.jpg', name='yanghyaCivil')
yanghuaSme = functools.partial(yanghuaCollege, urlNo='002', image='/static/img/sme.jpg', name='yanghyaSme')

到这里,我的爬虫系统终于正常工作了!

python初学,写的不好还请指教!

 

这篇关于python装饰器和偏函数联合运用碰到的问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1054888

相关文章

Python中你不知道的gzip高级用法分享

《Python中你不知道的gzip高级用法分享》在当今大数据时代,数据存储和传输成本已成为每个开发者必须考虑的问题,Python内置的gzip模块提供了一种简单高效的解决方案,下面小编就来和大家详细讲... 目录前言:为什么数据压缩如此重要1. gzip 模块基础介绍2. 基本压缩与解压缩操作2.1 压缩文

Python设置Cookie永不超时的详细指南

《Python设置Cookie永不超时的详细指南》Cookie是一种存储在用户浏览器中的小型数据片段,用于记录用户的登录状态、偏好设置等信息,下面小编就来和大家详细讲讲Python如何设置Cookie... 目录一、Cookie的作用与重要性二、Cookie过期的原因三、实现Cookie永不超时的方法(一)

Python内置函数之classmethod函数使用详解

《Python内置函数之classmethod函数使用详解》:本文主要介绍Python内置函数之classmethod函数使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 类方法定义与基本语法2. 类方法 vs 实例方法 vs 静态方法3. 核心特性与用法(1编程客

从入门到精通MySQL联合查询

《从入门到精通MySQL联合查询》:本文主要介绍从入门到精通MySQL联合查询,本文通过实例代码给大家介绍的非常详细,需要的朋友可以参考下... 目录摘要1. 多表联合查询时mysql内部原理2. 内连接3. 外连接4. 自连接5. 子查询6. 合并查询7. 插入查询结果摘要前面我们学习了数据库设计时要满

Python函数作用域示例详解

《Python函数作用域示例详解》本文介绍了Python中的LEGB作用域规则,详细解析了变量查找的四个层级,通过具体代码示例,展示了各层级的变量访问规则和特性,对python函数作用域相关知识感兴趣... 目录一、LEGB 规则二、作用域实例2.1 局部作用域(Local)2.2 闭包作用域(Enclos

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Python实现对阿里云OSS对象存储的操作详解

《Python实现对阿里云OSS对象存储的操作详解》这篇文章主要为大家详细介绍了Python实现对阿里云OSS对象存储的操作相关知识,包括连接,上传,下载,列举等功能,感兴趣的小伙伴可以了解下... 目录一、直接使用代码二、详细使用1. 环境准备2. 初始化配置3. bucket配置创建4. 文件上传到os

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操

使用Python实现可恢复式多线程下载器

《使用Python实现可恢复式多线程下载器》在数字时代,大文件下载已成为日常操作,本文将手把手教你用Python打造专业级下载器,实现断点续传,多线程加速,速度限制等功能,感兴趣的小伙伴可以了解下... 目录一、智能续传:从崩溃边缘抢救进度二、多线程加速:榨干网络带宽三、速度控制:做网络的好邻居四、终端交互

Python中注释使用方法举例详解

《Python中注释使用方法举例详解》在Python编程语言中注释是必不可少的一部分,它有助于提高代码的可读性和维护性,:本文主要介绍Python中注释使用方法的相关资料,需要的朋友可以参考下... 目录一、前言二、什么是注释?示例:三、单行注释语法:以 China编程# 开头,后面的内容为注释内容示例:示例:四