zoukankan      html  css  js  c++  java
  • Python高级特性

    参考原文

      廖雪峰Python高级特性讲解

    开篇:高级特性是用来简化我们常用操作的特性,合理利用高级特性可以使代码更简洁、明了。

    切片

      取list或tuple中的指定索引范围的操作,用循环十分繁琐,因此Python提供了切片(Slice)操作符,能大大简化这种操作。示例:

    >>> L = ['a', 'b', 'c', 'd']
    >>> L[0:3]
    ['a', 'b', 'c']

      L[0:3]表示从索引0开始取,直到索引3为止,但不包括索引3,即索引0,1,2正好3个元素。如果第一个索引是0,还可以省略:

    >>> L[:3]
    ['a', 'b', 'c']

      还可以倒着取,记住倒数第一个数的索引是-1:

    >>> L[-1:]
    ['d']
    >>> L[-2:-1]
    ['c']
    >>> L = list(range(100))
    >>> L[:10:2]
    [0, 2, 4, 6, 8]
    >>> L[:]
    [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, 64, 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 76, 77, 78, 79, 80, 81, 82, 83, 84, 85, 86, 87, 88, 89, 90, 91, 92, 93, 94, 95, 96, 97, 98, 99]
    >>> 

       tuple也是一种list,区别是不可变,因此tuple也可以切片,结果依然是tuple:

    >>> (0, 1, 2, 3, 4)[:3]
    (0, 1, 2)

      字符串也可以看成一种list,每个元素就是一个字符,在很多编程语言中针对字符串提供了很多的截取函数(如substring),其实目的就是对字符串切片。在Python中没有针对字符串的截取函数,只需切片:

    >>> 'ABCDEFGH'[::3]
    'ADG

    迭代

      对于一个list或tuple,我们可以通过for循环来遍历这个list或tuple,这种遍历我们称为迭代(Iteration)。在Python中,迭代是通过for ... in来完成的,而很多语言如C语言,迭代list是通过下标完成的:

    for (i=0;i<list.length;i++){
        n = list[i];
    }

      可以看出,Python的for循环抽象程度要高于C的for循环,因为Python的for循环不仅可以用在list或tuple上,还可以作用在其他可迭代的对象上,如迭代dict:

    >>> d = {'a': 1, 'b': 2, 'c': 3}
    >>> for key in d:
        print(key)
    
        
    a
    b
    c

    注:默认情况下,dict迭代的是key,如果要迭代value,可以用for value in d.values(),如果要同时迭代key和value,可以用for k,v in d.items()

       由于字符串也是可迭代对象,因此,也可以用作for循环:

    >>> for ch in 'ABC':
        print(ch)
    
        
    A
    B
    C

      所以怎么判断一个对象是可迭代对象呢?方法是通过collections模块的Iterable类型判断:

    from collections import Iterable
    print(isinstance('abc',Iterable)) # str是否可迭代 True
    print(isinstance([1,2,3],Iterable)) # lis是否可迭代 True
    print(isinstance(123,Iterable))  #整数是否可以迭代 False

      如果要对list实现类似Java那样的下标怎么办?Python内置的enumerate函数可以把list变成索引-元素对:

    >>> for i, value in enumerate(['a', 'b', 'c']):
        print(i, value)
    
        
    0 a
    1 b
    2 c

      在for循环里还可以引用多个变量,这在Python里是很常见的:

    >>> for x, y, z in[(1,2,3),(3,4,5)]:
        print(x, y, z)
    
        
    1 2 3
    3 4 5
    Tips:任何可迭代对象都可以作用于for循环,包括我们自定义的数据类型,只要符合迭代条件,就可以使用for循环

    列表生成式

      列表生成式(List Comprehensions)是Python内置的非常简单却强大的可以用来创建list的生成式。

    问题:怎么生成[1x1, 2x2, 3x3, ..., 10x10]?

    你可以使用循环:

    >>> L = []
    >>> for x in range(1, 11):
        L.append(x * x)
    
        
    >>> L
    [1, 4, 9, 16, 25, 36, 49, 64, 81, 100]

    但是使用循环难免显得太繁琐,而列表生成式可以用一行语句代替循环生成上面的list:

    >>> [x * x for x in range(1, 11)]
    [1, 4, 9, 16, 25, 36, 49, 64, 81, 100]

    还可以在for循环后面加上if判断用于筛选:

    >>> [x * x for x in range(1,11) if x % 2 == 0]
    [4, 16, 36, 64, 100]

    由于for循环可以使用多个变量,所以列表生成式也可以用多个变量来生成list:

    >>> d = {'x': 'A', 'y': 'B', 'z': 'C'}
    >>> [k + '=' + v for k, v in d.items()]
    ['x=A', 'y=B', 'z=C']
    Tips:运用列表生成式,可以快速生成list,也可以通过一个list推导出另一个list,而代码却十分简洁。

    生成器

      上面介绍了列表生成式,通过它我们可以直接创建一个列表,但是如果我们创建了一个包含100万个元素的列表,我们却仅仅需要访问前面几个元素,这不就使得后面元素占用的内存空间白白浪费了吗?

      所以,如果列表元素可以按照某种算法推算出来,在循环的过程中不断推算出后续的元素,这样就不必在一开始就创建完整的list,从而节省大量的空间。在Python中,这种一边循环,一边计算的机制,称为生成器generator。那么怎么创建一个generator呢?

      1.可以把列表生成式的[]改成(),就创建出了一个generator:

    >>> g = (x * x for x in range(10))
    >>> g
    <generator object <genexpr> at 0x0000018EBFF0F308>

    我们知道可以直接打印出list的每一个元素,但我们怎么打印出generator中的每一元素呢?通过它的定义,我们知道应该在循环的过程中,不断获取下一个元素。通过next() 函数可以获得generator的下一个返回值:

    >>> next(g)
    0
    >>> next(g)
    1

    若要获取generator中的所有元素,可以使用for 循环,因为generator也是可迭代对象:

    >>> g = (x * x for x in range(10))
    >>> for n in g:
        print(n)
    
        
    0
    1
    4
    9
    16
    25
    36
    49
    64
    81

       2.在介绍第2中方法前,我们先来谈一谈斐波那契数列(除第一,二两数外,任意一个数等于前两个数相加),我们应该能写出函数:

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

    上面的函数可以输出斐波那契数列数列的前N个数:

    >>> fib(6)
    1
    1
    2
    3
    5
    8
    'done'

      我们认真观察可以发现fib函数实际上是定义了斐波那契数列的推算规则(从第一个元素开始,推算出后面的元素),这种逻辑不正是generator的思想吗?我们只需要做一个小小的改动,就可以把上面的fib函数变成generator,只需把print(b)改成yield b 就可以了:

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

    这就是定义generator的另一种方法:如果一个函数定义中包含yield 关键字,那么这个函数就不再是一个普通的函数,而是一个generator

    >>> f = fib(6)
    >>> f
    <generator object fib at 0x0000018EBFF0F308>
    >>> for n in fib(6):
        print(n)
    
        
    1
    1
    2
    3
    5
    8

    但这样就拿不到generator的return 语句的返回值。如果想要拿到返回值,就必须捕获StopIteration错误,返回值包含在StopIteration value 中:

    >>> g = fib(6)
    >>> while True:
        try:
            x = next(g)
            print('g:',x)
        except StopIteration as e:
            print('Generator return value:', e.value)
            break
    
        
    g: 1
    g: 1
    g: 2
    g: 3
    g: 5
    g: 8
    Generator return value: done
    Tips:generator和函数的执行流程不一样。函数时顺序执行,遇到return语句或者最后一行函数语句就返回。而变成generator的函数,在每次调用next()的时候执行,遇到yield语句返回,再执行时从yield语句处继续执行。

    迭代器

      我们已经知道能作用于for循环的对象都是可迭代对象:Iterable。数据类型可以分为集合数据类型:如listtupledictsetstr等;另一类是generator:包括生成器和带yieldgenerator的函数。

      迭代器(Iterator):可以被next()函数调用并不断返回下一个值的对象。可以使用isinstance()判断一个对象是否是Iterator对象:

    >>> from collections import Iterator
    >>> isinstance((x for x in range(10)), Iterator)
    True
    >>> isinstance([], Iterator)
    False
    >>> isinstance({}, Iterator)
    False
    >>> isinstance('abc', Iterator)
    False

    可以发现生成器都是Iterator对象,但list、dict、str虽然是Iterable,却不是Iterator。可以用iter()函数使这项Iterable 变成Iterator

    >>> isinstance(iter([]),Iterator)
    True

      思考:为什么list、dict、str等数据类型不是Iterator

      这是因为Python的Iterator对象表示的是一个数据流,我们可以把这个数据流看做是一个有序序列,但我们却不能提前知道序列的长度,只能不断通过next()函数按需计算下一个数据,所以Iterator的计算是惰性的,只有在需要返回下一个数据时它才会计算。所以Iterator甚至可以表示一个无限大的数据流,例如全体自然数,而使用list是永远不可能存储全体自然数的。

    Tips:凡是可作用于for循环的对象都是Iterable类型;凡是可作用于next()函数的对象都是Iterator类型,表示一个惰性计算的序列;集合数据类型如list、dict等是Iterable但不是Iterator,不过可以通过iter()函数获得一个Iterator对象;Python的for循环本质上就是不断通过调用next()函数实现的。
  • 相关阅读:
    爬虫第二篇:爬虫详解之存储数据
    数据分析第六篇:机器学习分类
    数据分析第五篇:数据库多表连接操作
    数据分析第四篇:数据清洗
    pyspark搭建使用
    YARN 调度器
    InnoDB: Error: page xxx log sequence number xx xxx InnoDB: is in the future! Current system log sequence number xx xxx.
    瞬时连接所属进程
    NTP工作原理
    kudu NTP问题优化
  • 原文地址:https://www.cnblogs.com/yunche/p/8872424.html
Copyright © 2011-2022 走看看