1. 函数名的使用

其实函数名也是一个变量,但它是一个比较特殊的变量,与小括号配合可以执行函数的变量:

  • 函数名其实和内存一样,也可以使用print查看它的内存地址:
In[22]: def func1():
...: pass
...:
In[23]: print(func1)
<function func1 at 0x000002A24830C488>
  • 函数名赋值给其他变量
In[24]: def func2():
...: print('呵呵')
...:
In[25]: a = func2
In[26]: a()
呵呵
  • 函数也能当作容器类的元素:
In[27]: print(func2)
<function func2 at 0x000002A24830B048>
In[28]: lis = [func2, func2, func2]
In[29]: print(lis)
[<function func2 at 0x000002A24830B048>, <function func2 at 0x000002A24830B048>, <function func2 at 0x000002A24830B048>]
  • 函数名也能当作函数的参数:
In[30]: def func3():
...: print('i\'m func3')
...:
In[31]: def func4(fn):
...: fn()
...:
In[32]: func4(func3) # 把函数名func3作为参数传递给func4
i'm func3
  • 函数名也可以作为函数的返回值:
In[33]: def func5():
...: def func6():
...: print('this is sub function')
...: return func6 # 这里直接把函数名func6作为返回值返回给调用者
...:
In[34]: fn = func5() # 这里的fn就是func6了
In[35]: fn() # 加()执行函数
this is sub function

2. 闭包

闭包是指在内层函数中对外层函数(非全局)的引用

In[36]: def func6():
...: x = 24
...: def func7():
...: print(x) # 闭包
...: func7()
...: print(func7.__closure__) # 使⽤__closure__来检测函数是否是闭包.
...: # 使⽤函数名.__closure__返回cell就是闭包. 返回None就不是闭包 In[37]: func6()
24
(<cell at 0x000002A2482F8EE8: int object at 0x000000005BA86F00>,)

那么我们要怎么能在函数外面调用内部函数呢,其实很简单,把内部函数作为返回值返回给调用者即可:

In[38]: def func8():
...: x = 24
...: def func9():
...: print(x)
...: print(func9.__closure__)
...: return func9
...:
In[39]: fn = func8()
(<cell at 0x000002A2482F8DF8: int object at 0x000000005BA86F00>,)
In[40]: fn() # 这样就可以在函数外面使用了
24

那么闭包有什么用呢,我们再来看一个例子:

In[2]: def func1():
...: x = 23
...: def func2():
...: nonlocal x
...: x += 1
...: return x
...: return func2
...:
In[3]: fn = func1()
In[5]: fn()
Out[5]: 24
In[6]: fn()
Out[6]: 25
In[7]: fn()
Out[7]: 26
In[8]: fn()
Out[8]: 27
In[9]: x
Traceback (most recent call last):
File "D:\Environment\python-virtualenv\jupyter\lib\site-packages\IPython\core\interactiveshell.py", line 3265, in run_code
exec(code_obj, self.user_global_ns, self.user_ns)
File "<ipython-input-9-6fcf9dfbd479>", line 1, in <module>
x
NameError: name 'x' is not defined

从上面我们可以看出,x作为一个局部命名空间的变量,在使用是看起来更像是使用全局变量一样,但是最后的报错是证明了x并不是一个全局变量。这个现象就是闭包造成的,它可以把函数中的变量在外部使用,并且能让它常驻于内存。

3. 迭代器

我们之前使用for循环变量一个容器类的对象是,都有提要遍历的对象一定是要可迭代的,先看下可迭代对象里面都有什么:

In[15]: dir(list)
Out[15]:
['__add__',
'__class__',
...
...
'__imul__',
'__init__',
'__init_subclass__',
'__iter__', # 列表这里有个__iter__方法,代表这个是一个可迭代的对象
'__le__',
'__len__',
'__lt__',
'__mul__',
...
...
In[16]: dir(str)
Out[16]:
['__add__',
'__class__',
...
...
'__init__',
'__init_subclass__',
'__iter__', # 字符串也是有__iter__方法
'__le__',
'__len__',
...
...

如果自己尝试过的话会发现列表、字典、字符串和集合都会有这个方法,因为他们都是可迭代对象。

这是查看⼀个对象是否是可迭代对象的第⼀种办法. 我们还可以通过isinstence()函数来查看⼀个对象是什么类型的

l = [1,2,3]
l_iter = l.__iter__()
from collections import Iterable
from collections import Iterator
print(isinstance(l,Iterable)) #True
print(isinstance(l,Iterator)) #False
print(isinstance(l_iter,Iterator)) #True
print(isinstance(l_iter,Iterable)) #True

综上. 我们可以确定. 如果对象中有__iter__函数. 那么我们认为这个对象遵守了可迭代协议.就可以获取到相应的迭代器. 这⾥的__iter__是帮助我们获取到对象的迭代器. 我们使⽤迭代器中的__next__()来获取到⼀个迭代器中的元素. 那么我们之前讲的for的⼯作原理到底是什么? 继续看代码

In[17]: s = 'zzc'
In[18]: s_iter = s.__iter__() # 使用字符串的__iter__()方法
In[19]: s_iter.__next__()
Out[19]: 'z'
In[20]: s_iter.__next__()
Out[20]: 'z'
In[21]: s_iter.__next__()
Out[21]: 'c'
In[22]: s_iter.__next__()
Traceback (most recent call last):
File "D:\Environment\python-virtualenv\jupyter\lib\site-packages\IPython\core\interactiveshell.py", line 3265, in run_code
exec(code_obj, self.user_global_ns, self.user_ns)
File "<ipython-input-22-b111e2554a10>", line 1, in <module>
s_iter.__next__()
StopIteration

从上可以看出,只要一个对象有__iter__方法,那么我们认为这个对象遵守了可迭代协议,就可以获取到相应的迭代器(s_iter),然后后我们可以使用迭代器中的__netx__方法来获取下一个迭代器中的元素,直到抛出`StopIteration``异常时退出,

for循环的机制:

In[24]: l1 = ['zzc', '牛奶', 'PDD', '55开']
In[25]: for i in l1:
...: print(i)
...:
zzc
牛奶
PDD
55开

用while实现的for循环:

lis = ['zzc', '牛奶', 'PDD', '55开']
iter = lis.__iter__()
while 1:
try: # try/excpet是捕获异常的语句
ele = iter.__next__()
print(ele)
except StopIteration: # 当捕获到StopIteration异常时退出
break

总结:

  • Iterable: 可迭代对象. 内部包含__iter__()函数
  • Iterator: 迭代器. 内部包含__iter__() 同时包含__next__().
  • 迭代器的特点:
    1. 节省内存.
    2. 惰性机制
    3. 不能反复, 只能向下执⾏.

python学习笔记:第11天 闭包及迭代器的更多相关文章

  1. Deep learning with Python 学习笔记(11)

    总结 机器学习(machine learning)是人工智能的一个特殊子领域,其目标是仅靠观察训练数据来自动开发程序[即模型(model)].将数据转换为程序的这个过程叫作学习(learning) 深 ...

  2. python学习笔记之11:图像用户界面

    这里会介绍如何创建python程序的图像用户界面(GUI),也就是那些带有按钮和文本框的窗口等.目前支持python的所谓“GUI工具包”的有很多,本文简要介绍最成熟的跨平台pythonGUI工具包- ...

  3. python学习笔记(11)--文件与数据格式化

    文件的概念: 文件是数据的抽象和集合,是存储在辅助存储器上的数据序列,文件是数据存储的一种形式,文件的展现形态,文本文件和二进制文件. 文本文件输出: f.txt文件保存:“我是中国人” >&g ...

  4. python学习笔记(11)--爬虫下载漫画图片

    说明: 1. 某本子网站爬虫,现在只实现了扒取一页,已经凌晨两点了,又饿又困,先睡觉,明天再写总结吧! 2. 我是明天,我来写总结了! 3. 这个网站的结构是这样的: 主页: 主页-第1页-漫画1封面 ...

  5. python学习笔记(11)文件操作

    一.读文件 读写文件是最常见的IO操作.Python内置了读写文件的函数,用法和C是兼容的. 读写文件前,我们先必须了解一下,在磁盘上读写文件的功能都是由操作系统提供的,现代操作系统不允许普通的程序直 ...

  6. python学习笔记(11)--测验3: Python基础语法(下) (第7周)

    斐波那契数列计算 B 描述 斐波那契数列如下: F(0) = 0, F(1) = 1 F(n) = F(n-1) + F(n-2) 编写一个计算斐波那契数列的函数,采用递归方式,输出不超过n的所有斐波 ...

  7. python学习笔记(11)--词云

    中分词库  jieba 词云 wordcloud import jieba import wordcloud f = open("新时代中国特色社会主义.txt", "r ...

  8. python学习笔记(11)--数据组织的维度

    数据的操作周期 存储  -- 表示 -- 操作 一维数据表示 如果数据有序,可以使用列表[]:如果数据没有顺序,可以使用集合{} 一维数组存储 存储方式一:空格分隔 ,使用一个或多个空格分隔进行分隔, ...

  9. python学习笔记:(十五)迭代器和生成器

    一.迭代器: 1.迭代器是python最强大的功能之一,是访问集合元素的一种方式. 2.迭代器是一个可以记住遍历的位置的对象. 3.迭代器对象从集合的第一个元素开始访问,直到所有的元素被访问结束.迭代 ...

  10. Python实用笔记 (11)高级特性——迭代器

    这些可以直接作用于for循环的对象统称为可迭代对象:Iterable. 可以使用isinstance()判断一个对象是否是Iterable对象: >>> from collectio ...

随机推荐

  1. SQL点点滴滴_删除临时表

    select into 创建的表属于临时表,判断是否存在的方法 select c_adno,c_con_no into #temp from tb_contract IF OBJECT_ID( 'te ...

  2. UVaOJ 694 - The Collatz Sequence

    题目很简单,但是一开始却得到了Time Limit的结果,让人感到很诧异.仔细阅读发现,题目中有一个说明: Neither of these, A or L, is larger than 2,147 ...

  3. 证书制作 z

    一.WCF中的安全方式 说到安全就会涉及到认证,消息一致性和机密性,WCF的安全方式分为两种,即传输安全和消息安全. 传输安全和消息安全的区别:传输安全提供点对点的安全: 比如 A 提供服务,B和C直 ...

  4. [翻译]Elasticsearch重要文章之二:堆内存的大小和swapping

    Elasticsearch默认安装后设置的内存是1GB,对于任何一个业务部署来说,这个都太小了.如果你正在使用这些默认堆内存配置,你的集群配置可能有点问题. 这里有两种方式修改Elasticsearc ...

  5. linux系统开机突然黑屏,只有光标

    以前系统都是好好的,今天开机黑屏了,只有一个光标可以看见其他东西都看不见了. 经过检查发现是磁盘满了,将不用的文件删除之后可以正常开机了.特此记录一下!!! 有人问黑屏了怎么删文件? 我采用的方法是W ...

  6. windows时间同步脚本

    #!/usr/bin/env python# -*- coding:UTF-8 -*-# 脚本用于windows时间同步,设置window计划任务每五分钟执行一次 import timeimport ...

  7. zabbix 添加 ROS 软路由监控 WinBox设置

    如图设置

  8. 【深入理解JAVA虚拟机】第三部分.虚拟机执行子系统.3.函数调用与执行

    这章原名叫“虚拟机字节码执行引擎”,实际就是讲的函数如何调用和执行的. 1.概述 “虚拟机”是一个相对于“物理机”的概念,这两种机器都有代码执行能力, 其区别是物理机的执行引擎是直接建立在处理器. 硬 ...

  9. 【pbrt】在c++程序中使用pbrt进行渲染

    近段时间做一个关于水面的动画.由于我用c++实现水面动画的,然而使用c++我自己的渲染系统渲染结果被同学说是可视化不叫渲染,所以我决定修改一下…… 恰好进来在学习pbrt,所以索性就蛋疼了考虑直接用p ...

  10. No module named _sqlite3

    [root@lgj01 opsadmin]# python manage.py startapp accountTraceback (most recent call last):  File &qu ...