关于Python函数对象的名称空间和作用域

作者:灰勒塔德 时间:2023-08-15 02:00:42 

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关键字来进行 降维处理 。

注意事项:这个关键字是作用于嵌套函数的外部变量(相较于内部函数,外部函数是&ldquo;全局&rdquo;),这个变量的栈堆区会被销毁

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

来源:https://blog.csdn.net/m0_73633088/article/details/129018275

标签:Python,函数,作用域,名称空间
0
投稿

猜你喜欢

  • SQL Server数据库涉及到的数据仓库概念

    2009-01-15 12:58:00
  • jquery ajax 局部无刷新更新数据的实现案例

    2024-05-02 17:05:08
  • Python编程实现线性回归和批量梯度下降法代码实例

    2021-10-13 07:33:27
  • Python实现的径向基(RBF)神经网络示例

    2022-03-06 23:44:35
  • Python判断字符串是否xx开始或结尾的示例

    2023-02-07 18:35:28
  • 分析Python的Django框架的运行方式及处理流程

    2022-03-13 14:47:50
  • 详解Python如何求不同分辨率图像的峰值信噪比

    2023-03-25 09:51:58
  • sql中的常用的字符串处理函数大全

    2024-01-19 21:37:41
  • 基于Python的ModbusTCP客户端实现详解

    2022-02-03 10:54:25
  • Navicat连接mysql报错2003(10060)的解决方法

    2024-01-25 06:08:14
  • 基于PyTorch实现EdgeCNN的实战教程

    2023-12-30 22:28:20
  • 详解python UDP 编程

    2023-06-11 22:31:44
  • Python实现的生成自我描述脚本分享(很有意思的程序)

    2023-08-14 20:21:06
  • JS forEach跳出循环2种实现方法

    2024-04-29 13:19:28
  • Python3 assert断言实现原理解析

    2023-06-11 20:39:33
  • JS中ESModule和commonjs介绍及使用区别

    2023-10-20 22:23:51
  • php中的登陆login

    2023-10-08 10:49:30
  • pytorch查看torch.Tensor和model是否在CUDA上的实例

    2023-06-16 16:41:22
  • iframe全跨域高度自适应解决方案

    2008-12-21 16:16:00
  • Go 结构体、数组、字典和 json 字符串的相互转换方法

    2024-05-05 09:26:42
  • asp之家 网络编程 m.aspxhome.com