Python168的学习笔记8
#coding:utf8 #斐波那契数列,第三项起,每一项都等于前两项之和 def memo(func):
cache = {}#闭包
def wrap(*args):
if args not in cache:
cache[args] = func(*args)
return cache[args]
return wrap @memo
def fibonacci(n):
if n<=1:
return 1
return fibonacci(n-1)+fibonacci(n-2) #上楼梯算法,总过n个台阶,一次只能迈a-b个台阶,不能后退,问有几种走法
@memo
def climb(n,steps):
count = 0
if n==0:
count = 1
elif n>0:
for step in steps:
count += climb(n-step,steps)
return count
装饰器的简单应用还是明白的,但是要理解这两个递归算法。
关于函数的元数据,如:f.__name__,指的是def时指定的名字;f.__doc__,指的是函数文档字符串;f.__moudle__,指的是函数所属模块;f.__default__,指的是函数默认值;f.__dict__,指的是属性字典;f.__closure__,指的是函数的闭包。。。这些元数据都是func的属性。
使用装饰器可能会改变func的元数据。
#coding:utf8
from functools import update_wrapper,wraps,WRAPPER_ASSIGNMENTS,WRAPPER_UPDATES def mydecorator(func):
@wraps(func)#第一种
def wrapper(*args,**kw):
'''wrapper function'''
print 'In wrapper'
func(*args,**kw)
#wrapper.__name__ = func.__name__ 第二种
#update_wrapper(wrapper,func,('__name__','__doc__'),('__dict__',))#第四个参数是将func的属性更新到wrap
#第三种:update_wrapper后两个的默认参数分别是('__module__', '__name__', '__doc__');('__dict__',)
return wrapper @mydecorator
def example():
'''example function'''
print 'In example'
其实就是对处理后的func的属性进行重新赋回原来的值。
定义带参数的装饰器,就是用去判断参数的类型,因为要用过python3的库,所以先略过。
修改装饰器的属性。这样的好处是能够在运行中动态修改,而不是每次都要去装饰器处修改。
给包裹函数增加一个函数,作为包裹函数的属性,使其可以修改闭包中的自由变量。
import time
from functools import wraps
import logging def warn(timeout):
timeout = [timeout]#将参数设为只有一个数的list
def decorator(func):
@wraps(func)
def wrapper(*args,**kw):
start = time.time()
res = func(*args,**kw)
used = time.time()-start
if used > timeout[0]:
msg = '%s : %s >%s' %(func.__name__,used,timeout[0])
logging.warn(msg)
return res
def setTimeout(k):#注意这里
timeout[0] = k
wrapper.setTimeout = setTimeout
return wrapper
return decorator from random import randint
@warn(0.5)
def test():
print 'In test'
while randint(0,1):
print 'I am sleep'
time.sleep(1) for _ in range(10):
test() test.setTimeout(1)
for _ in range(10):
test()
使用类来定制装饰器。
#coding:utf8
import logging
from time import localtime,time,strftime,sleep class CallingInfo(object):
def __init__(self,name):
log = logging.getLogger(name)
log.setLevel(logging.INFO)#设置等级
fh = logging.FileHandler(name+'.log')#设置文件处理方法
log.addHandler(fh)#绑定方法到log上
log.info('Start'.center(50,'-'))#添加log文本的头部
self.log= log#绑定到类的属性上
self.formatter = '%(func)s ->[%(time)s - %(used)s -%(ncalls)s]'#定义了输出模板 def info(self,func):#这个就是平常的装饰器
def wrapper(*args,**kw):
wrapper.ncalls +=1
lt = localtime()#返回当地时间
start = time()
res = func(*args,**kw)
used = time() - start info = {}#建立info的字典
info['func'] = func.__name__
info['time'] = strftime('%x %X',lt)
info['used'] = used
info['ncalls'] = wrapper.ncalls msg = self.formatter % info#将字典映射到输入模板上
self.log.info(msg)#输出字符串
return res
wrapper.ncalls = 0#这个语句需要在这个位置的解释有待完善
return wrapper def setFromatter(self,formatter):
self.formatter = formatter def turnOn(self):
self.log.setLevel(logging.INFO) def turnOff(self):
self.log.setLevel(logging.WARN)#级别提高就不在那里输出了 cinfo1 = CallingInfo('mylog1')
cinfo2 = CallingInfo('mylog2') cinfo1.setFromatter('%(func)s ->[%(time)s -%(ncalls)s]')
cinfo2.turnOff()#这里可以看到是很方便地修改装饰器 @cinfo1.info
def f():
print 'in f' @cinfo1.info
def h():
print 'in h' @cinfo2.info
def g():
print 'in g' from random import choice
for _ in xrange(10):
choice([f,g,h])()
sleep(choice([0.5,1,1.5]))
Python168的学习笔记8的更多相关文章
- Python168的学习笔记7
关于多线程操作. 对于IO操作,如访问网站,写入磁盘这种需要时间等待响应的操作,多个cpu也几乎不能提高效率. 对于CPU密集型操作,如这个格式转换,可以通过多个cpu同时去进行. 但是对于pytho ...
- Python168的学习笔记6
如何派生内置不可变类型并修改实例化行为. 个人理解,如何派生出自己想要的类. class IntTuple(tuple): def __new__(cls,iterable): g = (x for ...
- Python168的学习笔记5
关于对csv文件的操作. python标准库中有csv的库,使用非常方便. import csv with open('pingan.csv','rb') as rf: reader = csv.re ...
- Python168的学习笔记4
关于普通文本文件的读写 python2.7中,未注明的字符都是以acsii来编码的,而要让字符能够通用,必须声明为unicode. s=u'你好',s.encode('utf8')就是指用utf8来进 ...
- Python168的学习笔记3
list.extend(),可以拓展list,a=(0,1),b=(2,3) a.extend(b),a就变成(0,1,2,3) 分割字符串(除去字符串中的,\/;之类的),如果用str.split( ...
- Python168的学习笔记2
关于for循环,其实质是利用被循环对象的__iter__,或者__getitem__属性接口,由可迭代对象得到迭代器.for循环就是不断调用.next(),直到最终捕获到stop. import re ...
- Python168的学习笔记1
在对list的条件选择有两种常用方法,直接使用filter函数,就是filter(func,sequence);另外一种就是迭代操作,类似 x for x in sequence func.这两种方法 ...
- js学习笔记:webpack基础入门(一)
之前听说过webpack,今天想正式的接触一下,先跟着webpack的官方用户指南走: 在这里有: 如何安装webpack 如何使用webpack 如何使用loader 如何使用webpack的开发者 ...
- PHP-自定义模板-学习笔记
1. 开始 这几天,看了李炎恢老师的<PHP第二季度视频>中的“章节7:创建TPL自定义模板”,做一个学习笔记,通过绘制架构图.UML类图和思维导图,来对加深理解. 2. 整体架构图 ...
随机推荐
- vue中使用cookie记住用户上次选择(本次例子中为下拉框)
最近工作中碰到一个需求,添加一条数据时,自动记住上次选择的下拉框的数据,刚开始觉得没思路,后来请教了项目组长,组长直接一句,这不很简单吧,直接用cookie,我:....... 好吧,都王的差不多了, ...
- JDK1.8源码TreeMap
基于红黑树(Red-Black tree)的 NavigableMap 实现:键的排序由构造方法决定:自然排序,Comparator排序:非线程安全(仅改变与现有键关联的值不是结构上的修改):线程安全 ...
- PE文件结构及其加载机制
一.PE文件结构 PE即Portable Executable,是win32环境自身所带的执行体文件格式,其部分特性继承自Unix的COFF(Common Object File Format)文件格 ...
- fc26 url
aarch64 http://linux.yz.yamagata-u.ac.jp/pub/linux/fedora-projects/fedora-secondary/releases/26/Ever ...
- C#基础学习之装箱,拆箱
装箱,拆箱这两个的大条件是有继承关系. 装箱:值类型转换为引用类型 拆箱:引用类型转换为之类 但是要注意大条件. string (引用类型) int(值类型) 这个转换因为没有继承关系,内存中没 ...
- OpenFlow1.3协议wireshark抓包分析
OpenFlow v1.0 v1.0协议消息列表如下: 分为三类消息:Controller-to-switch,asynchronous和symmertric. v1.0(包含至少一个流表,每个流表包 ...
- Django模型和ORM
一.ORM ORM介绍 ORM概念 对象关系映射(Object Relational Mapping,简称ORM)模式是一种为了解决面向对象与关系数据库存在的互不匹配的现象的技术. 简单的说,ORM是 ...
- python网络编程-optparse
Python 有两个内建的模块用于处理命令行参数: 一个是 getopt,<Deep in python>一书中也有提到,只能简单处理 命令行参数: 另一个是 optparse,它功能强大 ...
- Python3语法详解
一.下载安装 1.1Python下载 Python官网:https://www.python.org/ 1.2Python安装 1.2.1 Linux 平台安装 以下为在Unix & Linu ...
- spring配置文件头部配置解析(applicationContext.xml)
分享一个好的学习网站:http://how2j.cn?p=4509 相信大家对spring的配置文件应该都看的很多了,那么大家对配置文件头部的那一坨坨的东西到底是什么了解吗?下面我就把自己的一些见解和 ...