这里我不讨论 python 的一些有用的库或者框架,只从语言本身,最小支持的情况下谈论这门语言本身。语言的发展都是越来越接近Lisp,这也是Lisp这门语言伟大的原因。
下面我罗列一下我学习 python 的原因:
多编程范式
python是一门多范式的编程语言,所谓的过程式,面向对象和函数式的结合。
大部分人接触编程语言都是从过程式开始的,原因是因为过程式的程序方式与计算机运行方式是统一的,指令序列与运行过程是统一的。如典型的C,我也是从C开始学习的,过程式的程序语言设计编写程序较为简单,但是符合人机交互思考方式。
python虽然是一门面向对象语言,就连“ ”(空格)也可以看做是一个对象,但是python胜任过程式是没有问题的。
如不需要使用类的静态方法:
def a_plus_b(a,b): return a+b
1. Duck typing
Python在设计的时候将其当做一门面向对象的方式编写,且不说面向对象给软件设计带来的一些革命等,在python这样的动态语言中面向对象有一个亮点就是Duck typing(鸭子类型)。
关于鸭子类型,就是说,如果我认为一个抽象的东西会游泳会“嘎嘎”叫,我就可以把它当做鸭子。
def use_duck( Duck ): Duck.swim() Duck.gaga() class Duck: def swim(self): ... def gaga(self): ...
如果这样使用:
little_duck = Duck()
use_duck( little_duck )
关于Duck类,你可以给他取任何的名字,或者继承它取另一个名字,只需要实现 swim() gaga() 你就可以把它当做鸭子。
关于鸭子类型,很多人不理解为什么不需要提供一个接口来规定鸭子的行为,我既不支持也不反对,我的观点是这样的:
- 对于参数的检查,不符合动态语言的特性
- 提供了接口规范,那就不是鸭子类型了,直接叫多态得了
2. Python支持的函数式编程
首先是lambda 演算。
函数式编程的定义是将函数看做是变量一样的待遇,变量在程序中最简单的有什么待遇呢?
- 可以赋值
- 可以作为参数
- 可以改变值(Erlang例外)
- 且不说生命周期了和作用域了
λ演算背后蕴含着计算机可计算性的深厚知识,lambda也是图灵模型,是停机问题的一个否定答案,不仅仅是一个匿名函数那样简单。
关于 lambda 演算,看看这个程序做了什么:
map(lambda n:2*n,[1,2,3,4,5])
lambda n:2*n 本身作为一个匿名函数
lambda 本身作为一个参数传入 map()函数,这也就是说我的高阶函数,可以将函数变身看成是一个变量作为参数传递,这也是它作为函数受到的高等待遇
关于赋值和改变值,两种方式:
f = fun() 不改变函数状态,只改变名称,但是说明函数是可以赋值的
可以使用闭包作为改变函数的状态方式,或者使用装饰器来完成函数状态改变
函数式编程的使用也可以提高程序的可读性和减少代码,而且能够清晰的表达函数的功能,如MapReduce就是来自函数式编程的思想:
Map(func,List)
作用是将func 作用于List中的每一个元素。
以刚才的例子举例:
map(lambda n:2*n,[1,2,3,4,5])
此函数返回
[2,4,6,8,10]
重要的是在于知道这样的方式带给我们的清晰的设计方式。
当然函数式编程不是那么几句话就说完的,理解函数式编程的核心是理解 λ演算。
一些有意思的特性
1. 惰性计算:
看看完成一个斐波那契数列 python 可以怎么做:
> def fib(): a , b = 0 ,1 while 1: yield b a , b = b ,a+b > f = fib()
实际上由yield 生成了一个可迭代对象,每次调用f.next()就可以产生一个斐波那契值,而函数的内部状态是由迭代对象存储的。至于返回一个可迭代对象,如果需要确定迭代到多少位,可以使用 itertools.islice。
2. 协程
协程也是一个基于yield的概念,主要的模式是微线程的协作式工作模式:
def coroutine(func): def ret(): f = func() f.next() return f return ret @coroutine def consumer(): print "Wait to getting a task" while 1: n = (yield) print "Got %s",n import time def producer(): c = consumer() while 1: time.sleep(1) print "Send a task to consumer" c.send("task") if __name__ == "__main__": producer()
协程带来的好处是可以直接调度你的线程,这也是它为什么叫做协程而不是线程的原因,线程属于抢占式并发,协程属于协作式并发。
动态语言带来的好处
从程序设计带来的快感(我相信只有热爱这方面的人才有的感觉)来说,动态语言,比如python,节约了更多的时间可以用来陪女朋友或者老婆,或者老公。
当然,作为互联网时代快速开发来说,赶鸭子上线,也是《黑客与画家》上面介绍的,快速开发很重要,当然需要符合这方面的需求。
动态语言的CPU密集型运算必然比不过C/C++。
总之:人生苦短,我用python。