菜鸟教程的javascript闭包章节中,演示了js计数器的实现。 教程地址 http://www.runoob.com/js/js-function-closures.html
代码1
var add = (function(){
var counter = 0;
return function(){
return counter += 1;}
})(); add();
add();
add();
大概感觉会输出 1 2 3,但是第一次看的时候看得马马虎虎。
代码中用了匿名函数 闭包 自调用。教程中说要保持避免计数器的变量在全局被其他代码修改,就不使用全局变量了。
如果全局变量实现计数器就很简单了。
代码2
var counter = 0; function add() {
return counter += 1;
} add();
add();
add();
但还是需要理解闭包,计数器是个好例子。
在第一段代码中
var add变量指向外层函数的自调用,自调用以后,通过外层函数里面的return,add就指向了内层函数的函数名。
那么执行 add()就是执行内层函数了,所以会实现加1的功能,同时又能不使用全局变量。
把js第一段闭包代码翻译成python闭包代码就是这样
代码3
def outer():
counter = 0
def inner():
nonlocal counter
counter += 1
return counter
return inner add = outer()
print(add())
print(add())
print(add())
使用全局变量,把js第二段代码翻译成python是
代码4
counter= 0 def func():
global counter
counter += 1
return counter print (func())
print (func())
细心的就会发现python中还是与js区别了一点点,加了global nonlocal修饰,如果去掉这会是个python中很经典的错误
UnboundLocalError: local variable 'counter' referenced before assignment
这个不是本篇要讨论的啦,但还是说下,在python中函数访问局部变量是不需要特意去声明global的,但如果在函数中修改了那个变量,比如给变量赋值,那么就会把该变量当做成是局部变量了,但当成局部变量时候却没有初始化,所以会报错,这点需要注意。
在py2总没有nonlcal修饰符
可以这样做,可以使用字典对象什么的。
代码5
def outer():
counter = {0:0}
def inner():
# nonlocal counter counter[0] += 1
return counter[0]
return inner add = outer()
print(add())
print(add())
print(add())
闭包总难以理解的,很蛋疼有没有。
不就是要消全局吗,可以使用类来实现。实例属性在类的所有实例方法中(非staticmethod和classmethod)可以访问。
代码6
class Couter():
def __init__(self):
self.x = 0
def add(self):
self.x += 1
return self.x c = Couter()
print(c.add())
print(c.add())
print(c.add())
这样做,既不需要全局变量,也不需要闭包了,很容易理解。
如果使用函数又不闭包。可以这样做,这样避开了闭包,又避开了py2不支持 nonlocal语法
代码7
def func():
if not hasattr(func,'counter'):
func.counter = 0
else:
func.counter += 1
return func.counter print (func())
print (func())
总之就是避免一个问题啦,函数的局部变量每次都初始化,需要用全局变量。c语言中有静态变量修饰符,很容易解决这个计数器。
代码8
int f(){
static i=0;
i++;
return i;
}
f()
f()
在c语言中,可以通过加入static修饰符来声明静态变量,如果去掉static修饰符不管调用多少次函数都是返回0.。
闭包写起来很麻烦,但调用起来很实用。
如果是实现一个计数器,我认为使用类,就是代码6的方法比较好。这样还能很方便再在内中写个减数函数啥的,很容易扩展到其他的功能。
最后再加一种python方式,利用函数可变类型的默认参数
def add(l=[0]):
l[0] += 1
return l[0] print add()
print add()
print add() python的默认值参数只会在函数定义处被解析一次,此后每次调用函数的时候,默认值参数都会是这个值了。
这个笔试很爱考这个陷阱,可以利用这个笔试陷阱,来实现计数器。
笔试通常如下: