Python 实用技巧(上)

来源:互联网 发布:优化路政许可服务 编辑:程序博客网 时间:2024/04/29 18:00

Python 实用技巧(上)

本文作者: 伯乐在线 - 刘志军 。转载请参见文章末尾处的要求。

本文根据SO上的热门问答hidden features of python整理而成,早期有人做过类似的整理,但是内容比较旧而且比较粗糙,因此笔者在原文基础上加入自己的一些理解,另外那些高质量的评论也引入进来了。总之,这是一篇用心之作,希望你可以喜欢。

链式比较操作

1
2
3
4
5
6
7
8
9
10
11
>>> x =5
>>>1< x < 10
True
>>>10< x < 20
False
>>> x < 10< x*10< 100
True
>>>10> x <=9
True
>>>5==x > 4
True

你可能认为它执行的过程先是:1 < x,返回True,然后再比较True < 10,当然这么做也是返回True,比较表达式True < 10,因为解释器会把True转换成1False转换成0。但这里的链式比较解释器在内部并不是这样干的,它会把这种链式的比较操作转换成:1 < x and x < 10,不信你可以看看最后一个例子。这样的链式操作本可以值得所有编程语言拥有,但是很遗憾

 

枚举

1
2
3
4
5
6
7
8
9
>>> a =['a','b','c','d','e']
>>>forindex, item inenumerate(a):printindex, item
...
0a
1b
2c
3d
4e
>>>

用enumerate包装一个可迭代对象,可以同时使用迭代项和索引,如果你不这么干的话,下面有一种比较麻烦的方法:

1
2
fori inrange(len(a)):
    printi, a[i]

enumerate 还可以接收一个可选参数start,默认start等于0。enumerate(list, start=1),这样index的起始值就是1

 

生成器对象

1
2
3
x=(nforn infoo ifbar(n))  #foo是可迭代对象
>>>type(x)
<type'generator'>

你可以把生成器对象赋值给x,意味着可以对x进行迭代操作:

1
2
forn inx:
    pass

它的好处就是不需要存储中间结果,也许你会使用(列表推倒式):

1
2
3
x=[n forn infoo ifbar(n)]
>>>type(x)
<type'list'>

它比生成器对象能带来更快的速度。相对地,生成器更能节省内存开销,它的值是按需生成,不需要像列表推倒式一样把整个结果保存在内存中,同时它不能重新迭代,列表推倒式则不然。

 

iter()可接收callable参数

iter()内建函数接收的参数分为两种,第一种是:

1
iter(collection)---> iterator

参数collection必须是可迭代对象或者是序列 ,第二种是:

1
itercallable, sentinel) --> iterator

callable函数会一直被调用,直到它的返回结果等于sentinel,例如:

1
2
3
4
defseek_next_line(f):
    #每次读一个字符,直到出现换行符就返回
    forc initer(lambda: f.read(1),'\n'):
        pass

小心可变的默认参数

1
2
3
4
5
6
7
8
9
10
>>>deffoo(x=[]):
...     x.append(1)
...    printx
...
>>> foo()
[1]
>>> foo()
[1,1]
>>> foo()
[1,1,1]

取而代之的是你应该使用一个标记值表示“没有指定”来替换可变值,如:

1
2
3
4
5
6
7
8
9
>>>deffoo(x=None):
...    ifx isNone:
...         x =[]
...     x.append(1)
...    printx
>>> foo()
[1]
>>> foo()
[1]

 

发送值到生成器函数在中

1
2
3
4
5
6
7
defmygen():
    """Yield 5 until something else is passed back via send()"""
    a=5
    whileTrue:
        f=(yielda) #yield a and possibly get f in return
        iff isnot None:
            a=#store the new value

你可以:

1
2
3
4
5
6
7
8
9
>>> g =mygen()
>>> g.next()
5
>>> g.next()
5
>>> g.send(7#we send this back to the generator
7
>>> g.next()#now it will yield 7 until we send something else
7

如果你不喜欢使用空格缩进,那么可以使用C语言花括号{}定义函数:

1
2
3
>>>from__future__ importbraces   #这里的braces 指的是:curly braces(花括号)
  File"<stdin>", line 1
SyntaxError:nota chance

当然这仅仅是一个玩笑,想用花括号定义函数?没门。感兴趣的还可以了解下:

1
from__future__ importbarry_as_FLUFL

不过这是python3里面的特性,http://www.python.org/dev/peps/pep-0401/

 

切片操作中的步长参数

1
2
3
a=[1,2,3,4,5]
>>> a[::2# iterate over the whole list in 2-increments
[1,3,5]

还有一个特例:x[::-1],反转列表:

1
2
>>> a[::-1]
[5,4,3,2,1]

有关反转,还有两个函数reverse、reversed,reverse是list对象的方法,没有返回值,而reversed是内建方法,可接收的参数包括tuple、string、list、unicode,以及用户自定义的类型,返回一个迭代器。

1
2
3
4
5
6
7
8
9
>>> l =range(5)
>>> l
[0,1,2,3,4]
>>> l.reverse()
>>> l
[4,3,2,1,0]
>>> l2 =reversed(l)
>>> l2
<listreverseiteratorobjectat 0x99faeec>

 

装饰器

装饰器使一个函数或方法包装在另一个函数里头,可以在被包装的函数添加一些额外的功能,比如日志,还可以对参数、返回结果进行修改。装饰器有点类似Java中的AOP。下面这个例子是打印被装饰的函数里面的参数的装饰器,

1
2
3
4
5
6
7
8
9
10
11
12
13
>>>defprint_args(function):
>>>    defwrapper(*args,**kwargs):
>>>        print'Arguments:', args, kwargs
>>>        returnfunction(*args,**kwargs)
>>>    returnwrapper
 
>>> @print_args
>>>defwrite(text):
>>>    printtext
 
>>> write('foo')
Arguments: ('foo',) {}
foo

@是语法糖,它等价于:

1
2
3
4
>>> write =print_args(write)
>>> write('foo')
arguments: ('foo',) {}
foo

 

for … else语法

1
2
3
4
5
fori infoo:
    ifi ==0:
        break
else:
    print("i was never 0")

else代码块会在for循环正常结束后执行,除非遇到break(就不会执行),它等价于下面:

1
2
3
4
5
6
7
found=False
fori infoo:
    ifi ==0:
        found=True
        break
ifnot found:
    print("i was never 0")

不过这种语法看起来怪怪地,让人感觉是else块是在for语句块没有执行的时候执行的,很容易让人去类比 if else 的语法,如果是把else换成finally或许更容易理解

 

python2.5有个__missing__方法

dict的子类如果定义了方法__missing__(self, key),如果key不再dict中,那么d[key]就会调用__missing__方法,而且d[key]的返回值就是__missing__的返回值。

1
2
3
4
5
6
7
8
9
10
>>>classMyDict(dict):
... def__missing__(self, key):
...  self[key]=rv =[]
...  returnrv
...
>>> m =MyDict()
>>> m["foo"].append(1)
>>> m["foo"].append(2)
>>>dict(m)
{'foo': [1,2]}

在collections模块下有一个叫defaultdict的dict子类,它与missing非常类似,但是对于不存在的项不需要传递参数。

1
2
3
4
5
6
>>>fromcollections importdefaultdict
>>> m =defaultdict(list)
>>> m["foo"].append(1)
>>> m["foo"].append(2)
>>>dict(m)
{'foo': [1,2]}
原创粉丝点击