1.函数对象
前面我们学习了关于Python中的变量类型,例如int、str、bool、list等等……,其实函数我们可以去理解为一种更加高级的变量类型,也就是函数对象,其实是一个更加高级的变量容器,我们可以去把函数对象当做一个变量去使用。
(1)函数对象的引用
一个函数对象的形成是先定义一个函数,然后写入函数体,最后把这个函数赋值给一个变量,也就是函数的对象,然后我们可以去通过这个函数变量来实现函数相应的功能。(先赋值,再调用)
示例1:
#定义一个函数对象fun def fun(): print('hello word',end=' ') print('wwww') fun() #直接调用这个函数对象
这种写法是最直接的,没有返回值。
示例2:
def fun(): print('ww') return 123 kun = fun() print(kun,type(kun)) #输出结果:ww # 123 <class 'int'>
这个写法是定义了一个kun在变量,作为fun函数的对象,我们可以去通过这个对象去处理这个函数的返回值。
示例3:
def fun(): return 123 kun = fun print(kun,type(kun)) #输出结果:<function fun at 0x00000154828AD280> <class 'function'>
这里我没有加上()所以输出结果是fun这个变量的地址,
示例4:
def fun(): return 123 kun = fun print(kun(),type(kun)) #输出结果:123 <class 'function'>
这里先定义一个kun变量为fun的函数对象,下面如果要去调用这个函数的话就必须加上括号,如果没有括号那这个对象只是函数的地址。
(2)函数可以放到序列里面
函数的返回值是一种变量类型,同时序列可以储存变量,所以函数是可以放到序列里面去
这里就以列表为例子:
def fun(): return '牛马' kun = fun li=[1,5,kun] print(li) print(li[2]()) #输出结果:[1, 5, <function fun at 0x000001BFB1ABD280>] # 牛马
(3)函数可以作为参数 , 传递给另一个函数
示例:
def fun(): return '牛马' def ff(st): print(f'我是{st}') ff(fun()) #输出结果:我是牛马
(4)函数可以作为另一个函数的返回值
def ff(): print('我爱你') return 520 def fun(): return ff() a=fun() print(a,type(a)) #输出结果:我爱你 # 520 <class 'int'>
这里是fun函数返回值为ff函数,实际上是经历过了fun函数的功能,然后再次经历ff函数的功能,最后返回的是ff函数的520,故a的结果是520
2.名称空间
数据的名称是储存到栈区,而数据的内容是储存到堆区,当我们要去使用数据的内容时,我们可以通过数据的名称来直接去表示数据的内容,也就是通过栈区去找到堆区的东西,然后在堆区拿出来去使用。栈区和堆区是有对应关系的。
1.内建名称空间
生命周期:会随着Python启动而生成,程序关闭会摧毁
保存的数据:内置函数(input,print....)
生成顺序:程序里最先生成的
2.全局名称空间
生命周期:会随着Python启动而生成,程序关闭会摧毁
保存的数据:函数名,变量名
生成顺序:在内建名称空间的后面
3.局部名称空间
生命周期:随着函数的调用而生成,函数调用接受后就关闭
保存的数据:函数内定义的名字
生成顺序:随着函数的调用而生成
所以,我们定义的函数是属于局部名称空间,当函数被调用完成了之后就会被销毁,里面变量储存到的栈堆区都会被销毁
3.作用域
(1)作用域的理解
作用域实际上就是变量的作用范围,之前我们学了C语言都知道,有全局变量和局部变量,比如在for循环里面,里面的变量是一种局部变量,我们如果在主函数外面去定义一个变量,那就是全局变量。Python也是一样的道理,变量也是有相应的作用域。而变量的作用域是由定义的位置决定的
全局变量与局部变量的名字相同的时候,实质上是两个不同的变量, 其栈堆区是完全不一样的!所以我们不可以通过局部来改变全局变量,否则就报错
示例1:
#定义一个全局变量a a=15 def fun(): a=999 print('hhh') fun() print(a) #输出结果:15
因为函数是放到栈堆区,当函数调用完成了之后返回一个值,这个函数已经被销毁了,那么里面的变量也会与之销毁,所以不会影响到全局变量,故a的值还是15
示例2:
#定义一个全局变量a a=15 def fun(): a=999 return a print(fun()) print(a) #输出结果:999 # 15
这个函数调用完成了之后就返回一个值,这个值函数内部的功能的值,但是并没有对这个函数以外的值进行修改,所以a的结果还是15,而函数的返回值是999
示例3:
b=15 def fun(): b=b+99 fun() print(b) #输出结果:系统报错
因为b是一个全局变量,而函数里面的b是一个重新定义的局部变量,所以我们不可以去直接修改这个局部变量b,因为这个b是没有初始数值的,而且类型也未定,所以无法进行运算处理
对比
b=15 def fun(): b=0 b+=99 print(b) fun() print(b) #输出结果:99 15
对于这个而言,我给了函数里面b一个初始化值,所以可以进行运算处理,但这个b与外部全局变量的b是完全不相同,是两个不同的变量
(2)作用域的转换
在C语言中我们可以去设置静态变量(static),从而使得这个变量可以在函数进行修改,同样Python中可以利用关键字去修改变量的作用域,把局部变量作用到全局变量,或者把全局变量作用降级为局部
1.global 关键字 (局部变全局)
在函数中,变量是这个函数的局部变量,只仅仅作用到函数内部,我们对这个变量进行修改是不会影响外面的变量的,如果把这个局部变量转换为全局变量的话,那么这变量就可以在函数里面进行修改,而且还可以影响到外部,当新定义global的变量名字与之前已有的变量名字出现重复的话,新的变量内容会把之前的给覆盖掉,而且新的全局变量也可以在函数内部进行修改,从而影响全局。
注意事项:这个关键字是作用到全局变量,这个全局变量的栈堆区不会被销毁
b=15 def fun(): b=123 fun() print(b) #函数里面的b与外面的b是不同性质的!!! #输出结果:15
对比
b=15 def fun(): global b b=123 fun() print(b) #输出结果:123
重新定义了一个global后,此时这个b已经是一个新的全局变量因为前面已经有了一个全局变量b,所以这个新的全局变量会因为名字相同会把之前的全局变量b给直接覆盖掉(类似于类型相同且名字相同的文件会被新的给覆盖),故输出结果是新的全局变量b的结果:123
2.nonlocal 关键字 (全局变局部)
功能:降权,全局变成局部
前面讲到过,已知一个全局变量,但是我不可以在函数里面对这个全局变量进行修改,而且函数里面的变量都是新的变量,既是跟外面的全局变量名字一模一样,但是其储存的栈堆区是完全不同的,那么当我想去通过一个局部函数来修改这个全局变量的话,这时候我就可以通过nonlocal关键字来进行 降维处理 。
注意事项:这个关键字是作用于嵌套函数的外部变量(相较于内部函数,外部函数是“全局”),这个变量的栈堆区会被销毁
def fun(): w=99 def fun1(): w=100 fun1() return w print(fun()) #输出结果:99
VS
def fun(): w=99 def fun1(): nonlocal w w=100 fun1() return w print(fun()) #输出结果:100
这里,可以看出,w=99相对应fun1()是一个全局变量,而fun1()内部的w相对应fun()是一个局部变量,当我用nonlocal关键字去定义了fun()的变量w之后,这个w变量已经是属于fun1()函数内部的变量,这时候我们可以去修改这个w的值,从而去影响外部函数,最后w的返回值也就是被修改后的100