学会Python的这五点搞定作用域

学会Python的这五点搞定作用域


1、块级作用域

想想此时运行下面的程序会有输出吗?执行会成功吗?


  1. #块级作用域 
  2.    
  3. if 1 == 1: 
  4.     name = "lzl" 
  5.    
  6. print(name) 
  7.    
  8.    
  9. for i in range(10): 
  10.     age = i 
  11.    
  12. print(age) 

我们先看下执行结果:


  1. C:/Users/L/PycharmProjects/s14/preview/Day8/作用域/main.py 
  2. lzl 

Process finished with exit code 0

代码执行成功,没有问题;在Java/C#中,执行上面的代码会提示name,age没有定义,而在Python中可以执行成功,这是因为在Python中是没有块级作用域的,代码块里的变量,外部可以调用,所以可运行成功;

2、局部作用域

回顾之前学过的知识,我们学函数的时候,函数是个单独的作用域,Python中没有块级作用域,但是有局部作用域;看看下面的代码

#局部作用域


  1. def  func(): 
  2.     name = "lzl" 
  3.    
  4. print(name) 

运行这段代码,想想会不会有输出?


  1. Traceback (most recent call last): 
  2.   File "C:/Users/L/PycharmProjects/s14/preview/Day8/作用域/main.py", line 23, in <module> 
  3.     print(name) 
  4. NameError: name 'name' is not defined 

运行报错,我相信这个大家都能理解,name变量只在func()函数内部中生效,所以在全局中是没法调用的;对上面代码做个简单调整,再看看结果如何?


  1. #局部作用域 
  2.    
  3. def  func(): 
  4.     name = "lzl" 
  5.    
  6. func()          #执行函数 
  7. print(name) 

对之前的代码添加了一句代码,在变量name打印之前,执行了一下函数,此时打印会不会有变化?


  1. Traceback (most recent call last): 
  2.   File "C:/Users/L/PycharmProjects/s14/preview/Day8/作用域/main.py", line 23, in <module> 
  3.     print(name) 
  4. NameError: name 'name' is not defined 

执行依然报错,还是回到刚才那句话:即使执行了一下函数,name的作用域也只是在函数内部,外部依然无法进行调用;把前两个知识点记住,接下来要开始放大招了

3、作用域链

对函数做下调整,看看下面的代码执行结果如何?


  1. #作用域链 
  2.    
  3. name = "lzl" 
  4. def f1(): 
  5.     name = "Eric" 
  6.     def f2(): 
  7.         name = "Snor" 
  8.         print(name) 
  9.     f2() 
  10. f1() 

学过函数,肯定知道最后f1()执行完会输出Snor;我们先记住一个概念,Python中有作用域链,变量会由内到外找,先去自己作用域去找,自己没有再去上级去找,直到找不到报错

4、终极版作用域

好,铺垫了够了,终极版的来了~~


  1. #终极版作用域 
  2.    
  3. name = "lzl" 
  4.    
  5. def f1(): 
  6.     print(name) 
  7.    
  8. def f2(): 
  9.     name = "eric" 
  10.     f1() 
  11.    
  12. f2() 

想想最后f2()执行结果是打印“lzl”呢,还是打印“eric”?记住自己的答案,现在先不把答案贴出来,先看看下面这段代码:


  1. #终极版作用域 
  2.    
  3. name = "lzl" 
  4.    
  5. def f1(): 
  6.     print(name) 
  7.    
  8. def f2(): 
  9.     name = "eric" 
  10.     return f1 
  11.    
  12. ret = f2() 
  13. ret() 
  14.    
  15. #输出:lzl 

执行结果为“lzl”,分析下上面的代码,f2()执行结果为函数f1的内存地址,即ret=f1;执行ret()等同于执行f1(),执行f1()时与f2()没有任何关系,name=“lzl”与f1()在一个作用域链,函数内部没有变量是会向外找,所以此时变量name值为“lzl”;理解了这个,那么刚才没给出答案的那个终极代码你也知道答案了。


  1. #终极版作用域  
  2.     
  3. name = "lzl"  
  4.     
  5. def f1():  
  6.     print(name)  
  7.     
  8. def f2():  
  9.     name = "eric"  
  10.     f1()  
  11.     
  12. f2()  
  13.     
  14. # 输出:lzl

是的,输出的是“lzl”,记住在函数未执行之前,作用域已经形成了,作用域链也生成了。

5、新浪面试题


  1. li = [lambda :x for x in range(10)] 

判断下li的类型?li里面的元素为什么类型?


  1. print(type(li)) 
  2. print(type(li[0])) 
  3.    
  4. <class 'list'> 
  5. <class 'function'> 

可以看到li为列表类型,list里面的元素为函数,那么打印list里面第一个元素的返回值,此时返回值为多少?


  1. #lambada 面试题 
  2.    
  3. li = [lambda :x for x in range(10)] 
  4.    
  5. res = li[0]() 
  6. print(res) 
  7.    
  8. #输出:9 

li第一个函数的返回值为9还不是0,记住:函数在没有执行前,内部代码不执行;博客里面的代码可以自己练练,加深下印象。


作者:daisy

来源:51CTO

上一篇:Oracle修改redo log大小的方法


下一篇:【Android 逆向】x86 汇编 ( 使用 IDA 解析 x86 架构的动态库文件 | 使用 IDA 打开动态库文件 | IDA 中查找指定的方法 )