016.Python闭包函数以及locals和globals
一 闭包函数
内函数使用了外函数的局部变量,并且外函数把内函数返回出来的过程叫做闭包,这个内函数叫做闭包函数
1.1 闭包函数语法
def outer():
a = 5
def inner():
print(a)
return inner
对比正常的局部变量
局部变量的生命周期最短,在调用结束之后,立即释放.
def func():
a = 5
print(a)
func()
print(a)
执行
1.2 闭包函数的定义
def bibao():
people = "孙仲谋"
def caocao_say():
print("生子当如%s" % (people))
return caocao_say func = bibao()
# func = caocao_say
func()
执行
[root@node10 python]# python3 test.py
生子当如孙仲谋
1.3 闭包函数的特点(升级)
- 内函数使用了外函数的局部变量,外函数的局部变量与内函数发生绑定,延长该变量的生命周期(实际内存给它存储了这个值,暂时不释放)
示例
def grade(str1):
def func_in(str2):
print(str1+str2)
return func_in a=grade('小李:')
a('语文:118') b=grade('小红:')
b('语文:109')
b('数学:98')
执行
[root@node10 python]# python3 test.py
小李:语文:118
小红:语文:109
小红:数学:98
示例2
def family():
jiejie = "小红"
meimei = "小明"
money = 100 def jiejie_hobby():
nonlocal money
money -=40
print("姐姐%s喜欢买零食,买了辣条钱还剩下%s" % (jiejie,money))
return money def meimei_hobby():
nonlocal money
money -= 30
print("妹妹%s喜欢买衣服,还剩下%s" % (meimei,money))
return money def big_manager():
return (jiejie_hobby,meimei_hobby) return big_manager func = family()
# func = big_manager
print(func,type(func))
# func() = big_manager()
tup = func()
print(tup,type(tup)) # 调用姐姐
# jiejie = tup[0]
# jiejie() res1 = tup[0]()
print(res1)
# 调用妹妹
# meimei = tup[1]
# meimei() res2 = tup[1]()
print(res2)
执行
[root@node10 python]# python3 test.py
<function family.<locals>.big_manager at 0x7fee9c4f21e0> <class 'function'>
(<function family.<locals>.jiejie_hobby at 0x7fee9c4f20d0>, <function family.<locals>.meimei_hobby at 0x7fee9c4f2158>) <class 'tuple'>
姐姐小红喜欢买零食,买了辣条钱还剩下60
60
妹妹小明喜欢买衣服,还剩下30
30
示例3
def outer(val):
def inner(num):
return val + num
return inner func = outer(10)
# func = inner
res = func(5)
print(res)
执行
[root@node10 python]# python3 test.py
15
过程
func = outer(10)
val 形参接收到实参值10
因为内函数使用了外函数的局部变量val,val与内函数发生绑定,延长val的生命周期
res = func(5)
num 形参接收到实参值5
return 10 + 5 => return 15 返回到函数的调用处
func(5) 是调用处 所以
res = 15
1.4 获取闭包函数使用的变量
闭包函数.__closure__ 返回单元cell , cell 里面存的是对应的绑定变量
res = func.__closure__ # 获取到一个元组
res = res[0].cell_contents # res[0] 获取元组当中的第一个值 是一个cell单元 通过单元.cell_contents来获取其中的值,就会知道绑定的变量是谁了. cell_contents是一个属性
print(res,type(res))
# print(res.cell_content)
# (<cell at 0x000001D6DAE17708: int object at 0x000000005EC26D30>,)
示例
def outer(val):
def inner(num):
return val + num
return inner func = outer(10)
# func = inner
res = func(5)
print(res)
print(func.__closure__)
print(func.__closure__,type(func.__closure__))
res = func.__closure__[0]
print(res,type(res)) res = func.__closure__[0].cell_contents
print(res,type(res))
执行
[root@node10 python]# python3 test.py
15
(<cell at 0x7fe07763b7f8: int object at 0x7fe07750c7e0>,)
(<cell at 0x7fe07763b7f8: int object at 0x7fe07750c7e0>,) <class 'tuple'>
<cell at 0x7fe07763b7f8: int object at 0x7fe07750c7e0> <class 'cell'>
10 <class 'int'>
1.5 闭包的意义
- 闭包可以优先使用外函数中的变量,并对闭包中的值起到了封装保护的作用.外部无法访问.
模拟鼠标点击次数
num = 0
def click():
global num
num += 1
return num
res = click()
res = click()
res = click()
# num = 100 恶意串改,我们获取的值就会出现错误
res = click()
res = click()
print(res)
使用闭包函数重写鼠标点击次数
def click():
x = 0
def func():
nonlocal x
x+=1
return x
return func click = click()
res = click()
res = click()
res = click()
x = 100 #恶意修改,但是不影响结果
res = click()
res = click()
print(res)
执行
[root@node10 python]# python3 test.py
5
二 locals 和 globals
2.1 locals()
返回字典,获取当前作用域的所有内容
- 如果在函数里:获取locals()调用之前,该作用域出现的内容
- 如果在函数外:获取locals()打印返回值之前,该作用域出现的内容
a = 1
b = 2
print(locals())
执行
[root@node10 python]# python3 test.py
{'__name__': '__main__',
'__doc__': None, 如果在函数里:获取locals()调用之前,该作用域出现的内容\n 如果在函数外:获取locals()打印返回值之前,该作用域出现的内容\n',
'__package__': None,
'__loader__': <_frozen_importlib_external.SourceFileLoader object at 0x7ffa9ddb7208>,
'__spec__': None, '__annotations__': {},
'__builtins__': <module 'builtins' (built-in)>,
'__file__': 'test.py',
'__cached__': None,
'a': 1,
'b': 2,
'func': <function func at 0x7ffa9de8ae18>, #全局中又这个函数,但是函数内部的局部变量不能获取
'res': {...},
'c': 3}
所在作用域是局部命名空间,获取locals() 调用之前所出现的所有局部命名空间内容
c=3
d = 4
def func():
a = 1
b = 2 res = locals()
c = 3
print(res) f = 5
func()
执行
[root@node10 python]# python3 test.py
{'b': 2, 'a': 1}
2.2 globals()
返回字典,获取全局作用域的所有内容
- 如果在函数里: 获取globals()调用之前,全局作用域出现的内容
- 如果在函数外: 获取globals()打印返回值之前,全局作用域出现的内容
globals在全局作用域中,只获取globals 打印返回值之前的所有全局空间的内容
a = 1
b = 2
res = globals()
print(res)
c = 3
执行
[root@node10 python]# python3 test.py
{'__name__': '__main__',
'__doc__': None,
'__package__': None,
'__loader__': <_frozen_importlib_external.SourceFileLoader object at 0x7fcd746f5208>,
'__spec__': None,
'__annotations__': {},
'__builtins__': <module 'builtins' (built-in)>,
'__file__': 'test.py',
'__cached__': None,
'a': 1,
'b': 2,
'res': {...}}
globals在局部作用域中,也仍然获取全局空间的所有内容,但是是在globals,打印返回值之前的所有.
c = 13
d = 14
def func():
a = 11
b = 12
res = globals()
print(res)
f = 19
func()
执行
[root@node10 python]# python3 test.py
{'__name__': '__main__',
'__doc__': None,
'__package__': None,
'__loader__': <_frozen_importlib_external.SourceFileLoader object at 0x7f16a5144208>,
'__spec__': None,
'__annotations__': {},
'__builtins__': <module
'builtins' (built-in)>,
'__file__': 'test.py',
'__cached__': None,
'c': 13,
'd': 14,
'func': <function func at 0x7f16a5217e18>,
'f': 19}
只获取全局,没有ab的值
动态创建全局变量 利用globals
- globals 返回的是一个系统的全局字典,键是变量名,值是该标量所指向的值
dic = globals()
dic['hero'] = "风流倜傥,高大威猛,威武帅气,万人迷"
print(hero) # 动态创建5个全局变量p1~p5
def func():
for i in range(1,6):
dic["p%d" % (i)] = i func()
print(p1)
print(p2)
print(p3)
print(p4)
print(p5)
执行
[root@node10 python]# python3 test.py
风流倜傥,高大威猛,威武帅气,万人迷
1
2
3
4
5
016.Python闭包函数以及locals和globals的更多相关文章
- python学习笔记012——locals与globals
1 定义 globals() 功能:收集全局变量参数:无返回值:得到一个收集全局变量的字典(会包含系统的内置变量) locals() 功能:收集局部变量参数:无返回值:得到一个收集局部变量的字典 a ...
- python之函数用法locals()
# -*- coding: utf-8 -*- #python 27 #xiaodeng #python之函数用法locals() #locals() #说明:查找局部变量,返回一个名字/值对的字典对 ...
- python闭包函数与装饰器
目录 闭包函数 闭包概念 实际应用 装饰器 简介 简单版本装饰器 进阶版本装饰器 完整版本装饰器 装饰器模板 装饰器语法糖 装饰器修复技术 问题 答案 闭包函数 闭包概念 闭:定义在函数内部的函数 包 ...
- python闭包函数、装饰器
闭包函数的传值方式: 方式1:通过参数传值 def func(x): print(x)func(1) 方式2:闭包函数传值 def outter(x): def inner(): print(x) r ...
- Python闭包函数
闭包 闭包:python中的闭包从表现形式上定义(解释)为: 如果在一个内部函数里,对在外部作用域(但不是在全局作用域)的变量进行引用,那么内部函数就被认为是闭包(closure). 先看一个函数: ...
- python 闭包函数与装饰器
1.什么是闭包函数 (1):什么是闭包函数: #内部函数包含对外部作用域而非全局作用域的引用, 简而言之, 闭包的特点就是内部函数引用了外部函数中的变量. 在Python中,支持将函数当做对象使用,也 ...
- python闭包函数&装饰器
一.函数引用 函数可以被引用 函数可以被赋值给一个变量 def hogwarts(): print("hogwarts") # hogwarts() # 函数调用 print(ho ...
- python闭包函数及装饰器简介
目录: 闭包函数简介 闭包函数的实际应用 装饰器简介 装饰器初期-完整版 装饰器语法糖 闭包函数简介 1.定义在函数内部的函数(函数的嵌套) 2.内部函数运用外部函数局部名称空间中的变量名 注:函数名 ...
- 跟着太白老师学python day11 函数名的应用 globals(), locals()
1. 函数名就是内存地址 def func(): ') print(func) >>>> <function func at 0x00000000003DC1E0> ...
随机推荐
- JS基础学习第二天
类型转换 类型转换就是指将其他的数据类型,转换为String Number 或 Boolean 转换为String 方式一(强制类型转换): 调用被转换数据的toString()方法例子:var a ...
- vuejs集成echarts的一些问题
最近在做Beetlex的数据分析平台,在开发这个产品过程中涉及到大量的数据图表展示功能:由于产品前端使用的是vuejs开发,所以在集成echarts或多或少会碰到一些问题,在这里主要讲解一下碰到的问题 ...
- Dynamics CRM制作报表的时候让用户可以用自己的权限浏览数据
我们做SSRS报表的时候最头疼的问题就是用Sql查出来的数据都是全部数据没有做权限过滤,导致不同用户看到的数据是一样的. 确实Dynamics CRM产品的数据库时有对这个做处理的,其中每个实体都会有 ...
- OrchardCore 如何动态加载模块?
前言 今天,我们再次讨论下OrchardCore,通过初期调研,我们项目采用OrchardCore底层设施支持模块化,同时根据业务场景,额外还需支持二次开发,于是有了本文,若有不同解决方案,欢迎留言探 ...
- Github Pages+Gridea搭建个人博客
1 概述 Github Pages可以用来托管个人网站,静态的,便于用来实现博客,可以在一个仓库的settings中开启: Gridea是一个静态博客写作客户端,所有文件都在本地,没有数据库,实现简单 ...
- 网络编程Netty入门:EventLoopGroup分析
目录 Netty线程模型 代码示例 NioEventLoopGroup初始化过程 NioEventLoopGroup启动过程 channel的初始化过程 Netty线程模型 Netty实现了React ...
- SpringBoot自动装配,比较全的吧,来看看吧~
文章挺长,表达不好,希望能有获~~~~~~~ Spring也提供使用注解来注册bean,为什么要用SpringBoot呢? 使用Spring应用,比如SpringMVC还行需要配置ViewResolv ...
- aws eks上部署 ingress-nginx 加NLB
转载自https://kubernetes.github.io/ingress-nginx/deploy/#aws In AWS we use a Network load balancer (NLB ...
- K8S(17)二进制的1.15版本部署hpa自动伸缩
K8S(17)二进制部署的K8S(1.15)部署hpa功能 目录 K8S(17)二进制部署的K8S(1.15)部署hpa功能 零.参考文件: 一.生成metrics-proxy证书 二.修改apise ...
- 趁五一撸个纯Flutter版的心情日记App
前言 最近真的是太忙了,只能趁着五一期间把之前Android版的心情日记App移植到Flutter平台,并且已经上架华为应用市场以及苹果App Store. 整体框架沿用了两个星期,用Flutter撸 ...