铁乐学Python_day11_闭包函数
一、【函数名】
1)函数名本质上它也是一种变量,特殊的变量;
(碰到同名其它变量,依照从上往下的代码执行赋值。)
单独打印函数名,输出的是它对应的内存地址:
例:
def funcl():
print(666)
print(funcl)
返回如下:
<function funcl at 0x0000000000DF5488>
2)函数名可以赋值给其它变量名;
def funcl():
print(666)
f1 = funcl
f1()
666
3)函数名可以作为容器类的元素;
def funcl():
print(666)
# print(funcl)
f1 = funcl
f1()
def f2():
print(222)
def f3():
print(333)
def f4():
print(444)
li =[f1, f2, f3, f4]
print(li)
# 返回的是对应的内存地址
for i in li:
i()
# 依次执行函数,可以看到函数名作为列表(容器)的元素也是可以的。
666
222
333
444
4)函数名可以作为参数;
def f1():
print(666)
def f2(a):
a()
print(777)
f2(f1)
输出的结果:
666
777
上例中f1函数名是作为f2函数的参数。
5)函数名可以作为函数的返回值。
例:
def f1():
print(666)
def f2():
return f1
f2()()
# 输出666
def f3():
def f4():
print(777)
return f4
f3()()
#输出 777
以上例子可以看出函数名也可以作为函数的返回值,而且通过return返回的嵌套函数可以调用执行成功。
像上例中的f3()()如果改成f4()是会报错:NameError: name 'f4' is not defined,原因就是全局命名空间中找不到f4()函数定义。所以这是命名空间很神奇的一个地方。
二、【闭包函数】
内层函数对外层函数,非全局变量的引用。
判断闭包函数:函数名.closure(),返回的值中有cell这个元素表示该函数为闭包函数。
闭包函数的机制:当函数开始执行时,如果遇到了闭包,在内存当中会为闭包开辟一个内存空间,用于将闭包中的变量等值放入其中,并不会随着函数的执行完毕而消失。
函数内部定义的函数称为内部函数
主要作用有:
1)缓存,节省内存空间;例如爬虫用,(不断重复爬取同一网页的情况下)
2)装饰器,最能完整体现出闭包的作用。
由于作用域的关系,
我们不能直接拿到函数内部的变量和函数了。如果我们就是想拿怎么办呢?通过返回值!
函数内的变量要想在函数外部用,可以直接返回这个变量,那么如果想在函数外部调用函数内部的函数呢?
是不是直接就把这个函数的名字返回就好了?
这才是闭包函数最常用的用法。所以这也是装饰器之所以是闭包函数最能完整体现用法的原因。
例:闭包函数的运用,引用外层的函数。
def func():
name = 'eva'
def inner():
print(name)
return inner
func() # 直接执行func,并没有调用到func里面的inner函数,不会输出eva
f = func()
f() # 相当于是执行func()(),而func()中返回了inner函数,
# 所以又相当于是在func函数体内部中执行func(inner),所以就能正常执行了。
#输出
eva
例:判断是否为闭包函数
函数名.__closure__()
#输出的__closure__有cell元素 :是闭包函数
def func():
name = 'eva'
def inner():
print(name)
print(inner.__closure__)
return inner
f = func()
f()
print('我是华丽的分割线'.center(30, '-'))
#输出的__closure__为None :不是闭包函数(没有引用外层,试图引用的是全局的变量,所以不为闭包)
name = 'egon'
def func2():
def inner():
print(name)
print(inner.__closure__)
return inner
f2 = func2()
f2()
(<cell at 0x0000000000BF9138: str object at 0x0000000000BE0D18>,)
eva
-----------我是华丽的分割线-----------
None
egon
例:闭包的嵌套(两层,还可以嵌套到三层,一般三层己够满足需要了)
下面是借助不断给函数名赋值成变量来调用嵌套的函数的,这个方法其实在说到装饰器时会用上,为什么要简化成一个变量名()的方式,就是为了装饰其他函数用(不用改变其他函数的表现形式)
def wrapper():
money = 1000
def func():
name = 'eva'
def inner():
print(name, money)
return inner
return func
f = wrapper()
i = f()
i()
eva 1000
例:闭包函数获取网络应用(简单爬虫雏形)
下例为我爬取我的个人wordpress博客首页所用的简单代码,
到最后的内容需要用到decode解码才能看到页面源代码。
还可以更进一步使用文件操作写入到文件。
from urllib.request import urlopen
def index():
url = "https://www.tielemao.com"
def get():
return urlopen(url).read()
return get
tielemao = index()
content = tielemao()
print(content.decode('utf-8'))
写到文件上保存:
from urllib.request import urlopen
def index():
url = "https://www.tielemao.com"
def get():
return urlopen(url).read()
return get
tielemao = index()
content = tielemao()
# print(content.decode('utf-8'))
s = content.decode('utf-8')
with open('tielemao_index.html', encoding='utf-8', mode='a') as f1:
f1.write(s)

end
2018-4-3
铁乐学Python_day11_闭包函数的更多相关文章
- 铁乐学python_Day44_IO多路复用
目录 IO模型介绍 阻塞IO(blocking IO) 非阻塞IO(non-blocking IO) 多路复用IO(IO multiplexing) 异步IO(Asynchronous I/O) IO ...
- 铁乐学python_Day43_协程
铁乐学python_Day43_协程 引子 之前我们学习了线程.进程的概念,了解了在操作系统中进程是资源分配的最小单位,线程是CPU调度的最小单位. 按道理来说我们已经算是把cpu的利用率提高很多了. ...
- 铁乐学python_Day42_线程池
铁乐学python_Day42_线程池 concurrent.futures 异步调用模块 concurrent.futures模块提供了高度封装的异步调用接口 ThreadPoolExecutor: ...
- 铁乐学python_Day42_锁和队列
铁乐学python_Day42_锁和队列 例:多个线程抢占资源的情况 from threading import Thread import time def work(): global n tem ...
- 铁乐学python_Day39_多进程和multiprocess模块2
铁乐学python_Day39_多进程和multiprocess模块2 锁 -- multiprocess.Lock (进程同步) 之前我们千方百计实现了程序的异步,让多个任务可以同时在几个进程中并发 ...
- 铁乐学python_Day38_多进程和multiprocess模块1
铁乐学python_Day38_多进程和multiprocess模块1 [进程] 运行中的程序就是一个进程. 所有的进程都是通过它的父进程来创建的. 因此,运行起来的python程序也是一个进程,那么 ...
- 铁乐学Python_Day34_Socket模块2和黏包现象
铁乐学Python_Day34_Socket模块2和黏包现象 套接字 套接字是计算机网络数据结构,它体现了C/S结构中"通信端点"的概念. 在任何类型的通信开始之前,网络应用程序必 ...
- 铁乐学python_day25_序列化模块
铁乐学python_day25_序列化模块 部份内容摘自博客http://www.cnblogs.com/Eva-J/ 回顾内置方法: __len__ len(obj)的结果依赖于obj.__len_ ...
- 铁乐学python_day24_面向对象进阶1_内置方法
铁乐学python_day24_面向对象进阶1_内置方法 题外话1: 学习方法[wwwh] what where why how 是什么,用在哪里,为什么,怎么用 学习到一个新知识点的时候,多问问上面 ...
随机推荐
- 面试:C/C++常见库函数实现
1. void *mymemcpy(void *dest, const void* src, size_t n): 内存拷贝函数,memcpy函数的功能是从源src所指的内存地址的起始位置开始拷贝n个 ...
- WPF 进程间传递参数
WPF 进程间传递参数 在软件开发中有时需要在一个软件中启动另一个软件,这时用Process.Start(“软件路径”)可以启动另一个软件.如果在这个过程中还需要传递一些参数给新启动 ...
- linux下软、硬链接的创建和删除
linux下软.硬链接的创建和删除 在Linux系统中,内核为每一个新创建的文件分配一个Inode(索引结点),每个文件都有一个惟一的inode号.文件属性保存在索引结点里,在访问文件时,索引结点被复 ...
- Node.js自定义对象事件监听与发射
一.Node.js是以事件驱动的,那我们自定义的一些js对象就需要能监听事件以及发射事件.在Node.js中事件使用一个EventEmitter对象发出,该对象在events模块中.它应该是使用观察者 ...
- 破解栅栏密码python脚本
今天遇到一个要破解的栅栏密码,写了个通用的脚本 #!/usr/bin/env python # -*- coding: gbk -*- # -*- coding: utf_8 -*- # Author ...
- MVC中页面传值方式总结
MVC中的页面传值,通常指Controller和view之间的数据传递,经常用到的有几种方式,总结如下: 一.Controller----------->View(控制器传到视图) 1.View ...
- centos6.5下yum安装mysql5.5
第一步就是看linu是否安装了mysql,经过rpm -qa|grep mysql查看到centos下安装了mysql5.1,那就开始卸载咯 2 接下来就是卸载mysql5.1了,命令:rpm -e ...
- 3.C#基础篇-->堆和栈
一.前言 堆与栈对于理解.NET中的内存管理.垃圾回收.错误和异常.调试与日志有很大的帮助.垃圾回收的机制使程序员从复杂的内存管理中解脱出来,虽然绝大多数的C#程序并不需要程序员手动管理内存,但这并不 ...
- Q:判断链表中是否存在环的相关问题
问题:如何判断一个单向链表中是否存在环? 例如: 链表中存在环(B-->D): <-- <--^ | | v | A-->B-->C-->D 链表中不存在环: A- ...
- 中小型研发团队架构实践五:Redis快速入门及应用
Redis的使用难吗?不难,Redis用好容易吗?不容易.Redis的使用虽然不难,但与业务结合的应用场景特别多.特别紧,用好并不容易.我们希望通过一篇文章及Demo,即可轻松.快速入门并学会应用. ...