其实关于返回局部变量不只是python的问题,凡是使用堆栈结构处理函数的语言都会有这样的问题,切记不要返回局部变量。因为当创建函数的堆栈撤销,所有对局部变量的修改都灰飞烟灭。来看我的小例子
1 def handle(): 2 class myClass(object): 3 def __init__(self): #对各种基本数据类型测试 4 self.x = [] #列表 5 self.y = None #数值 6 self.z = {} #字典 7 self.a = str() #字符串 8 9 10 def _handle(self): 11 self.x.append('hello', 'world') 12 self.y = 1 13 self.z['hello'] = 'world' 14 self.a = "".join("hello") 15 16 17 my_class = myClass() 18 19 return my_class.x, my_class.y, my_class.z, my_class.a 20 pass 21 if __name__ == '__main__': 22 23 my_x, my_y, my_z, my_a = handle() 24 pass
我们再来看执行的情况:
来看看啊,所有局部变量都保持这初始的值,所有堆栈内部的修该都灰飞烟灭了吧,所以人生苦短,远离局部。同样苦短的还有局部变量和外部变量同名的问题,依然很具迷惑性
最近在摆弄ITchat这个库,遇到不少matplotlib的小坑,后期要详细整理,先上一个容易犯的小错,自己犯了好几次了,记录一下。代码如下:
1 import itchat 2 import pickle 3 4 myFriends = None 5 def dumpFriends(): 6 itchat.login() 7 friends = itchat.get_friends(update=True)[0:] 8 with open("C:\Users\fyc\Desktop\friends.txt","w") as f: 9 pickle.dump(friends, f) 10 11 def loadFriend(): 12 with open("C:\Users\fyc\Desktop\friends.txt", "r") as f: 13 myFriends = pickle.load(f) 14 pass 15 16 17 18 if __name__ == '__main__': 19 dumpFriends() 20 loadFriend() 21 pass
功能主要是把微信好友列表信息序列化到文件,再读取,重点在与myFriends这个变量。一时大意,局部变量与全局变量同名了,后果是:执行完所有,myFriends没有值,依然是None
通过breakpoint,我们记录错误的细节,如下图,我截了两个图,分别是全局变量myFriends的地址,和局部变量myFriends的地址。
全局变量myFriends:
局部变量myFriends:
至此,看到重点了吧,myFriends的地址变了,说白了,这本就是两个东西,什么东西呢?还是用标签解释合适。
这正是Python以值为中心的理念,不想c/c++系列,以变量为中心,切记切记,不要同名