生成器对象

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

先来看一段代码:

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. 一次关于关系抽取(RE)综述调研的交流心得

    本文来自于一次交流的的记录,{}内的为个人体会. 基本概念 实事知识:实体-关系-实体的三元组.比如, 知识图谱:大量实时知识组织在一起,可以构建成知识图谱. 关系抽取:由于文本中蕴含大量事实知识,需 ...

  2. Head标签里面的dns-prefetch,preconnect,prefetch和prerender

    开始 今天突然心血来潮想起前端性能优化的问题,这基本是老生常谈的事情了,面试随便都能说上几个,但是还是有点疑问:就是Head标签了,记忆中Head可是藏龙卧虎,各种技能都有,当然这些不可能都一一记住, ...

  3. jquery+html5+canvas实现图片 预览 压缩 上传

    javascirpt (function($){ $.fn.extend({ aiiUpload:function(obj) { if(typeof obj !="object") ...

  4. 小程序入门系列之 tabBar

    本系列为简单入门系列,以一定概括性思路来叙述内容,具体可以查看官网 大部分的电商应用都是底部或顶部多 tab 的模式. 下面我们从配置角度来分析一下: 第一个:position 配置如下: 默认是 b ...

  5. HTML5 Audio & Video 属性解析

    一.HTML 音频/视频 方法 play() play() 方法开始播放当前的音频或视频. var myVideo=document.getElementById("video1" ...

  6. ES6-11学习笔记--类与继承

    ES5 中的类与继承: 类的定义: function People(name, age) { // this指向当前实例化对象 console.log(this); // 实例属性 this.name ...

  7. TINY语言采用递归下降分析法编写语法分析程序

    目录 自顶向下分析方法 TINY文法 消左提左.构造first follow 基本思想 python构造源码 运行结果 参考来源:聊聊编译原理(二) - 语法分析 自顶向下分析方法 自顶向下分析方法: ...

  8. 查看k8s的yaml里面反馈的错误,不仅仅是通过日志来查看错误

    使用:kubectl describe pod......例如: kubectl describe pod api-55f5d8d49d-kzmcj ///////////////////////// ...

  9. Java-GUI编程之菜单组件

    前面讲解了如果构建GUI界面,其实就是把一些GUI的组件,按照一定的布局放入到容器中展示就可以了.在实际开发中,除了主界面,还有一类比较重要的内容就是菜单相关组件,可以通过菜单相关组件很方便的使用特定 ...

  10. Java学习day12

    Set集合特点:不能包含重复的元素:没有所有的方法,不能通过普通的for循环遍历集合 哈希值是JDK根据对象的地址或者字符串或者数字算出来的int类型的数值,同一个对象多次调用hashCode()方法 ...