Python基础(十三)—切片、yield、生成器、序列化JSON

切片 Slice

切片操作基本表达式:object[start_index:end_index:step]

  • 表达式解释
    • step为步长参数,类似range()里的步长参数。得到的序列从starting_index(包含starting_index)开始,每次以步长前进,即starting_index + step,直到ending_index(不包含ending_index)结束。
    • step:正负数均可,其绝对值大小决定了切取数据时的‘‘步长”,而正负号决定了“切取方向”,正表示“从左往右”取值,负表示“从右往左”取值。当step省略时,默认为1,即从左往右以增量1取值。
    • start_index:表示起始索引(包含该索引本身);该参数省略时,表示从对象“端点”开始取值,至于是从“起点”还是从“终点”开始,则由step参数的正负决定,step为正从“起点”开始,为负从“终点”开始。
    • end_index:表示终止索引(不包含该索引本身);该参数省略时,表示一直取到数据“端点”,至于是到“起点”还是到“终点”,同样由step参数的正负决定,step为正时直到“终点”,为负时直到“起点”。

简单的来说,就是start_index、end_index确定起始两个点及方向(start_index -> end_index),而step则是确定方向和步长。

  • 例子
list = ['a', 'b', 'c', 'd', 'e', 'f', 'g']

print(list[5:3])  # 默认1表示从左往右取值,start_index=5到end_index=3决定了从左往右取值,矛盾,为空

print(list[1:3:-1])  # -1表示从右往左取值,start_index=1到end_index=3决定了从右往左取值,矛盾,为空
print(list[5:3:-1])  # -1表示从右往左取值,start_index=5(f)到end_index=3(d)决定了从左往右取值,['f', 'e']

print(list[-1:3:1])  # 1表示从左往右取值,start_index=-1(g)在end_index=3(d)的右边,因此方向从右往左取值,矛盾,[]
print(list[1:-3:1])  # 1表示从左往右取值,start_index=1(b)在end_index=-3(e)的左边,因此方向从右往左取值,['b', 'c', 'd']

print(list[-2:-5:-1])  # -1表示从右往左取值,start_index=-2(f)在end_index=3(c)的右边,因此方向从右往左取值,['f', 'e', 'd']
print(list[-2:-5:1])  # 1表示从左往右取值,start_index=-2(f)在end_index=3(c)的右边,因此方向从右往左取值,矛盾,[]

yield

首先上一个讲解yield很好的博文:https://blog.csdn.net/mieleizhi0522/article/details/82142856#commentBox

  • 先将yield看成return
    如果你还没有对yield有个初步分认识,那么你先把yield看做“return”,这个是直观的,它首先是个return,普通的return是什么意思,就是在程序中返回某个值,返回之后程序就不再往下运行了,但是使用yield以下代码及结果会是这样的:
def foo():
    print("starting...")
    while True:
        res = yield 4
        print("res:", res)
g = foo()
print(next(g))
print("*"*20)
print(next(g))

"""
starting...
4
********************
res: None
4
"""
  • 以上代码单步调试
    • 1.程序开始执行以后,因为foo函数中有yield关键字,所以foo函数并不会真的执行,而是先得到一个生成器g(相当于一个对象);
    • 2.直到调用next方法,foo函数正式开始执行,先执行foo函数中的print方法,然后进入while循环
    • 3.程序遇到yield关键字,然后把yield想想成return,return了一个4之后,程序停止,并没有执行赋值给res操作,此时next(g)语句执行完成,所以输出的前两行(第一个是while上面的print的结果,第二个是return出的结果)是执行print(next(g))的结果
    • 4.程序执行print("*"20),输出20个
    • 5.又开始执行下面的print(next(g)),这个时候和上面那个差不多,不过不同的是,这个时候是从刚才那个next程序停止的地方开始执行的,也就是要执行res的赋值操作,这时候要注意,这个时候赋值操作的右边是没有值的(因为刚才那个是return出去了,并没有给赋值操作的左边传参数),所以这个时候res赋值是None,所以接着下面的输出就是res:None
    • 6.程序会继续在while里执行,又一次碰到yield,这个时候同样return 出4,然后程序停止,print函数输出的4就是这次return出的4。
  • yield和return的关系和区别
    带yield的函数是一个生成器,而不是一个函数了,这个生成器有一个函数就是next函数,next就相当于“下一步”生成哪个数,这一次的next开始的地方是接着上一次的next停止的地方执行的,所以调用next的时候,生成器并不会从foo函数的开始执行,只是接着上一步停止的地方开始,然后遇到yield后,return出要生成的数,此步就结束。
  • next()和send()
    上面已经说到,next()是接着上一次的next停止的地方执行,而send是发送一个参数给res,使用send的话,开始执行的时候,先接着上一次(return 4之后)执行,先把7赋值给了res,然后执行next的作用,代码如下:
def foo():
    print("starting...")
    while True:
        res = yield 4
        print("res:",res)
g = foo()
print(next(g))
print("*"*20)
print(g.send(7))

"""
starting...
4
********************
res: 7
4
"""
5.程序执行g.send(7),程序会从yield关键字那一行继续向下运行,send会把7这个值赋值给res变量
6.由于send方法中包含next()方法,所以程序会继续向下运行执行print方法,然后再次进入while循环
7.程序执行再次遇到yield关键字,yield会返回后面的值后,程序再次暂停,直到再次调用next方法或send方法。

生成器 generator

  • 生成器的定义
    根据以上yield的例子,其实我们应该清楚了,在python中,一遍循环一边计算的机制,成为生成器:generator
L = [x * x for x in range(10)]
print(L)  # [0, 1, 4, 9, 16, 25, 36, 49, 64, 81]
g = (x * x for x in range(10))
print(g)  # <generator object <genexpr> at 0x02F1D630>
  • 类似于列表生成式的生成器循环
    每次使用next()来获取下一个元素,当无元素是,抛出StopIteration的异常。当然也可以使用循环in,这样就不用了担心StopIteration异常。

  • 函数+yield的生成器
    下面通过斐波那契数列的生成器,进行进一步的讲解。

def fib(max):
    n, a, b = 0, 0, 1
    while n < max:
        yield b
        a, b = b, a+b
        n = n+1
    return 'done'

for i in fib(5):
    print(i)

# 为了获取return 'done',必须捕获StopIteration的异常
g = fib(5)
while True:
    try:
        print(next(g))
    except StopIteration as e:
        print('return %s' % e.value)
        break

序列化pickling、JSON

  • pickling
    pickling前面已经讲过了,pickling前面已经讲过,虽然可以保存数据,但是却只能用于Python,并且不同的版本之间彼此可能都不兼容。只适合保存那些不重要的数据,不能反序列化也没关系。下面主要讲更符合Web标准的JSON序列化。
  • JSON
    JSON不仅是标准格式,并且比XML更快,也可以在Web页面中直接读取,非常方便。
    • python数据类型与json数据类型的映射关系
PythonJSON
dictobject
list, tuplearray
str, unicodestring
int, long, floatnumber
TRUETRUE
FALSEFALSE
Nonenull
  • JSON的方法
方法描述
json.dumps()将 Python 对象编码成 JSON 字符串
json.loads()将已编码的 JSON 字符串解码为 Python 对象
json.dump()将Python内置类型序列化为json对象后写入文件
json.load()读取文件中json形式的字符串元素转化为Python类型
  • python对象转换为json
    以下是将元组、字典、列表等转化为JSON格式的数据。由于json.dumps 进行序列化时,默认使用ascii编码,因此有中文进行序列化时,需要加上ensure_ascii=False。
import json

t = ('lz', 18)
print('t:', json.dumps(t))

di = dict(name='\u7f57\u6b63', age='18')
print('di:', json.dumps(di, ensure_ascii=False))

d = {1:'a', 'b':2, 3: 4}
print('d:', json.dumps(d))

list = [{'a':'A', 2:'B', 'c':3}, ['d', 4], 'e', 5]
print('list:', json.dumps(list))
"""
t: ["lz", 18]
di: {"name": "罗正", "age": "18"}
d: {"1": "a", "b": 2, "3": 4}
list: [{"a": "A", "2": "B", "c": 3}, ["d", 4], "e", 5]
"""
  • json转为python对象
import json
def json_to_dict():
    data = '''
    {
    "name": "Python书籍",
    "origin_price": 66,
    "pub_date": "2018-4-14 17:00:00",
    "store": ["京东", "淘宝"],
    "author": ["张三", "李四", "Jhone"],
    "is_valid": true,
    "is_sale": false,
    "meta": {
        "isbn": "abc-123",
        "pages": 300
    },
    "desc": null
    }
    '''
    res = json.loads(data)
    print(res)
    return res
if __name__ == '__main__':

    res1 = json_to_dict()
    print(type(res1))
"""
{'name': 'Python书籍', 'origin_price': 66, 'pub_date': '2018-4-14 17:00:00', 'store': ['京东', '淘宝'], 'author': ['张三', '李四', 'Jhone'], 'is_valid': True, 'is_sale': False, 'meta': {'isbn': 'abc-123', 'pages': 300}, 'desc': None}
<class 'dict'>
"""

个人博客:Loak 正 - 关注人工智能及互联网的个人博客
文章地址:Python基础(十三)—切片、yield、生成器、序列化JSON

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值