一、【函数名】

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_闭包函数的更多相关文章

  1. 铁乐学python_Day44_IO多路复用

    目录 IO模型介绍 阻塞IO(blocking IO) 非阻塞IO(non-blocking IO) 多路复用IO(IO multiplexing) 异步IO(Asynchronous I/O) IO ...

  2. 铁乐学python_Day43_协程

    铁乐学python_Day43_协程 引子 之前我们学习了线程.进程的概念,了解了在操作系统中进程是资源分配的最小单位,线程是CPU调度的最小单位. 按道理来说我们已经算是把cpu的利用率提高很多了. ...

  3. 铁乐学python_Day42_线程池

    铁乐学python_Day42_线程池 concurrent.futures 异步调用模块 concurrent.futures模块提供了高度封装的异步调用接口 ThreadPoolExecutor: ...

  4. 铁乐学python_Day42_锁和队列

    铁乐学python_Day42_锁和队列 例:多个线程抢占资源的情况 from threading import Thread import time def work(): global n tem ...

  5. 铁乐学python_Day39_多进程和multiprocess模块2

    铁乐学python_Day39_多进程和multiprocess模块2 锁 -- multiprocess.Lock (进程同步) 之前我们千方百计实现了程序的异步,让多个任务可以同时在几个进程中并发 ...

  6. 铁乐学python_Day38_多进程和multiprocess模块1

    铁乐学python_Day38_多进程和multiprocess模块1 [进程] 运行中的程序就是一个进程. 所有的进程都是通过它的父进程来创建的. 因此,运行起来的python程序也是一个进程,那么 ...

  7. 铁乐学Python_Day34_Socket模块2和黏包现象

    铁乐学Python_Day34_Socket模块2和黏包现象 套接字 套接字是计算机网络数据结构,它体现了C/S结构中"通信端点"的概念. 在任何类型的通信开始之前,网络应用程序必 ...

  8. 铁乐学python_day25_序列化模块

    铁乐学python_day25_序列化模块 部份内容摘自博客http://www.cnblogs.com/Eva-J/ 回顾内置方法: __len__ len(obj)的结果依赖于obj.__len_ ...

  9. 铁乐学python_day24_面向对象进阶1_内置方法

    铁乐学python_day24_面向对象进阶1_内置方法 题外话1: 学习方法[wwwh] what where why how 是什么,用在哪里,为什么,怎么用 学习到一个新知识点的时候,多问问上面 ...

随机推荐

  1. 面试:C/C++常见库函数实现

    1. void *mymemcpy(void *dest, const void* src, size_t n): 内存拷贝函数,memcpy函数的功能是从源src所指的内存地址的起始位置开始拷贝n个 ...

  2. WPF 进程间传递参数

    WPF 进程间传递参数          在软件开发中有时需要在一个软件中启动另一个软件,这时用Process.Start(“软件路径”)可以启动另一个软件.如果在这个过程中还需要传递一些参数给新启动 ...

  3. linux下软、硬链接的创建和删除

    linux下软.硬链接的创建和删除 在Linux系统中,内核为每一个新创建的文件分配一个Inode(索引结点),每个文件都有一个惟一的inode号.文件属性保存在索引结点里,在访问文件时,索引结点被复 ...

  4. Node.js自定义对象事件监听与发射

    一.Node.js是以事件驱动的,那我们自定义的一些js对象就需要能监听事件以及发射事件.在Node.js中事件使用一个EventEmitter对象发出,该对象在events模块中.它应该是使用观察者 ...

  5. 破解栅栏密码python脚本

    今天遇到一个要破解的栅栏密码,写了个通用的脚本 #!/usr/bin/env python # -*- coding: gbk -*- # -*- coding: utf_8 -*- # Author ...

  6. MVC中页面传值方式总结

    MVC中的页面传值,通常指Controller和view之间的数据传递,经常用到的有几种方式,总结如下: 一.Controller----------->View(控制器传到视图) 1.View ...

  7. centos6.5下yum安装mysql5.5

    第一步就是看linu是否安装了mysql,经过rpm -qa|grep mysql查看到centos下安装了mysql5.1,那就开始卸载咯 2 接下来就是卸载mysql5.1了,命令:rpm -e ...

  8. 3.C#基础篇-->堆和栈

    一.前言 堆与栈对于理解.NET中的内存管理.垃圾回收.错误和异常.调试与日志有很大的帮助.垃圾回收的机制使程序员从复杂的内存管理中解脱出来,虽然绝大多数的C#程序并不需要程序员手动管理内存,但这并不 ...

  9. Q:判断链表中是否存在环的相关问题

    问题:如何判断一个单向链表中是否存在环? 例如: 链表中存在环(B-->D): <-- <--^ | | v | A-->B-->C-->D 链表中不存在环: A- ...

  10. 中小型研发团队架构实践五:Redis快速入门及应用

    Redis的使用难吗?不难,Redis用好容易吗?不容易.Redis的使用虽然不难,但与业务结合的应用场景特别多.特别紧,用好并不容易.我们希望通过一篇文章及Demo,即可轻松.快速入门并学会应用. ...