51Testing软件测试论坛

 找回密码
 (注-册)加入51Testing

QQ登录

只需一步,快速开始

微信登录,快人一步

手机号码,快捷登录

查看: 4142|回复: 0
打印 上一主题 下一主题

[转贴] Python字典不是不可以排序,是你方法没用对!

[复制链接]
  • TA的每日心情
    无聊
    昨天 09:34
  • 签到天数: 1052 天

    连续签到: 2 天

    [LV.10]测试总司令

    跳转到指定楼层
    1#
    发表于 2021-2-19 10:19:09 | 只看该作者 回帖奖励 |正序浏览 |阅读模式
    字典是Python语言中的一种数据结构,每一个字典元素是由一对key-value组成的。而字典的key和value分别以集合(Set)形似组织,以便快速查询。集合的存储形似通常是树的结构,所以搜索非常快。我们可以单独通过字典的keys方法和values方法获取键集合和值集合的可迭代对象,代码如下:
    1.  x = {'x':20,'a':12,'b':5}
    2.   print(x.keys())
    3.   print(x.values())
    复制代码
    执行这段代码,会输出如下的内容:
    1.  dict_keys(['x', 'a', 'b'])
    2.   dict_values([20, 12, 5])
    复制代码
     PS:dict_keys和dict_values是Python的两个内部类,他们都采用了树的结构对数据进行组织。现在的问题是,集合是无序的(因为是以树状结构存储的),但由于某些要求,我们期望得到有序的键值,这就引出了下面的几个问题:  Q1:集合可以排序吗?
      Q2:排序后,可以获取成对的key-value吗?
      Q3:除了按键值排序,可以按值排序吗?
      为了回答这几个问题,请继续看下面的内容。
      1. 集合可以排序吗?
      这个问题的答案是:No。既然称为集合,由于数据存储形式的原因,肯定是无序的,但我们可以做一个折中的处理。既然集合不能排序,那么可以将集合中的数据排序后,放到一个可以排序的数据结构(例如,列表)里,这样可以在一定程度上解决我们的问题。看下面的代码:
    1. x = {'x':20,'a':12,'b':5}
    2.   keys = sorted(x.keys())
    3.   values = sorted(x.values())
    4.   print(type(keys))
    5.   print(type(values))
    6.   print(keys)
    7.   print(values)
    复制代码
    在这段代码中,使用了sorted函数,该函数可以用于对序列进行排序,并将排序结果放到一个列表中,最后返回这个列表,所以执行这段代码,会输出如下的内容:
    1. <class 'list'>
    2.   <class 'list'>
    3.   ['a', 'b', 'x']
    4.   [5, 12, 20]
    复制代码
    现在我们的第一个问题解决了,使用sorted函数,将集合按升序排列,并以列表形式体现。如果想降序排列,那么需要将reverse参数设置为True,代码如下:
    1. keys = sorted(x.keys(), reverse=True)
    2.   values = sorted(x.values(), reverse=True)
    复制代码
     2. 排序后,可以获取成对的key-value吗?  现在来解决第2个问题。本文讨论的是字典,所以即使对key和value排序,也不是我们期望的,我们期望的是得到字典中的key-value对,而且是已经排序好的key-value对。这个问题也非常好解决,既然已经得到了排好序的key,那么就利用key从字典中获得对应的value,这样就可以组成一对key-value了,代码如下:
    1. x = {'x':20,'a':12,'b':5}
    2.   keys = sorted(x.keys())
    3.   for key in keys:
    4.       print(f"{key}:{x[key]}")
    复制代码
    执行这段代码,会输出如下的内容:
    1.  x:20
    2.   b:5
    3.   a:12
    复制代码
    很明显,输出的key-value对是按key升序排列的。  3. 除了按键值排序,可以按值排序吗?
      那值排序是没问题的,但排好序的值是没什么用的。因为字典并不能通过value反向获取key。所以通过前面的方法只能获取排好序的值。所以需要为sorted函数的key参数,该参数用于指定使用key还是value进行排序。如果使用的是Python 3.7或以上版本。可以为key参数值指定lambda表达式,代码如下:
    1.  y1 = {k: v for k, v in sorted(x.items(), key=lambda item: item[1])}
    2.   print(y1)
    3.   # 按值降序排列
    4.   y2 = {k: v for k, v in sorted(x.items(), key=lambda item: item[1], reverse=True)}
    5.   print(y2)
    复制代码
    这段代码使用了for-in表达式生成了字典,其中x是待排序的字典。items获取了key-value对。而lambda表达式的item就是当前的key-value对。item[0]表示key,item[1]表示value,所以使用item[1],就表示按value排序,而使用item[0],就是按key排序。  执行这段代码,会输出如下的结果:
    1.  {'b': 5, 'a': 12, 'x': 20}
    2.   {'x': 20, 'a': 12, 'b': 5}
    复制代码
    如果不想使用for-in表达式,也可以使用dict函数,代码如下:print(dict(sorted(x.items(), key=lambda item: item[1])))如果不想使用lambda表达式,或需要更复杂的排序规则,可以自定义排序规则函数,代码如下:
    1. def dict_val(x):
    2.       return x[1]
    3.   sorted_x = sorted(x.items(), key=dict_val)
    复制代码
    如果读者用的是3.6或以下版本的Python,可以使用下面的代码:
    1. import operator
    2.   # 按value
    3.   sorted_x = sorted(x.items(), key=operator.itemgetter(1))
    4.   print(type(sorted_x))  # list
    5.   print(sorted_x)  # 所以用列表替代
    6.   import operator
    7.   # 按key
    8.   sorted_x = sorted(x.items(), key=operator.itemgetter(0))
    9.   print(sorted_x)
    复制代码
    在这段代码中,会返回排好序的列表,列表元素是元组形式,第1个值是key,第2个值是value。当然,你可以将这些数据重新插入新的字典中。执行这段代码,会输出如下内容:
    1. [('b', 5), ('a', 12), ('x', 20)]

    2.   [('a', 12), ('b', 5), ('x', 20)]
    复制代码
     如果仍然想通过key搜索字典中的value,可以利用OrderedDict对象将sorted_x转换为有序字典,代码如下:
    1. import collections
    2.   sorted_dict = collections.OrderedDict(sorted_x)
    3.   print(type(sorted_dict))
    4.   print(sorted_dict)
    5.   print(sorted_dict.get('b'))
    复制代码
    执行这段代码,会输出如下内容:
    1. <class 'collections.OrderedDict'>
    2.   OrderedDict([('a', 12), ('b', 5), ('x', 20)])
    3.   5
    复制代码
     总结:本文的核心就是sorted函数,通过这个函数,既可以对key和value集合进行排序,也可以对key-value对进行排序,如果是后者,需要通过key指定到底是用key,还是用value排序,sorted函数返回的是排好序的列表,如果要得到排好序的字典,或者使用for-in表达式处理,或者使用dict函数转换。当然,如果只是想通过key搜索value,也可以将数据放到OrderedDict对象中。


















    分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
    收藏收藏1
    回复

    使用道具 举报

    本版积分规则

    关闭

    站长推荐上一条 /1 下一条

    小黑屋|手机版|Archiver|51Testing软件测试网 ( 沪ICP备05003035号 关于我们

    GMT+8, 2024-11-28 02:54 , Processed in 0.063849 second(s), 26 queries .

    Powered by Discuz! X3.2

    © 2001-2024 Comsenz Inc.

    快速回复 返回顶部 返回列表