7.10 函数的嵌套等

一。命名关键字参数。(了解)

  1.在函数阶段,写在*与** 可变长参数之间的形参称为命名关键字参数。

  在给命名关键字参数传值时,只能用关键字为其传值。诸如以下函数的形参

def func(x,y=1,*args,z=3,m,**kwargs):
    print(x,y)
    print(args)
    print(z,m)
    print(kwargs)

  z=3,m,这些函数型参看似会报错,因为关键字参数写在了位置参数的前面,但其实是在*args和**kwargs之间的位置参数。所以,给其传参时统一使用关键字传参

func(1,2,1,2,3,4,5,6,7,78,8,9,0,z=69,m=999,o=666999,l = 999666)

二。函数对象

  1.函数是第一类对象:函数名指向的值可以被当成参数传递。

  如同变量名一样,函数的名字也指向一个地址,打印变量名就是打印其地址

def func():
    print('from func')
# print(func)
print(id(func))
f = func
func()
print(f)
f()

  像函数名被print时,输出的就是它的详细内存地址,而当函数名接触到()时会被优先执行,包括使用另一变量指向该函数名,如f。运行结果如下

1982808399520
from func
<function func at 0x000001CDA8971EA0>
from func

  2.函数名可以被当做参数传递给其他函数

def func():
    print('from func')

def index(args):
    print(args)
    args()
    print('from index')
# index(1)
index(func)

  上代码中在函数调用阶段,func就被以函数名的方式赋值给index的参数,而index将其函数名打印,输出的也就是func的内存地址,再运行该函数,最后输出自己的打印,所以输出的结果应该是:

<function func at 0x000001E05AB91EA0>
from func
from index

  3.函数名可以 被当作函数的返回值。

def index():
    print('index')

def func():
    print('func')
    return index
res = func()
print(res)
res()

  上述代码中,func将index函数名返回给func函数,所以func()所代表的返回值 就是index并执行,将index赋值额、给res并打印,其结果必定是index的内存地址。res()代表的是执行index函数所以结果如下:

func
<function index at 0x000002238A071EA0>
index

  4.函数名可以作为容器类型的参数

def func():
    print('func')
print(func())
l = [1,2,func,func()]  # [1,2,<function func at 0x000001F7D79899D8>,None]
print(l)

  函数名所代表的集合到列表中 ,可以被用来调用。运行结果如下

func
None
func
[1, 2, <function func at 0x00000267F2071EA0>, None]

  这种特性可以被用来编写可添加式菜单,如

func_dict = {
    '1':register,
    '2':login,
    '3':transfer,
    '4':shopping,
    '5':pay,
}
while True:
    print(msg)
    choice = input('请现在你想要执行的功能>>>:').strip()
    if choice in func_dict:
        func_dict.get(choice)()  # 函数名()

  将各个函数模块都添加到字典,然后根据用户输入的key值获取其 存储的函数名,在以函数名()的形式执行代码。如此实现可添加的菜单。

三。函数的嵌套调用

  在函数的编写中,也可以调另外一个函数,只要在调用主函数之前定义该函数即可正常运行:

def my_max(x,y):
    if x > y:
        return x
    return y

def my_max4(a,b,c,d):
    res1 = my_max(a,b)
    res2 = my_max(res1,c)
    res3 = my_max(res2,d)
    return res3
print(my_max4(1,2,10,4))
#输出结果>>>10

四.函数的嵌套定义

  在函数的定义阶段,也可以通过其他函数的定义来实现某种功能。

def outer():
    x = 1
    print('outer')
    def inner():
        print('inner')
    # print(inner)
    return inner

res = outer()
print(res)
res()
#输出结果>>>outer
#<function outer.<locals>.inner at 0x000001875EBBC9D8>
#inner

五。名称空间:

  名称空间里存放了变量名与变量值的内存地址的绑定关系的地方。

  想要访问一个变量的值,必须先去名称空间中拿到对应的名字,才能够访问该变量的值。

  名称空间有很多种:

x = 1
if 1 ==1 :
    y = 2
# print(y)
# while True:
#     z = 3
for i in [1,2,3]:
    print(i)
print(i)
def func():
    username = 'jason'
print(x)
func()

  名称空间分3类:

  1.内置名称空间:python解释器提前给你定义好的名字(已经存放到内置名称空间里)

  如len,max min 等内置函数的名字都在内置名称空间

  2.全局名称空间:文件级别的代码

  如上函数所示,x,y,等都可以说是全局变量。

  3.局部名称空间:函数体内创建的名字都属于局部名称空间的。

  如上所示代码中username等。

  生命周期:每种名称都有他的作用时间

  1.内置名称空间:只要python解释器一启动,该名称空间就会立马创建,当解释器关闭时,内置名称空间也会自动销毁

  2.全局名称空间:只要右键运行py文件,就会自动创建全局变量空间:py程序运行结束时自动销毁

  3.局部名称空间:函数被调用的时候自动创建。函数指向结束变立即销毁,这种方式被称为动态创建,动态销毁。

六。名称的查找顺序

  在名称查找的过程中顺序如下

  当在局部函数中寻找名称时,会按照局部名称空间-》全局名称空间-》内置名称空间-》的顺序查询名称,而且值得注意的是当查询顺序确定以后就不会修改,所以

def func():
    x = 1
    def index():
        print(x)  # 获取全局找x
    return index

res = func()
x = 999
res()
#输出结果>>>1

七。作用域

  1.在其名称的使用时,内置名称空间,全局名称空间是全局有效的,而局部名称空间只在该局域内有效。

   修改值的时候也是如此,在局部时只能修改局部的数据,不能修改全局的数据,各个局部之间也不能修改,所以,要想实现局部修改全局的功能需要两个关键字的帮助:nonlocal,global

  global

  代表在局部修改全局的变量,如果想要修改多个值,之间用逗号隔开

x = 1  # 不可变类型
username = 'jason'
def func():
    # x.append('嘿嘿嘿')
    global x,username  # 修改全局变量 而不是创建局部名称空间
    x = 999
    username = 'egon'
func()
print(x)
print(username)
#输出结果>>>999
#egon

  像这样的修改如在函数中修改全局变量x,和username的值。使用的就是global

  nonlocal

  当想要在函数中定义的函数里改变原来定义的函数中的变量时,,就是使用nonlocal关键字实现,如果想改变多个值,值与值之间要加逗号。

  

def func():
    x = 1
    def index():
        nonlocal x
        x = 2
    index()
    print(x)
func()
#输出结果>>>2

 

posted on 2019-07-10 17:34  一只萌萌哒的提莫  阅读(186)  评论(0编辑  收藏  举报