zoukankan      html  css  js  c++  java
  • day10-函数进阶

    1.函数的命名空间

      命名空间分为三种:

      内置命名空间:

        是python解释器一启动就可以使用的名字,存储在内置命名空间中,启动解释器的时候被加载进内存里。比如list,tuple等等。

      全局命名空间:

        是在程序从上到下被执行的过程中依次加载进内存的,放置了我们设置的所有变量名和函数名。

    1 a = 1
    2 def func():
    3     print(a)
    4 fun()

      局部命名空间:

        就是函数内部定义的名字,当调用函数的时候 才会产生这个名称空间 随着函数执行的结束 这个命名空间就又消失了。

    1 def func():
    2     a = 1
    3     print(a)
    4 func()

      在局部:可以使用全局、内置命名空间中的名字。

      在全局:可以使用内置命名空间中的名字,但是不能用局部命名空间中的名字。

      在内置:不能使用局部和全局的名字的。

      注意以下几点:

      (1)在正常情况下,直接使用内置的名字

      (2)当我们在全局定义了和内置名字空间中同名的名字时,会使用全局的名字

      (3)当在一个函数内,有我要使用的变量名,我就回去使用它,没有的话,我就会向上级去寻找,找到了就用,找不到就继续寻找,如果还找不到,就报错

    1 def input():
    2     print('input')
    3 def func():
    4     input()
    5 func()
    6 # result:input

      (4)多个函数应该拥有多个独立的局部名字空间,不互相共享,即在函数A中不能使用函数B的局部变量

      (5)函数名()---函数的调用  等价于  函数的内存地址()---函数的调用

    1 def input():
    2     print('input')
    3 def func():
    4     print(input)# 这里输出的是函数的内存地址---<function input at 0x000001D05CEEC2F0>
    5 func()

    2.函数的作用域

      全局作用域 —— 作用在全局 —— 内置和全局名字空间中的名字都属于全局作用域  ——用globals()可以打印出来

      局部作用域 —— 作用在局部 —— 函数(局部名字空间中的名字属于局部作用域) ——用locals()可以打印出来

     1 a = 1
     2 b = 2
     3 def func():
     4     x = 'aaa'
     5     y = 'bbb'
     6     print(locals())
     7     print(globals())
     8 func()
     9 print(globals())
    10 print(locals()) #本地的
    11 # result:
    12 # {'x': 'aaa', 'y': 'bbb'}
    13 # {'__name__': '__main__', '__doc__': None, '__package__': None, '__loader__': <_frozen_importlib_external.SourceFileLoader object at 0x0000013D05871CF8>, '__spec__': None, '__annotations__': {}, '__builtins__': <module 'builtins' (built-in)>, '__file__': 'E:/Python/PyChram/WorkPlace/day01/Demo.py', '__cached__': None, 'input': <function input at 0x0000013D058AC2F0>, 'func': <function func at 0x0000013D05A5A8C8>, 'a': 1, 'b': 2}
    14 # {'__name__': '__main__', '__doc__': None, '__package__': None, '__loader__': <_frozen_importlib_external.SourceFileLoader object at 0x0000013D05871CF8>, '__spec__': None, '__annotations__': {}, '__builtins__': <module 'builtins' (built-in)>, '__file__': 'E:/Python/PyChram/WorkPlace/day01/Demo.py', '__cached__': None, 'input': <function input at 0x0000013D058AC2F0>, 'func': <function func at 0x0000013D05A5A8C8>, 'a': 1, 'b': 2}
    15 # {'__name__': '__main__', '__doc__': None, '__package__': None, '__loader__': <_frozen_importlib_external.SourceFileLoader object at 0x0000013D05871CF8>, '__spec__': None, '__annotations__': {}, '__builtins__': <module 'builtins' (built-in)>, '__file__': 'E:/Python/PyChram/WorkPlace/day01/Demo.py', '__cached__': None, 'input': <function input at 0x0000013D058AC2F0>, 'func': <function func at 0x0000013D05A5A8C8>, 'a': 1, 'b': 2}

      可以看到,内置函数中的locals打印的就是内置函数中定义的变量,而globals不论在哪里都打印全局变量,至于函数外的locals,他会将globals看作是一个笨的的变量,所以他打印出来的也是全局的。总结如下:

      (1)globals 永远打印全局的名字

      (2)locals输出什么,根据locals所在的位置

    1 a = 1
    2 def func():
    3     global a
    4     a = 2
    5 func()
    6 print(a)

      在上面代码中,在函数外,若是想要打印出函数内的变量,只要在函数内对这个变量加上一个global就行了,但是一般不建议这么做,因为别人看你的代码时,只会看到你定义的全局变量a = 1,谁会看到你函数内又将他变成了a = 2呢,所以建议不要使用global。

    3.函数的嵌套调用

    1 def max(a,b):
    2     return a if a>b else b
    3 def the_max(x,y,z):  #函数的嵌套调用
    4     c = max(x,y)
    5     return max(c,z)
    6 print(the_max(1,2,3))

    4.函数的嵌套定义

    a = 1
    def outer():
        a = 1
        def inner():
            a = 2
            def inner2():
                nonlocal a  #声明了一个上面第一层局部变量
                a += 1   #不可变数据类型的修改
            inner2()
            print('##a## : ', a)
        inner()
        print('**a** : ',a)
    
    outer()
    print('全局 :',a)
    # result:
    # ##a## :  3
    # **a** :  1
    # 全局 : 1

      这里展示的是函数的嵌套定义,同时也展示了nonlocal的使用方法。

      nonlocal可以使内部函数使用外部函数的变量。它的使用规则是找到距离当前函数上面最近一个函数的变量。比如说,nonlocal a,那么我在上面一个函数上找是否有a,有的话就用它,没有的话就继续找,直到有为止。

      nonlocal的声明,当修改这个变量后,会影响到被找到的那个函数的变量,也就是说,若你在内部函数修改了外部变量,则外部变量也会被修改。

      nonlocal只能作用于局部变量,不能作用于全局变量。

    5.函数的返回值

    def func():
        print(123)
    
    def wahaha(f):
        f()
        return f           #函数名可以作为函数的返回值
    qqxing = wahaha(func)   # 函数名可以作为函数的参数
    qqxing()
    # result:
    # 123
    # 123

     6.闭包

      所谓闭包就是内部函数调用外部函数的变量。

    def outer():
        a = 1
        def inner():
            print(a)#调用的是外部函数的a,所以这是一个闭包
        print(inner.__closure__)
    outer()#(<cell at 0x000001FAA0251918: int object at 0x00007FF96C376290>,)---在这里打印出来了一个cell什么的,就说明这个函数是闭包
    print(outer.__closure__)#None---这里输出为空,说明它不是一个闭包

      闭包的常用形式:

      在这里说明一下闭包的好处,当使用闭包之后,我就不用每次都去声明这个a变量,因为使用闭包之后,a就一直存在内存中,不会消失,即大大节省了时间。

    def outer():
        a = 1
        def inner():
            print(a)
        return  inner
    inn = outer()
    inn()

      闭包的简单用处:

    import urllib  #模块
    from urllib.request import urlopen
    ret = urlopen('https://www.52pojie.cn/').read()
    print(ret)

      上面代码是获取一个网页的源码。

    from urllib.request import urlopen
    def get_url():
        url = 'https://www.52pojie.cn/'
        def get():
            ret = urlopen(url).read()
            print(ret)
        return get
    get_re = get_url()
    get_re()

      使用闭包之后,我就不用每次都去声明这个url了,因为他一直在内存中,节省了事件。

  • 相关阅读:
    花匠
    积木
    Hello world
    老鼠走迷宫全部路径
    今天下午选做题目
    整数高精度运算——加法
    博客启航
    解线性不定方程
    关于完全背包问题
    关于最小代价子母树
  • 原文地址:https://www.cnblogs.com/missdx/p/10925226.html
Copyright © 2011-2022 走看看