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

相关文章

linux生产者,消费者问题

pthread_cond_wait() :用于阻塞当前线程,等待别的线程使用pthread_cond_signal()或pthread_cond_broadcast来唤醒它。 pthread_cond_wait() 必须与pthread_mutex 配套使用。pthread_cond_wait()函数一进入wait状态就会自动release mutex。当其他线程通过pthread

问题:第一次世界大战的起止时间是 #其他#学习方法#微信

问题:第一次世界大战的起止时间是 A.1913 ~1918 年 B.1913 ~1918 年 C.1914 ~1918 年 D.1914 ~1919 年 参考答案如图所示

2024.6.24 IDEA中文乱码问题(服务器 控制台 TOMcat)实测已解决

1.问题产生原因: 1.文件编码不一致:如果文件的编码方式与IDEA设置的编码方式不一致,就会产生乱码。确保文件和IDEA使用相同的编码,通常是UTF-8。2.IDEA设置问题:检查IDEA的全局编码设置和项目编码设置是否正确。3.终端或控制台编码问题:如果你在终端或控制台看到乱码,可能是终端的编码设置问题。确保终端使用的是支持你的文件的编码方式。 2.解决方案: 1.File -> S

vcpkg安装opencv中的特殊问题记录(无法找到opencv_corexd.dll)

我是按照网上的vcpkg安装opencv方法进行的(比如这篇:从0开始在visual studio上安装opencv(超详细,针对小白)),但是中间出现了一些别人没有遇到的问题,虽然原因没有找到,但是本人给出一些暂时的解决办法: 问题1: 我在安装库命令行使用的是 .\vcpkg.exe install opencv 我的电脑是x64,vcpkg在这条命令后默认下载的也是opencv2:x6

【操作系统】信号Signal超详解|捕捉函数

🔥博客主页: 我要成为C++领域大神🎥系列专栏:【C++核心编程】 【计算机网络】 【Linux编程】 【操作系统】 ❤️感谢大家点赞👍收藏⭐评论✍️ 本博客致力于知识分享,与更多的人进行学习交流 ​ 如何触发信号 信号是Linux下的经典技术,一般操作系统利用信号杀死违规进程,典型进程干预手段,信号除了杀死进程外也可以挂起进程 kill -l 查看系统支持的信号

Python 字符串占位

在Python中,可以使用字符串的格式化方法来实现字符串的占位。常见的方法有百分号操作符 % 以及 str.format() 方法 百分号操作符 % name = "张三"age = 20message = "我叫%s,今年%d岁。" % (name, age)print(message) # 我叫张三,今年20岁。 str.format() 方法 name = "张三"age

问题-windows-VPN不正确关闭导致网页打不开

为什么会发生这类事情呢? 主要原因是关机之前vpn没有关掉导致的。 至于为什么没关掉vpn会导致网页打不开,我猜测是因为vpn建立的链接没被更改。 正确关掉vpn的时候,会把ip链接断掉,如果你不正确关掉,ip链接没有断掉,此时你vpn又是没启动的,没有域名解析,所以就打不开网站。 你可以在打不开网页的时候,把vpn打开,你会发现网络又可以登录了。 方法一 注意:方法一虽然方便,但是可能会有

java中查看函数运行时间和cpu运行时间

android开发调查性能问题中有一个现象,函数的运行时间远低于cpu执行时间,因为函数运行期间线程可能包含等待操作。native层可以查看实际的cpu执行时间和函数执行时间。在java中如何实现? 借助AI得到了答案 import java.lang.management.ManagementFactory;import java.lang.management.Threa

vue同页面多路由懒加载-及可能存在问题的解决方式

先上图,再解释 图一是多路由页面,图二是路由文件。从图一可以看出每个router-view对应的name都不一样。从图二可以看出层路由对应的组件加载方式要跟图一中的name相对应,并且图二的路由层在跟图一对应的页面中要加上components层,多一个s结尾,里面的的方法名就是图一路由的name值,里面还可以照样用懒加载的方式。 页面上其他的路由在路由文件中也跟图二是一样的写法。 附送可能存在

vue+elementui--$message提示框被dialog遮罩层挡住问题解决

最近碰到一个先执行this.$message提示内容,然后接着弹出dialog带遮罩层弹框。那么问题来了,message提示框会默认被dialog遮罩层挡住,现在就是要解决这个问题。 由于都是弹框,问题肯定是出在z-index比重问题。由于用$message方式是写在js中而不是写在html中所以不是很好直接去改样式。 不过好在message组件中提供了customClass 属性,我们可以利用