生成器对象

生成器对象其实本质还是迭代器,只不过这个迭代器的内容可以由我们直接来定义了,所以它也可以称为自定义迭代器。

先来看一段代码:

def index():
print('abc')
yield
print(index()) # 输出:<generator object index at 0x000001DEEAF00200>

可以看到,在加了关键字yield之后,函数变成了生成器对象了。如果想要执行函数里的代码,就需要使用__next__()方法。

def index():
print('abc')
yield
res = index()
res.__next__() # 输出:abc

多个yield

def index():
print('abc')
yield
print('def')
yield
print('gh')
yield
res = index()
res.__next__() # 输出:abc
res.__next__() # 输出:def
res.__next__() # 输出:gh

在yield后也可以跟上一个值

def index():
print('abc')
yield 123
print('def')
yield 456
res = index()
r1 = res.__next__() # 输出:abc
print(r) # 输出:123
r2 = res.__next__() # 输出:def
print(r) # 输出:456

结论:

  1. 在函数内部添加了yield关键字之后,函数就会变成生成器对象,并且可以调用__next__方法。
  2. 当有多个yield关键字时,每执行一次__next__方法时函数代码会运行到下个yield处并停留。
  3. 如果yield后跟上了一个值,则执行__next__方法时会返回yield后的值。

补充:yield关键字的作用

除了以上的作用外,yield还可以接收外界的传值

def eat(name):
print(f'{name}准备干饭')
while True:
food = yield
print(f'{name}正在吃{food}')
res = eat('jason')
# 使用send方法前需要调用一次__next__启动
res.__next__() # 输出:jason准备干饭
# 使用send方法给yield传值
res.send('kfc') # 输出:jason正在吃kfc
res.send('米饭') # 输出:jason正在吃米饭

自定义range方法

在了解完生成器对象后,这时候我们就可以自己定义一个和range一样功能的方法了。

先实现两个参数的情况:

def my_range(start, end):
"""
start: 起步位置
end: 结束位置
"""
# 如果start一直累加后值等于end则退出循环
while start < end:
yield start
# 每被执行一次__next__方法起始位置加1
start += 1
for i in my_range(1, 4):
print(i)

实现了两个参数后,我们可以开始解决一个参数的情况

def my_range(start, end=None):
"""
start: 起步位置
end: 结束位置,默认为空值
"""
# 如果没有给end传值,说明只有一个参数,取值范围为[0,start)
if not end:
end = start
start = 0
# 如果start一直累加后值等于end则退出循环
while start < end:
yield start
# 每被执行一次__next__方法起始位置加1
start += 1
for i in my_range(1, 4):
print(i)

最后解决三个参数的情况

def my_range(start, end=None, step=1):
"""
start: 起步位置
end: 结束位置,默认为空值
step:步长,默认为1
"""
# 如果没有给end传值,说明只有一个参数,取值范围为[0,start)
if not end:
end = start
start = 0
# 如果start一直累加后值等于end则退出循环
while start < end:
yield start
# 每被执行一次__next__方法起始位置加步长
start += step
for i in my_range(1, 4):
print(i)

生成器表达式

直入主题,生成器结构

res = (i for i in 'jason')
print(res) # 输出:<generator object <genexpr> at 0x000002AC49CF0200>
"""可以看到,它并不会像列表生成式一样直接输出"""
"""因为生成器内部的代码只有在调用__next__迭代取值的时候才会执行"""
print(res.__next__()) # 输出:j
print(res.__next__()) # 输出:a

模块

简介

模块就是一系列功能的结合体,只需要导入就可以直接使用,它极大的提升了开发效率。比如我自己要实现一个功能可能要写好几行代码,但是如果模块里可以实现这个功能,那我可以直接拿来使用。

模块的三种来源方式:

  1. 内置的模块

    这些是python自带的模块,直接用代码导入就可以使用。

  2. 自定义模块

    这些是自己写的代码封装成模块,可以自己使用或发布到网上

  3. 第三方模块

    这些是由别人发布到网上的,我们可以下载过来使用

模块的四种表现形式:

  1. 使用python代码编写的py文件
  2. 多个py文件组成的文件夹,也可以称为包
  3. 已被编译为共享库或DLL的c或C++扩展(了解)
  4. 使用C编写并链接到python解释器的内置模块(了解)

模块的导入方式

第一种:import ...

语法结构:

import 模块1, 模块2, ...

这种导入方式可以使用模块内部的变量和方法,一个模块只会被导入一次,不管你执行了多少次import。

案例一:

创建一个md.py文件

name = '来自md.py'
print(name)

创建一个main.py文件

import md

执行main.py文件,此时会输出

来自md.py

说明在导模块时,会执行被导入模块的内部代码。

案例二:

创建一个md.py文件

name = '来自md.py的name变量'
def index():
print('来自md.py的index函数')

创建一个main.py文件

import md
name = '来自main.py的name变量'
print(name)
print(md.name)
def index():
print('来自main.py的index函数')
index()
md.index()

执行main.py文件,此时会输出

来自main.py的name变量
来自md.py的name变量
来自main.py的index函数
来自md.py的index函数

说明导入模块后,想要模块内部变量和函数需要使用模块加点的方式,直接使用只会从当前文件寻找。

图解:

第二种:from ... import ...

语法结构:

from 模块名 import 名称1,名称2,...

这种方式使用模块内部的名称可以不需要用模块加点的方式了,可以直接使用。

案例一:

创建一个md.py文件

name = '来自md.py'
print(name)

创建一个main.py文件

from md import name

执行main.py文件,此时会输出

来自md.py

说明在导模块时,会执行被导入模块的内部代码。

案例二:

创建一个md.py文件

name = '来自md.py'
def index():
print('来自md.py的index函数')

创建一个main.py文件

from md import name, index
print(name)
index()

执行main.py文件,此时会输出

来自md.py
来自md.py的index函数

说明使用from + import方法导入模块后,可以直接使用模块内部的名称,所以这时候就需要避免出现重名的情况。

图解

补充

  1. 导入模块时可以给模块起别名
import time as t
t.sleep(3) # 等价于time.sleep(3) from md import name as n, index as x
print(n) # 等价于print(name)
x() # 等价于index()
  1. 导入全部名称
from md import *
"""如果模块文件中使用了__all__限制可以使用的名字,那么*号就会失效,只能使用__all__后面列举的名字"""
# 在md.py文件中
__all__ = ['name']
# 这时如果使用from md import *只能使用name变量
  1. 建议

导入个模块时,可以把有相似功能的模块使用同一个import导入,如果没有建议分开导入。

python之生成器与模块的更多相关文章

  1. Python迭代器生成器,模块和包

      1.迭代器和生成器 2.模块和包 1.迭代器 迭代器对象要求支持迭代器协议的对象,在Python中,支持迭代器协议就是实现对象的__iter__()和__next__()方法.    其中__it ...

  2. Python推导表达式、迭代器、生成器、模块和包

    推导表达式 yield用法 模块的概念和导入方法 包和包管理 推导表达式(利用for,一个一个地放入数据) 列表推导 集合推导 字典推导 迭代器 迭代 for 迭代变量 in 可迭代对象 每一次循环都 ...

  3. python学习之random模块

    Python中的random模块用于生成随机数.下面介绍一下random模块中最常用的几个函数. random.random random.random()用于生成一个0到1的随机符点数: 0 < ...

  4. python笔记之itertools模块

    python笔记之itertools模块 itertools模块包含创建有效迭代器的函数,可以用各种方式对数据进行循环操作,此模块中的所有函数返回的迭代器都可以与for循环语句以及其他包含迭代器(如生 ...

  5. 【转】Python 3的pathlib模块:驯服文件系统

    [转]Python 3的pathlib模块:驯服文件系统 https://python.freelycode.com/contribution/detail/1248 Python部落(python. ...

  6. python中“生成器”、“迭代器”、“闭包”、“装饰器”的深入理解

    一.生成器 1.什么是生成器? 在python中,一边循环一边计算的机制,称为生成器:generator. 2.生成器有什么优点? 1.节约内存.python在使用生成器时对延迟操作提供了支持.所谓延 ...

  7. Python高级编程-itertoos模块

    Python的内建模块itertools提供了非常有用的用于操作迭代对象的函数. 首先我们看看itertools模块提供的几个“无限”迭代器, import itertools naturals = ...

  8. python中“生成器”、“迭代器”、“闭包”、“装饰器”的深入理解

    python中"生成器"."迭代器"."闭包"."装饰器"的深入理解 一.生成器 1.生成器定义:在python中,一边 ...

  9. python基础,函数,面向对象,模块练习

    ---恢复内容开始--- python基础,函数,面向对象,模块练习 1,简述python中基本数据类型中表示False的数据有哪些? #  [] {} () None 0 2,位和字节的关系? # ...

随机推荐

  1. DOM 小总结

    DOM 是什么 文档对象模型,是针对 HTML 和 XML 文档的一个 API (应用程序编程接口), 描绘了一个层次化的节点树. D: document 当 web 浏览器浏览一个页面的时候,DOM ...

  2. Kurento安装与入门12——Show DataChannel

    Show DataChannel 本示例允许用户在文本框中输入任意文字,输入的文字将以字幕的形式显示在传回的视频中. 官网文档 [Show DataChannel[1] 首先从github上获取代码( ...

  3. Python窗口学习之给按钮美化

    tkinter的按钮很丑也很难改 怎么办呢? 最好的方法就是不用按钮! 给Label添加点击事件,和按钮的作用是一样的! 代码: #!/usr/bin/env python # -*- coding: ...

  4. Centos6.9 安装zabbix3.4 过程

    Centos6.9 安装zabbix3.4 过程 1.安装apache httpd 一开始忘记截图(略...) # yun install httpd 完成后,启动httpd服务 # service ...

  5. css3,css的基础全局运用

    浮动 1.浮动起来的盒子不占用位置,浮动了一个盒子下面的标准流的盒子会顶上来 ​ 可用清除浮动的方法来解决标准流会顶替位置的问题 清除浮动给父盒子加overflow: hidden; 鼠标经过事件 : ...

  6. 《头号玩家》AI电影调研报告(四)

    五. 现实中对于技术的实现 1.自由变身 电影中,主人公借助"魔幻眼镜"让自己快速变成另一个人的模样,现实中我们一样也能做到! 在今年游戏开发者大会(GDC)上,一个名为" ...

  7. 第一阶段:Java基础之数组

    注意点: @Java语言是把数组当作一个"对象"来看待的 @把数组分为两部分看,一部分是数组的引用,放置在栈内存中,一部分是数组对象,放置在堆内存中 @数组的引用可以指向任何有效的 ...

  8. spring-bean依赖注入-02(通过p命名空间注入)

    上一篇博客讲述了为什么使用spring依赖注入,怎么注入,详见 spring-bean依赖注入-01(等你来点击) 废话不多说,开始使用p命名空间进行set注入 使用另外一种注入方式是这样的(具体实现 ...

  9. 使用SQL的FOR XML PATH('')将字段用逗号隔开

    FOR XML PATH('') 将查询结果显示为XML 经常用来将查询结果按逗号分隔后显示至某一字段 select * from Area结果 添加FOR XML PATH('')后 select ...

  10. Oracle双字段约束

    Oracle里有unique约束,意思是该字段唯一. 但如果是两个字段呢? 比如说一个会员等级表 ID NAME POINT DISCOUNT PRIVILEGE MID 1019 普通会员 0 10 ...