一次优化web项目的经历记录(三)
一次优化web项目的经历记录
这段时间以来的总结与反思
前言:最近很长一段时间没有更新博客了,忙于一堆子项目的开发,严重拖慢了学习与思考的进程。
开水倒满了需要提早放下杯子,晚了就会烫手,这段时间以来,写的东西越来越不严谨,各种低级错误频出,早该停下总结并巩固一下了。
但出于一些原因一直没付诸于行,终于,烫到手了
第三章:yield与generator
Footprint.get_pics() 里到底发生了什么呢?
抱歉,最近事情有点多,更新晚了,我们继续。get_pics 本身耗时100多s,然而其内部的几个函数耗时加起来居然远低于这个值,为什么呢?
我就不绕弯了,答案就是本章的标题,yield与generator。我是怎么确定的呢?因为目前还剩下的 get_pics 内部的仅有的几个方法里,最可疑的只有这个方法:iter_directory
iter_directory 是用来迭代阿里云oss上的某个目录(其实是前缀相同的所有对象),以得到其中的所有文件(对象)的。很显然,这个方法需要调用外部api!
而作为一个调用外部api的方法,其执行2679次的总耗时居然仅有0.03秒,我的天哪,怎么可能!
是我的监听器有什么漏洞吗?
按照我的代码逻辑,在某个被注册的方法进入前,将会获取当前时间戳一次,而在其退出后则立即得到时间差,累加进方法总耗时。
我们不妨假设这样一种情景:有一个方法f,其返回值为方法g。g不会在f内部得到执行,它仅仅被返回,并且在f外部被调用并执行,用python描述如下:
@monitor.register
def f():
def g():
pass
return g
...
if __name__ == '__main__':
g = f()
for _ in range(0, 1000):
g()
print monitor.report()
那么很显然的,我的监听器实际上只监听了f方法的执行总耗时,而g的执行发生在f之外,成功“逃脱”了我的监视。
上面这个假设与yield有什么关系呢?
def f():
print '2'
yield 'f'
print '3'
if __name__ == '__main__':
print '1'
f()
print '4'
来猜测一下,上面的几个数字的打印顺序是什么呢?1234?如果你这么认为,那或许你需要补一补迭代器的常识了:
正确的答案应该是14。是的,没有2,也没有3。当你执行 f() 时,函数 f 根本没有得到真正的调用!准确的说,函数 f 其实被调用了,但这个 f 却不是你说认识的 f!
是的,当函数或方法内部使用了yield关键字时,实际上它已经不再是它自己了。当你执行 f() ,真正发生的并不是 f 内部的东东顺序执行,而是构造了一个 迭代器。
只有你执行 f().next(),也就是在返回的迭代器上执行 next() 方法,代码才会从f 内部开始解释执行,知道遇到yield关键字并立即返回。
再次执行 next() 时,会从上次离开的地方继续。如果遇不到yield了,则已迭代完毕,抛出一个 StopIteration 。
现在真相大白了, iter_directory 这个方法就是造成瓶颈的真凶。很显然,由于 iter_directory 事实上是一个generator,注册它其实是监听了它的迭代器生成方法。得到的耗时其实仅仅是生成迭代器的耗时。
真正造成严重延时的真凶,就是访问阿里云oss查找对象的方法,而这个操作是虽然看似在 iter_directory 里,但其实是在它的迭代器构造方法之外的,所以检测不到。
接下来就没什么好说的了,找到了耗时的真凶就该对它做优化了。这很容易,我采取了缓存策略,从memcache上获取缓存的值,而不是每次都从阿里云里去查找,除非接到更新信号或memcache上的值为空。
这次的事故中我学到了什么呢?
首先是,自己留下的坑总会自己跳。当初学python时,对yield的理解仅限于迭代器模式的一种语法糖,而没有深刻的去了解其实现的机制,没有认识到其对原函数|方法的装饰改造效果。于是就有了这次的不愉快的踩坑经历。
其次,疲劳代码很容易出错。无论怎么想,我也不明白自己当初怎么会写出类似于“每次需要某值都遍历某树”这种可谓脑残低效至极的代码,唯一的解释就是累了,没有仔细思考布局。
甚至类似的地方,在这次优化过程中还发现了很多处,虽然最终验证影响都远不如这个地方那么大,但看着自己写出的这么糟糕的代码还是很让人不爽的。真希望抛开一切来一次大重构。
最后,定期的总结思考很重要。这次的问题最直接的原因其实还不是对yield理解不透彻,虽然这是个隐患,但这次的导火索是在阿里云对象存储这块的糟糕的设计。
以前只是简单的用过阿里云oss服务,而最近的几个项目却恰好都深度依赖于它。但由于几个项目时间都赶得比较紧,在阿里云oss这块就只能调通能用就过。
内心恐怕早也意识到oss这块需要好好设计设计,规划一下了,但毕竟还是偷懒了。这次这个项目暴露出的之前设计上的不合理真的很重要,它直接改变了我对下一个项目(素材发布共享平台)的结构的设计。
很难想象如果再晚点发现,当这一套糟糕的模型已经用在多个项目中后,会有多麻烦。
本系列到此结束,谢谢
一次优化web项目的经历记录(三)的更多相关文章
- 一次优化web项目的经历记录(二)
一次优化web项目的经历记录 这段时间以来的总结与反思 前言:最近很长一段时间没有更新博客了,忙于一堆子项目的开发,严重拖慢了学习与思考的进程. 开水倒满了需要提早放下杯子,晚了就会烫手,这段时间以来 ...
- 一次优化web项目的经历记录(一)
一次优化web项目的经历记录 这段时间以来的总结与反思 前言:最近很长一段时间没有更新博客了,忙于一堆子项目的开发,严重拖慢了学习与思考的进程.开水倒满了需要提早放下杯子,晚了就会烫手,这段时间以来, ...
- Java Web项目搭建过程记录(struts2)
开发工具:eclipse 搭建环境:jdk1.7 tomcat 8.0 基础的java开发环境搭建过程不再赘述,下面从打开eclipse 之后的操作开始 第一步: 创建项目,File -> ...
- windows系统中ubuntu虚拟机安装及web项目到服务上(三)
项目在ubuntu虚拟机下的部署 一:将war从本地通过Xftp 4 传到虚拟机tomcat目录下的webapps目录下 2:修改tomcat下的server.xml <Host name=& ...
- 用maven工具管理web项目的错误记录:org.springframework.beans.factory.xml.XmlBeanDefinitionStoreException
运行异常报告日志: 严重: Context initialization failedorg.springframework.beans.factory.xml.XmlBeanDefinitionSt ...
- 小白的首个maven web项目Step1软件安装三(8.0.15mysql及workbench安装)
直接先开始下 MySQL 和 Workbench(mysql的可视化工具) ,注意下得是镜像版 .msi 后缀的 (mysql是纯控制面板的呈现方式,想要界面化操作可以装可视化工具,这里我装的是wor ...
- 部署WEB项目到服务器(三)安装mysql到linux服务器(Ubuntu)详解
突发奇想,想在自己电脑上部署一个web网站. 1,首先是下载一个适合自己已安装服务器版本的mysql数据库. 这里使用网上的链接http://dev.mysql.com/downloads/mysql ...
- mave之:java的web项目必须要的三个jar的pom形式
jsp-api javax.servlet-api jstl <!-- jsp --> <dependency> <groupId>javax.servlet< ...
- eclipse修改web项目部署路径
Eclipse中用Tomcat发布的Web项目,更改其部署路径 我的Eclipse的工作目录是D:/workspace 先配置Tomcat 选择你的tomcat版本 点击next 这里先不要把项目添加 ...
随机推荐
- Netty版本升级血泪史之线程篇
1. 背景 1.1. Netty 3.X系列版本现状 根据对Netty社区部分用户的调查,结合Netty在其它开源项目中的使用情况,我们可以看出目前Netty商用的主流版本集中在3.X和4.X上,其中 ...
- 2.5.2 使用alertdialog 创建列表对话框
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout ...
- openwrt使用3G上网卡
尊敬的大大.感谢你抽空指导我 我的设备是db120 mu350 和广东无限卡 版本是OpenWrt Backfire 10.03.336 DIY full 一. 没有安装到kmod-us ...
- 使用 Gradle 实现 TFS 构建自动化
发布于 2014-07-16 作者 陈 忠岳 感谢微软开放技术有限公司(简称"微软开放技术")发布的构建模板,我们现在便可以在 Team Foundation Server(TFS ...
- Linux Shell编程(11)——退出和退出状态
exit命令一般用于结束一个脚本,就像C语言的exit一样.它也能返回一个值给父进程.每一个命令都能返回一个退出状态(有时也看做返回状态).一个命令执行成功返回0,一个执行不成功的命令则返回一个非零值 ...
- HDU -- 4496
D-City Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65535/65535 K (Java/Others)Total Subm ...
- 宝洁HR
宝洁HR系统的测试犯了很多错误 1 最基本也是最弱智的错误:测试根本不仔细,多轮测试后仍然会发现前几轮应该发现的bug. 纠结测试不仔细的原因 a 个人工作坏习惯 老是认为理所当然,对于一些内容,总 ...
- Json 的日期格式转换成DateTime
JSON 的日期形式:”/Date(1242357713797+0800)/” , 下面我们就用以下C#的方法将他转换成DateTime类型: /// <summary> /// Json ...
- Android学习笔记(一)Android应用程序的组成部分
Android应用程序由松散耦合的组件组成,并使用应用程序Manifest绑定到一起:应用程序Manifest描述了每一组件和它们之间的交互方式,还用于指定应用程序元数据.其硬件和平台要求.外部库以及 ...
- JavaScript高级程序设计10.pdf
String类型有几种操作字符串的方法 concat()方法拼接任意多个字符串,不修改原字符串 var stringValue=“hello ”; var result=stringValue.con ...