首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 开发语言 > perl python >

python语言for循环顶用字典保存lambda函数时为什么只保存最后一个lambda方法呢

2012-08-22 
python语言for循环中用字典保存lambda函数时为什么只保存最后一个lambda方法呢?我尝试写了个类,代码如下Py

python语言for循环中用字典保存lambda函数时为什么只保存最后一个lambda方法呢?
我尝试写了个类,代码如下

Python code
import osimport sysimport stringclass MyOs:    def __init__(self,osType='win32',logger=sys.stdout):        self.logger=logger            self.osType=osType        if osType=='win32':            self.seperator='\\'        else:            self.seperator='/'    def setFunc(self):        selfClass=self.__class__        funcList={'mkdir':selfClass.getMkdirCmd,                'copy':selfClass.getCopyCmd,                'move':selfClass.getMoveCmd,                'delete':selfClass.getDelCmd}        for funcName,function in funcList.items():            print funcName,"==",function            selfClass.__dict__[funcName]= lambda *argv: self.executeCmd(function(*argv))                def executeCmd(self,cmd):        os.system(cmd)        self.logger.write(cmd+'\n')    def getCopyCmd(self,src,des):        if self.osType=='win32':            cmd="copy "+src+" "+des        else:            cmd="mv "+src+" "+des        return cmd        def getMoveCmd(self,src,des):        if self.osType=='win32':            cmd="move "+src+" "+des        else:            cmd="mv "+src+" "+des        return cmd        def getDelCmd(self,fileName):        if self.osType=='win32':            cmd='del '+fileName        else:            cmd='rm -rf '+fileName        return cmd    def getMkdirCmd(self,dirName):        if self.osType=='win32':            cmd='mkdir '+dirName        else:            cmd='mkdir -p '+dirName        return cmdif __name__=='__main__':    myOs=MyOs('win32')    myOs.setFunc()    myOs.mkdir(r'c:\test')#代码完结


在这个类中我定义了一个setFunc方法
Python code
def setFunc(self):        selfClass=self.__class__        funcList={'mkdir':selfClass.getMkdirCmd,                'copy':selfClass.getCopyCmd,                'move':selfClass.getMoveCmd,                'delete':selfClass.getDelCmd}        for funcName,function in funcList.items():            print funcName,"==",function            selfClass.__dict__[funcName]= lambda *argv: self.executeCmd(function(*argv))

这个方法的本意是想动态生成类方法,但是我在运行该脚本的时候,却发现所有动态生成的方法都变成最后那个lambda方法,即动态生成的mkdir,copy,move,delete方法都变成了 lambda *argv: self.executeCmd(selfClass.getDelCmd(*argv))。该脚本的运行结果如下
C:\>MyNewOs.py
move == <unbound method MyOs.getMoveCmd>
copy == <unbound method MyOs.getCopyCmd>
mkdir == <unbound method MyOs.getMkdirCmd>
delete == <unbound method MyOs.getDelCmd>
找不到 c:\test
del c:\test
希望各位大大能解释一下,为什么会有这么奇怪的事情发生呢?我用的python版本为2.7.1rc1,小弟先谢谢各位了

[解决办法]
探讨
for funcName,function in funcList.items():
print funcName,"==",function
selfClass.__dict__[funcName]= lambda *argv: self.executeCmd(function(*argv))

[解决办法]
函数本体在调用的时候,才开始解析执行,所以function都会是同一个。固定参数的话可以加个有默认值的参数,类似:
lambda arg, f=function: self.executeCmd(f(arg))

不过你用不定参数的方式,没法再补一个参数,试试下面方式:
...
def wrapper(func):
def f(*args):
self.executeCmd(func(*args))
return f

for funcName,function in funcList.items():
print funcName,"==",function
selfClass.__dict__[funcName]= wrapper(function)
[解决办法]
sry上面wrapper的缩进乱了...
Python code
def wrapper(func):    def f(*args):        self.executeCmd(func(*args))    return f 

热点排行