python学习第十三天

一、可变长的参数

*与**在形参与实参中的应用

可变长指的是参数的个数不固定

站在实参的角度,实参是用来为形参赋值的,如果实参的个数不固定,那么必须要有对应的形参能够接收溢出实参

1.1、在形参名前加*

*会把溢出的位置实参存成元组,然后赋值其后的形参名

def func(x,*y):  # y=(2,3,4)
    print(x)
    print(y)
func(1,2,3,4)
func(1)
func() # 报错,位置形参x必须被传值

def add(*nums):
    res=0
    for num in nums:
        res+=num
    return res

res=add(1,2,3)
print(res)

1.2、在形参名前加**

**会把溢出的关键字实参存成字典,然后赋值其后的形参名

def func(x, **y):  # y=(2,3,4)
    print(x)
    print(y)


func(1,a=111,b=222,c=333)
func(a=111, b=222, x=1, c=333)

1.3、在实参前加*

*会把其后的值打散成位置实参

def func(x, y, z):
    print(x, y, z)
nums = [1, 2, 3]
func(*nums)  # func(1,2,3)

1.4、在实参前加**

**会把其后的值打散关键字实参

def func(x, y, z):
    print(x, y, z)
dic = {'y': 111, 'z': 222, 'x': 333}
func(**dic)  # func(y=111,z=222,x=333)

1.5、在形参与实参中混用*与**

def index(x,y,z,a,b,c):
    print("index===>",x,y,z,a,b,c)

def wrapper(*args, **kwargs):  # args=(1, 2, 3,)   kwargs={"a":1,"b":2,"c":3}
    index(*args, **kwargs)  # index(*(1, 2, 3,),**{"a":1,"b":2,"c":3})
                           #  index(1,2,3,c=3,b=2,a=1)


wrapper(1, 2, 3, a=111, b=222, c=333)

1.6、函数参数总结(了解)

def func(x,y=222,*args,**kwargs):
    pass

# 命名关键字形参(**)
def func(x, y=222, *args, n=777,m, **kwargs):  # m,n必须按照关键字实参的格式为其赋值
    print(x)  # 1
    print(y)  # 2
    print(args)  # (3,4,5)
    print("m===>", m)
    print("n===>", n)
    print(kwargs)


# func(1,2,3,4,5,6666666)
# func(1,2,3,4,5,m=6666666)
func(1, 2, 3, 4, 5, n=88888,m=6666666, a=11, b=22, c=33)

二、函数对象

函数对象指的是函数可以被当成变量去使用

2.1、可以被赋值

def foo():  # foo = 函数的内存地址
    print('from foo')
f = foo
print(f is foo)
f()

2.2、可以当作参数传给一个函数

def foo():
    print('form foo')
def bar(func):
    print(func)
    func()
bar(foo)

2.3、可以当成一个函数的返回值

def foo():
    print('form foo')
def bar(func):
    return func
res=bar(foo)
print(res) # <function foo at 0x000001AAF636AA60>

2.4、可以当成容器类型的元素

def foo():
    print('form foo')
l=[foo]
print(l)
l[0]()
运行结果:
[<function foo at 0x000001EC8C8AAA60>]
form foo

示例:

def login():
    print('登录功能......')

def withdraw():
    print('提现功能......')

def transfer():
    print('转账功能......')

def recharge():
    print('充值功能')

func_dic={
    "1": [login,"登录"],
    "2": [withdraw,"提现"],
    "3": [transfer,"转账"],
    "4": [recharge,"充值"]
}

# func_dic["1"][0]()


while True:
    print("0    退出")
    for k in func_dic:
        print("%s    %s" %(k,func_dic[k][1]))

    choice = input("请输入你的指令编号: ").strip()
    if choice == "0":
        break
    if choice in func_dic:
        func_dic[choice][0]()
    else:
        print('输入的指令不存在')

三、函数嵌套

3.1、函数的嵌套调用

def bar():
    print('from bar')

def foo():
    print('from foo')
    bar()

foo()

案例:

def max2(x,y):
    if x > y:
        return x
    else:
        return y

def max4(a,b,c,d):
    res1 = max2(a,b)
    res2 = max2(res1,c)
    res3 = max2(res2,d)
    print(res3)

max4(1,2,3,4)

3.2、函数的嵌套定义

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

    # f2 = 函数的内存地址
    def f2():
        print("from f2")

f1()

定义在函数内的函数的特点:正常情况只能再函数体内调用

from math import pi

def circle(radius,mode=0):
    def perimiter(radius):
        return 2 * pi * radius

    def area(radius):
        return pi * (radius ** 2)

    if mode == 0:
        return perimiter(radius)
    elif mode == 1:
        return area(radius)

res1=circle(3,0)
res2=circle(3,1)
print(res1)
print(res2)
def func():
    x = 10
    print(x)
    def f2():
        print('from f2')
    f2()
func()
运行结果:
10
from f2

四、名称空间与作用域

4.1、名称空间

4.1.1、内置名称空间

存放的是内置的名字,如print\input\len

生命周期: 解释器启动则产生,解释器关闭则销毁

4.1.2、全局名称空间

存放的是顶级的名字

生命周期: python程序运行时则产生,python程序结束则销毁

x = 10

def func():
    x = 111
    print(x)


if 1:
    y = 6666

4.1.3、局部名称空间

函数内的名字

生命周期: 调用函数时则产生,函数调用结束则销毁

4.2、名字的查找优先级

从当前位置往外查找,如果当前是在局部:局部名称空间->全局名称空间->内置名称空间
从当前位置往外查找,如果当前是在全局:全局名称空间->内置名称空间

示范1:

def func():
    len = 222
    # print(len) # 222

# len = 111

func()

print(len) # 111

名称空间可以理解为一层套一层的关系,问题是嵌套关系是在函数定义阶段生成的,还是在函数调用阶段生成的???(嵌套关系是在函数定义阶段生成的)

x = 111

def foo():
    print(x)

def bar(f):
    x=222
    f()

bar(foo) # 111

一个非常重要的结论:名称空间的嵌套关系是函数定义阶段(即扫描语法时)就固定死的,与函数的调用位置无关

x = 111
def func():
    print(x)
    x=2222 # 报错

func()

练习:

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

    f2()

f1()

4.3、作用域

4.3.1、全局范围/全局作用域

内置名称空间+全局名称空间
特点:全局存活,全局有效

案例1:
x = 10

def func(x): # x = 值10的内存地址
    # x = 值10的内存地址
    x = 20

func(x) # func(值10的内存地址)
print(x)


案例2
x = [11,22,33]

def func(x): # x = 列表[11,22,33]的内存地址
    # x = 列表[11,22,33]的内存地址
    # x=444444444444
    x[0] = 66666

func(x) # func(列表[11,22,33]的内存地址)
print(x)

4.3.2、局部范围/局部作用域

局部名称空间

特点:临时存活,局部有效

x = [11,22,33]
def func():
    x[0] = 66666

func()
print(x) # [66666, 22, 33]

4.3.3、了解global()、nonlocal(***)

案例1:
x = 10
def func():
    global x
    x=22

func()
print(x) # 22

案例:2:nonlocal声明名字是来自于外层函数的(***)
x = 10
def f1():
    x=111

    def f2():
        nonlocal x
        x=222

    f2()
    print(x)


f1()
print(x)

运行结果:
222
10
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值