大话Python切片,助你彻底掌控切片小妖精

切片语义

  生活中切黄瓜、切萝卜、一本书的每一页等等都是符合切片的语义

  切片的语义是从某个东西中通过某种手段拿到某个整体的一部分

  切片是拿来主义,建立在已经有的序列上,有黄瓜才能切黄瓜

 

 列表 -> python中最典型的序列类型

  形式 [vaule, ...]

nums = [17, 71, 93, 69, 79, 14, 33, 7, 64, 58, 77, 75, 98, 61, 39, 50, 1, 69, 60, 20]
names = ["北门吹雪", "小一", "王五", "李四", "张三", "王二"]

 那如何对其切片呢?

  [start:end:step]

  切片是建立在已有的列表上通过索引手段取得列表的一部分,计算机语言叫读取

   索引是下标访问,下标是一个整数,一个从0开始的递增的数,如 0 1 2 3 4 5 6 7 8 ...

def see_seq(iterable):
    """通过索引遍历序列,遍历序列的语义是依次读取序列中的每一个元素"""
    # 序列开始的下标为 0,也就是索引为 0
    # 序列第一个元素的下标是0,也就是第n个元素的下标为 n-1
    # index 就是下标,从 0 到 序列最后一个元素索引 len(iterable) - 1
    index = 0
    while True:
        # 通过索引读完索引对应的值 -> [i]
        print(iterable[index])

        # 每次读取完,则读取下一个,索引在原来基础上加一
        index += 1

        # 判断边界条件,也就是序列是否读完,读完则推出循环
        # 序列的元素个数为 n = len(iterable), 那么最后一个元素的下标为 n-1
        if len(iterable) - 1 == index:
            break

    # 懂python的都知道for循环,for循环专用于序列的遍历
    # 但for循环建立的基础语义有 迭代器 临时变量 可迭代对象 计数器,复杂难以理解
    # 但for循环语句简洁
    pass


if __name__ == "__main__":
    # 初始列表
    nums = [17, 71, 93, 69, 79, 14, 33, 7, 64, 58, 77, 75, 98, 61, 39, 50, 1, 69, 60, 20]
    names = ["北门吹雪", "小一", "王五", "李四", "张三", "王二"]

    # 调用函数,通过索引方式遍历序列
    see_seq(nums)
    see_seq(names)

  切片是建立在索引之上,通过从开始索引取到结束索引但不包括结束索引

  

[start:end:step]

  切片最重要的计算公式  -end < (start + 0... * step)  < end

  1. start 开始索引,省略则为0, start = 0

  2. end 结束索引,省略则为列表长度(列表中元素的个数), end = len(seqence)

  3. step 为步长,索引之间的间隔,等差数列

def get_slice(_list, start=None, end=None, step=None,):
    """模拟实现切片语义"""
    # 省略开始值则为 0
    if start is None:
        start = 0
    # 省略结束值则为其长度 len(_list)
    if end is None:
        end = len(_list)
    # 省略步长,则步长为 1,阻止小朋友传入0步长
    if step is None or step == 0:
        step = 1

    # 切片实现总要公式-end < (start + 0... * step)  < end
    # 保存结果列表
    result = []
    # 核心计数count,始终是从0开始自增1
    count = 0
    while True:
        # 核心计算公式
        list_index = start + count * step
        # 通过核心公式计算出索引对列表取值
        elem = _list[list_index]

        # 处理步长为正数情况,list_index 的值为 正数
        if step > 0 and not (list_index < end):
            break

        # 处理步长为负数情况, list_index 的值为 负数
        if step < 0 and not(list_index > end):
            break

        result.append(elem)
        # 计数器自增1
        count += 1

    # 返回结果
    return result


if __name__ == "__main__":
    # 初始列表
    names = ["北门吹雪", "小一", "王五", "李四", "张三", "王二", "小明", "小爱", "小溪"]

    # python内置切片
    slice_one = names[-1:-5:-1]
    # 自定义切片语义的实现
    slice_two = get_slice(names, -1, -5, -1)

    # 对比发现一致
    print(slice_one)
    print(slice_two)

 亲,切片不难

附带for循环遍历序列的方式

 for 循环的原理是

  1. 把序列变成一个迭代器,然后保存在一个内部临时变量中 temp_var = iter(iterable)
  2. 通过内置函数next()调用迭代器中的 __next__方法获取下一个数据 item = next(temp_var)
  3. 到了边界,迭代器会触发一个迭代器耗尽异常,for循环会自动捕获,终止for循环
def see_seq(iterable):
    """通过for循环遍历序列,遍历序列的语义是依次读取序列中的每一个元素"""
    # 序列开始的下标为 0,也就是索引为 0
    # 序列第一个元素的下标是0,也就是第n个元素的下标为 len(iterable) - 1

    # for 循环自动处理索引、索引边界
    # 自动依次取出 0 到 len(iterable)-1 索引对应的值
    # 是不是很简洁?
    for item in iterable:
        print(item)

    # 打印分隔符
    print("等价for语句".center(20, "-"))
    # for 循环的原理是
    # 1. 把序列变成一个迭代器,然后保存在一个内部临时变量中 temp_var = iter(iterable)
    # 2. 通过内置函数next()调用迭代器中的 __next__方法获取下一个数据  item = next(temp_var)
    # 3. 到了边界,迭代器会触发一个迭代器耗尽异常,for循环会自动捕获,终止for循环
# 等价语句 temp_var = iter(iterable) # 把序列变成一个迭代器 while True: try: item = next(temp_var) # next()调用迭代器 except StopIteration: # 迭代器耗尽异常,推出循环 break # 以上是for循环自动完成,这一行才是循环体 print(item) if __name__ == "__main__": # 初始列表 nums = [17, 71, 93, 69, 79, 14, 33, 7, 64, 58, 77, 75, 98, 61, 39, 50, 1, 69, 60, 20] names = ["北门吹雪", "小一", "王五", "李四", "张三", "王二"] # 调用函数,通过索引方式遍历序列 see_seq(nums) see_seq(names)

  



原文地址:https://www.cnblogs.com/2bjiujiu/p/13753287.html