感悟Python的closure

最近在用django做一个小东西,目录结构什么的都设置好了,要写views.py。
因为仅仅是测试,所以输出一个字符串表示一下就可以了,于是开始:

def index(request):
    return HttpResponse("index page!")

def foo(request):
    return HttpResponse("foo page!")

def bar(request):
    return HttpResponse("bar page!")

......


我可没兴趣写这么蛋疼的东西,于是希望通过一个列表自动生成这些函数。
折腾了一会,得出了以下的代码:


# Create your views here.
from django.http import HttpResponse
import sys

def gen(lst):
    for i in lst:
        def _f(request):
            return HttpResponse("%s page!" % i)
        _f.func_name = i
        yield _f

l = [
    'index',
    'purchase',
    'message',
    'setting',
    'message_post',
    'message_anonpost',
    'setting_update',
    'ajax_history',
    'ajax_purchase',
    'ajax_message_setviewed'
]

for i in gen(l):
    sys.modules[__name__].__dict__[i.func_name] = i


恩, 看起来很让人满意,于是测试一下,发现所有的view都只显示"ajax_message_setviewed page!",也就是最后一个。
搞的我挺郁闷,这是怎么回事呢。。。。。。

仔细做实验外加分析了一下,发现所有的函数共享这同一个i变量(就是gen函数里的i)。

恩 因为gen函数只调用了一次,所以i变量就只创建了一次,所以所有的函数都引用同一个i了。

知道症结后,改写代码如下:

# Create your views here.
from django.http import HttpResponse
import sys

def gen(fn):
    def _f(request):
        return HttpResponse("%s page!" % fn)
    _f.func_name = fn
    return _f

l = [
    'index',
    'purchase',
    'message',
    'setting',
    'message_post',
    'message_anonpost',
    'setting_update',
    'ajax_history',
    'ajax_purchase',
    'ajax_message_setviewed'
]

for i in l:
    sys.modules[__name__].__dict__[i] = gen(i)

问题解决了

你可能感兴趣的:(Ajax,django,python,F#)