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

darksun9972 2012-06-11 08:19:02
我尝试写了个类,代码如下

import os
import sys
import string
class 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 cmd
if __name__=='__main__':
myOs=MyOs('win32')
myOs.setFunc()
myOs.mkdir(r'c:\test')
#代码完结



在这个类中我定义了一个setFunc方法

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,小弟先谢谢各位了
...全文
368 3 打赏 收藏 转发到动态 举报
AI 作业
写回复
用AI写文章
3 条回复
切换为时间正序
请发表友善的回复…
发表回复
angel_su 2012-06-12
  • 打赏
  • 举报
回复
sry上面wrapper的缩进乱了...
def wrapper(func):
def f(*args):
self.executeCmd(func(*args))
return f
angel_su 2012-06-12
  • 打赏
  • 举报
回复
函数本体在调用的时候,才开始解析执行,所以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)
panghuhu250 2012-06-11
  • 打赏
  • 举报
回复
[Quote=引用楼主 的回复:]
for funcName,function in funcList.items():
print funcName,"==",function
selfClass.__dict__[funcName]= lambda *argv: self.executeCmd(function(*argv))
[/Quote]
因为所有的lambda中function都是同一个变量,该变量最后的值是MyOs.getDelCmd。

为什么不直接定义mkdir,move等方法?这种直接去改变__dict__的办法应该尽量避免。

37,743

社区成员

发帖
与我相关
我的任务
社区描述
JavaScript,VBScript,AngleScript,ActionScript,Shell,Perl,Ruby,Lua,Tcl,Scala,MaxScript 等脚本语言交流。
社区管理员
  • 脚本语言(Perl/Python)社区
  • WuKongSecurity@BOB
加入社区
  • 近7日
  • 近30日
  • 至今

试试用AI创作助手写篇文章吧