zoukankan      html  css  js  c++  java
  • 函数参数以及名称空间作用域

    在函数定义阶段括号内定义的参数,称之为形式参数,简称形参,本质就是变量名。

    def func(x, y):
        print(x)
        print(y)
    

    1.2 实参

    func(1, 2)
    

    二、位置参数

    def func(x, y):
        print(x)
        print(y)
    

    特点:按照位置定义的形参,都必须被传值,多一个不行,少一个也不行。

    2.2 位置实参

    func(1, 2)
    

    特点:按照位置为对应的形参依次传值。

    关键字实参

    在调用函数时,按照key=value的形式为指定的参数传值,称为关键字实参。

    func(y=2, x=1)
    

    特点:可以打破位置的限制,但仍能为指定的形参赋值。

    注意:

    1. 可以混用位置实参和关键字实参,但是位置实参必须在关键字实参的左边。
    2. 可以混用位置实参和关键字实参,但不能对一个形参重复赋值。
    func(x, y=2)
    func(y=2, x)  # SyntaxError: positional argument follows keyword argument
    func(x, x=1)  # NameError: name 'x' is not defined
    

    在定义阶段,就已经被赋值。

    def func(x, y=10):
        print(x)
        print(y)
        
    func(2)
    

    默认形参

    特点:在定义阶段就已经被赋值,意味着在调用时可以不用为其赋值。

    注意:

    1. 位置形参必须放在默认形参的左边。
    2. 默认形参的值只在定义阶段赋值一次,也就是说默认参数的值在函数定义阶段就已经固定了。
    m = 10
    
    
    def foo(x=m):
        print(x)
    
    
    m = 111
    foo()  # 10
    
    1. 默认参数的值通常应该是不可变类型。
    # 演示形参是可变类型
    def register(name, hobby, hobby_list=[]):
        hobby_list.append(hobby)
        print(f"{name} prefer {hobby}'")
        print(f"{name} prefer {hobby_list}")
    
    
    register('nick', 'read')
    register('tank', 'zuipao')
    register('jason', 'piao')
    nick prefer read'
    nick prefer ['read']
    tank prefer zuipao'
    tank prefer ['read', 'zuipao']
    jason prefer piao'
    jason prefer ['read', 'zuipao', 'piao']
    # 修改形参是可变类型代码
    def register(name, hobby, hobby_list=None):
        if hobby_list is None:
            hobby_list = []
        hobby_list.append(hobby)
        print(f"{name} prefer {hobby}'")
        print(f"{name} prefer {hobby_list}")
    
    
    register('nick', 'read')
    register('tank', 'zuipao')
    register('jason', 'piao')
    nick prefer read'
    nick prefer ['read']
    tank prefer zuipao'
    tank prefer ['zuipao']
    jason prefer piao'
    jason prefer ['piao']
    

    五、总结

    1. 大多数情况的调用值一样,就应该将该参数定义成位置形参
    2. 大多数情况的调用值一样,就应该将该参数定义成默认形参

    可变长参数

    可变长参数:指的是在调用函数时,传入的参数个数可以不固定

    调用函数时,传值的方式无非两种,一种是位置实参,另一种是关键字实参,因此形参也必须得有两种解决方法,以此来分别接收溢出的位置实参(*)与关键字实参(**)

    一、可变长形参之*

    def sum_self(*args):
        res = 0
        for num in args:
            res += num
        return res
    
    
    res = sum_self(1, 2, 3, 4)
    print(res)
    10
    

    二、可变长实参之*

    def func(x, y, z, *args):
        print(x, y, z, args)
    
    
    func(1, *(1, 2), 3, 4)
    1 1 2 (3, 4)
    

    三、可变长形参之**

    def func(**kwargw):
        print(kwargw)
    
    
    func(a=5)
    {'a': 5}
    

    四、可变长实参之**

    def func(x, y, z, **kwargs):
        print(x, y, z, kwargs)
    
    
    func(1, 3, 4, **{'a': 1, 'b': 2})
    1 3 4 {'a': 1, 'b': 2}
    

    五、可变长参数应用

    args: ()
    kwargs: {'name': 'nick', 'sex': 'male', 'age': 19}
    name: nick, age: 19, sex: male
    

    六、命名关键字形参

    def register(x, y, **kwargs):
        if 'name' not in kwargs or 'age' not in kwargs:
            print('用户名和年龄必须使用关键字的形式传值')
            return
        print(kwargs['name'])
        print(kwargs['age'])
    
    
    register(1, 2, name='nick', age=19)
    nick
    19
    

    命名关键字形参:在函数定义阶段,*后面的参数都是命名关键字参数。

    特点:在传值时,必须按照key=value的方式传值,并且key必须命名关键字参数的指定的参数名。

    def register(x, y, *, name, gender='male', age):
        print(x)
        print(age)
    
    
    register(1, 2, x='nick', age=19)  # TypeError: register() got multiple values for argument 'x
    

    函数内部的函数只能在函数内部调用,不能在函数外部调用,通过接下来的学习你将会知道为什么会出现这种情况。

    def f1():
        def f2():
            print('from f2')
        f2()
    
    f2()  # NameError: name 'f2' is not defined
    

    名称空间和作用域

    一、名称空间
    1.1 内置名称空间

    内置名称空间:存放Pyhton解释器自带的名字,如int、float、len

    生命周期:在解释器启动时生效,在解释器关闭时失效

    1.2 全局名称空间

    生命周期:在文件执行时生效,在文件执行结束后失效

    x = 1
    
    
    def func():
        pass
    
    
    l = [1, 2]
    
    if 3 > 2:
        if 4 > 3:
            z = 3
    

    1.3 局部名称空间

    生命周期:在文件执行时函数调用期间时生效,在函数执行结束后失效

    def f1():
        def f2():
            print('from f2')
        f2()
    
    f1() 
    

    46名称空间与作用域-简单.png?x-oss-process=style/watermark

    1.4 加载顺序
    1.5 查找顺序

    由于名称空间是用来存放变量名与值之间的绑定关系的,所以但凡要查找名字,一定是从三者之一找到,查找顺序为:
    从当前的所在位置开始查找,如果当前所在的位置为局部名称空间,则查找顺序为:局部--》全局--》内置。

    x = 1
    y = 2
    len = 100
    
    
    def func():
        y = 3
        len = 1000
        print(f"y: {y}")
        print(f"len: {len}")
        # print(a)  # NameError: name 'a' is not defined
    
    
    func()
    y: 3
    len: 1000
    x = 1
    
    
    def func():
        print(x)
    
    
    x = 10
    func()
    10
    

    二、作用域

    46名称空间和作用域-地图.jpg?x-oss-process=style/watermark

    2.1 全局作用域

    # 全局作用域
    x = 1
    
    
    def bar():
        print(x)
    
    
    bar()
    1
    

    2.2 局部作用域

    # 局部作用域
    def f1():
        def f2():
            def f3():
                print(x)
            x = 2
            f3()
        f2()
    
    
    f1()
    2
    

    2.3 注意点

    # 作用域注意点
    x = 1
    
    
    def f1():  # 定义阶段x=1
        print(x)
    
    
    def f2():
        x = 2
        f1()
    
    
    f2()
    1
    

    2.4 函数对象+作用域应用

    from inner
    

    三、补充知识点

    x = 1
    
    
    def f1():
        x = 2
    
        def f2():
            #         global x  # 修改全局
            x = 3
        f2()
    
    
    f1()
    print(x)
    1
    x = 1
    
    
    def f1():
        x = 2
    
        def f2():
            global x  # 修改全局
            x = 3
        f2()
    
    
    f1()
    print(x)
    3
    

    3.2 nonlocal关键字

    x = 1
    
    
    def f1():
        x = 2
    
        def f2():
            #         nonlocal x
            x = 3
    
        f2()
        print(x)
    
    
    f1()
    2
    x = 1
    
    
    def f1():
        x = 2
    
        def f2():
            nonlocal x
            x = 3
    
        f2()
        print(x)
    
    
    f1()
    3
    

    3.3 注意点

    lis = []
    
    
    def f1():
        lis.append(1)
    
    
    print(f"调用函数前: {lis}")
    f1()
    print(f"调用函数后: {lis}")
    调用函数前: []
    调用函数后: [1
    
  • 相关阅读:
    Spring源码情操陶冶-AbstractApplicationContext#obtainFreshBeanFactory
    Spring源码情操陶冶-AbstractApplicationContext#prepareRefresh
    Spring源码情操陶冶-AbstractApplicationContext
    Spring源码情操陶冶-ContextLoader
    Spring源码情操陶冶-ContextLoaderListener
    Spring mybatis源码篇章-MapperScannerConfigurer
    Spring mybatis源码篇章-动态SQL节点源码深入
    Spring mybatis源码篇章-动态SQL基础语法以及原理
    Spring mybatis源码篇章-Mybatis的XML文件加载
    Spring mybatis源码篇章-Mybatis主文件加载
  • 原文地址:https://www.cnblogs.com/zhouxuchong/p/11544740.html
Copyright © 2011-2022 走看看