zoukankan      html  css  js  c++  java
  • Python 高效编程技巧实战(2-1)如何在列表,字典, 集合中根据条件筛选数据

    Python 高效编程技巧实战(2-1)如何在列表,字典, 集合中根据条件筛选数据

    学习目标

    1、学会使用 filter 借助 Lambda 表达式过滤列表、集合、元组中的元素;
    2、学会使用列表解析、字典解析和集合解析;
    3、学会使用随机数的库生成测试用例;
    4、学会使用列表生成式创建列表。

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

    知识点与例题讲解

    下面我们给出几种在列表、字典集合中筛选数据的解决方案:
    1、列表
    (1)filter 函数:filter(lamdba x:x>0,data)
    (2)列表解析:[x for x in data if x>=0]
    2、字典:
    字典解析:{k:v for k,v in d.iteritems() if v>90}
    3、集合:
    集合解析:{x for x in s if x%3==0}

    例1:过滤列表[3,9,-1,10,20,-2]中的负数,只得到正数。

    下面展示的是一种传统的做法,这种做法的思路很简单,就是将列表中的元素进行遍历,然后逐个进行判断,符合条件的就添加到另一个集合中。
    解答1:

    # -*- coding:utf-8 -*-
    data = [3, 9, -1, 10, 20, -2]
    res = []
    for x in data:
        if x >= 0:
            res.append(x)
    
    print res
    
    [3, 9, 10, 20]
    

    程序运行结果:[3, 9, 10, 20]。
    其实,我们可以使用 Python 提供的 filter 函数或者“列表解析”的语法来完成。

    解答2:使用 filter 函数,传入一个 Lambda 表达式,达到过滤元素的目的。

    # -*- coding:utf-8 -*-
    from random import randint
    
    # 注意:randint(-10, 10)  可以取到边界值 -10 和 10
    data = [randint(-10, 10) for _ in xrange(10)]
    
    print '随机生成的列表', data
    
    filter_data = filter(lambda x: x > 0, data)
    print '使用 filter 和 lambda 表达式过滤得到的新列表', filter_data
    
    随机生成的列表 [6, 7, -8, 6, 1, 10, 5, -10, 5, 4]
    使用 filter 和 lambda 表达式过滤得到的新列表 [6, 7, 6, 1, 10, 5, 5, 4]
    

    执行结果:
    随机生成的列表 [5, -3, 6, 5, -2, 4, -6, -5, 7, -3]
    使用 filter 和 lambda 表达式过滤得到的新的列表 [5, 6, 5, 4, 7]。

    说明:1、列表生成式在 python2.x 中使用 xrange() ,在 python3.x 中使用 range();
    2、randint(-10, 10) 可以取到边界值 -10 和 10;
    3、使用列表解析生成随机数的示例代码:

    # -*- coding:utf-8 -*-
    from random import randint
    
    # 注意:randint(-10, 10)  可以取到边界值 -10 和 10
    data = [randint(-10, 10) for _ in xrange(10)]
    
    print data
    
    [-7, -5, 2, 10, -8, 1, 6, 5, 4, -4]
    

    4、使用 data.len() 得到列表的长度。

    解答3:可以使用列表解析的方式,达到过滤列表的效果:

    # -*- coding:utf-8 -*-
    from random import randint
    
    # 注意:randint(-10, 10)  可以取到边界值 -10 和 10
    data = [randint(-10, 10) for _ in xrange(10)]
    
    print '随机生成的列表', data
    
    filter_data = [x for x in data if x > 0]
    print '使用列表解析过滤的列表', filter_data
    
    随机生成的列表 [8, -10, -4, 4, 7, 10, -7, -9, -7, -1]
    使用列表解析过滤的列表 [8, 4, 7, 10]
    

    执行结果:
    随机生成的列表 [8, -2, -3, 6, 8, -10, -4, 5, 8, 2]
    使用列表解析过滤的列表 [8, 6, 8, 5, 8, 2]

    结论:列表解析(解答3)和 filter (解答2)的方式会更快一些,远快于我们使用传统的方案(解答1),我们可以使用 IPython 统计函数执行时间的工具类 timeit 来测试一下(待补充)。

    例2:筛选出字典 {'liwei':80,'wudi':88,'yuanlian':75,'zhouguang':90} 中高于 85 的项

    首先,我们先生成一组随机的字典:

    # -*- coding:utf-8 -*-
    from random import randint
    data = {x: randint(60, 100) for x in xrange(1, 21)}
    print data
    
    {1: 87, 2: 93, 3: 66, 4: 82, 5: 65, 6: 72, 7: 94, 8: 75, 9: 99, 10: 79, 11: 71, 12: 86, 13: 65, 14: 85, 15: 73, 16: 71, 17: 86, 18: 88, 19: 68, 20: 92}
    

    执行结果:{1: 91, 2: 100, 3: 95, 4: 93, 5: 79, 6: 83, 7: 79, 8: 80, 9: 77, 10: 98, 1。

    说明:1、使用列表解析的方式生成的字典,要使用花括号括起来,这一点和例 1 的中使用中括号要区别开来;
    2、字典的元素是键值对,所以使用 x: randint(60, 100) ,注意键和值中间有一个冒号;
    3、xrange(1,21) 得到的列表不包括21,而 randint(60,100) 包括60,也包括 100,这一点要区别开来。

    下面我们来解答这个问题,根据题目要求,我们须要根据值进行过滤,得到分数(value)大于 90 分的同学。我们的思路是:我们既然要同时迭代键和值,就要迭代它的 iteritems(同时迭代字典中的键和值)。

    # -*- coding:utf-8 -*-
    
    from random import randint
    
    data = {x: randint(60, 100) for x in xrange(1, 21)}
    
    filter_data = {k: v for k, v in data.iteritems() if v > 90}
    
    print '随机生成的字典',data
    print '过滤以后的字典数据',filter_data
    
    随机生成的字典 {1: 60, 2: 79, 3: 61, 4: 87, 5: 74, 6: 73, 7: 91, 8: 84, 9: 78, 10: 63, 11: 83, 12: 100, 13: 63, 14: 91, 15: 61, 16: 75, 17: 78, 18: 79, 19: 78, 20: 77}
    过滤以后的字典数据 {12: 100, 14: 91, 7: 91}
    

    执行结果:随机生成的字典 {1: 75, 2: 75, 3: 83, 4: 86, 5: 84, 6: 60, 7: 85, 8: 76, 9: 86, 10: 67, 11: 98, 12: 83, 13: 82, 14: 64, 15: 86, 16: 85, 17: 95, 18: 70, 19: 75, 20: 76}
    过滤以后的字典数据 {17: 95, 11: 98}。

    说明:1、对比列表解析和字典解析的语法:
    列表解析:[x for x in data if x>0]
    字典解析:{k:v for k,v in data.iteritems() if v>90}
    2、默认情况下, dict 迭代的是 key 。如果要迭代 value ,可以用 for value in d.itervalues(),如果要同时迭代 key 和 value ,可以用 for k, v in d.iteritems()。因此,对于字典这个数据结构而言,可以遍历 iteritems 、所有的键、所有的值。

    下面,我们来看一下集合解析的语法。

    例3:筛选出集合{77,89,32,20,15,28}中能被 3 整除的元素

    解法:

    # -*- coding:utf-8 -*-
    
    from random import randint
    
    # 随机生成在 1 到 50(包括1和50)的列表
    data = [randint(1, 50) for x in xrange(1, 21)]
    print 'data', data
    # 把列表放入集合中,删除了重复的元素
    s = set(data)
    print 's', s
    # 使用集合的解析,得到了集合中能被 3 整除的元素
    filter_set = {x for x in s if x % 3 == 0}
    print 'filter_set', filter_set
    
    data [17, 48, 25, 19, 35, 23, 43, 25, 20, 45, 42, 5, 48, 25, 42, 22, 50, 45, 16, 17]
    s set([16, 35, 5, 42, 43, 45, 48, 17, 50, 19, 20, 22, 23, 25])
    filter_set set([48, 42, 45])
    

    执行结果:data [1, 45, 21, 16, 42, 50, 16, 39, 13, 36, 16, 43, 50, 3, 24, 34, 15, 1, 17, 9]
    s set([1, 34, 3, 36, 39, 9, 42, 43, 45, 13, 16, 17, 50, 21, 24, 15])
    filter_set set([3, 36, 39, 9, 42, 45, 15, 21, 24])

    说明:
    1、根据列表得到一个集合:s = set(list),可以将列表中重复的元素去除;
    2、集合的解析没有冒号,表达式的结果也是使用花括号括起来的。

    参考资料

    1、廖雪峰老师的 Python 2.7 教程中关于“列表生成式”的讲解:
    网址:http://www.liaoxuefeng.com/wiki/001374738125095c955c1e6d8bb493182103fac9270762a000/00138681963899940a998c0ace64bb5ad45d1b56b103c48000

  • 相关阅读:
    Jersey的异常处理
    REST响应处理
    jersey REST的接口简述
    Firebird 同一字段的多行合并为一行
    Firebird/InterBase内置函数使用说明
    发布FireBird数据库所需要DLL文件
    unidac连接FireBird数据库
    打开与关闭Linux防火墙
    Linux FTP 命令
    Linux 命令修改系统时间
  • 原文地址:https://www.cnblogs.com/liweiwei1419/p/7085312.html
Copyright © 2011-2022 走看看