Python的defaultdict模块和namedtuple模块
admin
2023-07-31 01:42:47
0

在Python中有一些内置的数据类型,比如int, str, list, tuple, dict等。Python的collections模块在这些内置数据类型的基础上,提供了几个额外的数据类型:namedtuple, defaultdict, deque, Counter, OrderedDict等,其中defaultdict和namedtuple是两个很实用的扩展类型。defaultdict继承自dict,namedtuple继承自tuple。

一、defaultdict

 1. 简介

在使用Python原生的数据结构dict的时候,如果用d[key]这样的方式访问,当指定的key不存在时,是会抛出KeyError异常的。但是,如果使用defaultdict,只要你传入一个默认的工厂方法,那么请求一个不存在的key时, 便会调用这个工厂方法使用其结果来作为这个key的默认值。

defaultdict在使用的时候需要传一个工厂函数(function_factory),defaultdict(function_factory)会构建一个类似dict的对象,该对象具有默认值,默认值通过调用工厂函数生成。

2. 示例

下面给一个defaultdict的使用示例:

123456789101112131415161718192021 In [1]: from collections import defaultdict In [2]: s = [(\’xiaoming\’, 99), (\’wu\’, 69), (\’zhangsan\’, 80), (\’lisi\’, 96), (\’wu\’, 100), (\’yuan\’, 98), (\’xiaoming\’, 89)] In [3]: d = defaultdict(list) In [4]: for k, v in s:   ...:     d[k].append(v)   ...:      In [5]: dOut[5]: defaultdict(<type \’list\’>, {\’lisi\’: [96], \’xiaoming\’: [99, 89], \’yuan\’: [98], \’zhangsan\’: [80], \’wu\’: [69, 100]}) In [6]: for k, v in d.items():   ...:     print \’%s: %s\’ % (k, v)   ...:     lisi: [96]xiaoming: [99, 89]yuan: [98]zhangsan: [80]wu: [69, 100]

对Python比较熟悉的同学可以发现defaultdict(list)的用法和dict.setdefault(key, [])比较类似,上述代码使用setdefault实现如下:

12345 s = [(\’xiaoming\’, 99), (\’wu\’, 69), (\’zhangsan\’, 80), (\’lisi\’, 96), (\’wu\’, 100), (\’yuan\’, 98), (\’xiaoming\’, 89)]d = {} for k, v in s:    d.setdefault(k, []).append(v)

3. 原理

从以上的例子中,我们可以基本了defaultdict的用法,下面我们可以通过help(defaultdict)了解一下defaultdict的原理。通过Python console打印出的help信息来看,我们可以发现defaultdict具有默认值主要是通过__missing__方法实现的,如果工厂函数不为None,则通过工厂方法返回默认值,具体如下:

123456 def __missing__(self, key):    # Called by __getitem__ for missing key    if self.default_factory is None:        raise KeyError((key,))    self[key] = value = self.default_factory()    return value

从上面的说明中,我们可以发现一下几个需要注意的地方:

a). __missing__方法是在调用__getitem__方法发现KEY不存在时才调用的,所以,defaultdict也只会在使用d[key]或者d.__getitem__(key)的时候才会生成默认值;如果使用d.get(key)是不会返回默认值的,会出现KeyError;

b). defaultdict主要是通过__missing__方法实现,所以,我们也可以通过实现该方法来生成自己的defaultdict,代码入下:

12345678910111213141516 In [1]: class MyDefaultDict(dict):   ...:     def __missing__(self, key):   ...:         self[key] = \’default\’   ...:         return \’default\’   ...:      In [2]: my_default_dict = MyDefaultDict() In [3]: my_default_dictOut[3]: {} In [4]: print my_default_dict[\’test\’]default In [5]: my_default_dictOut[5]: {\’test\’: \’default\’}

4. 版本

defaultdict是在Python 2.5之后才加入的功能,在旧版本的Python中是不支持这个功能的,不过,知道了它的原理,我们可以自己实现一个defaultdict。

1234567891011121314151617181920212223242526272829303132333435363738394041424344 # http://code.activestate.com/recipes/523034/try:    from collections import defaultdictexcept:    class defaultdict(dict):         def __init__(self, default_factory=None, *a, **kw):            if (default_factory is not None and                not hasattr(default_factory, \’__call__\’)):                raise TypeError(\’first argument must be callable\’)            dict.__init__(self, *a, **kw)            self.default_factory = default_factory         def __getitem__(self, key):            try:                return dict.__getitem__(self, key)            except KeyError:                return self.__missing__(key)         def __missing__(self, key):            if self.default_factory is None:                raise KeyError(key)            self[key] = value = self.default_factory()            return value         def __reduce__(self):            if self.default_factory is None:                args = tuple()            else:                args = self.default_factory,            return type(self), args, None, None, self.items()         def copy(self):            return self.__copy__()         def __copy__(self):            return type(self)(self.default_factory, self)         def __deepcopy__(self, memo):            import copy            return type(self)(self.default_factory, copy.deepcopy(self.items()))         def __repr__(self):            return \’defaultdict(%s, %s)\’ % (self.default_factory, dict.__repr__(self))

二、namedtuple

namedtuple主要用来产生可以使用名称来访问元素的数据对象,通常用来增强代码的可读性,在访问一些tuple类型的数据时尤其好用。其实,在大部分时候你应该使用namedtuple替代tuple,这样可以让你的代码更容易读懂,更加pythonic。举个例子:

1234567891011121314151617 from collections import namedtuple # 变量名和namedtuple中的第一个参数一般保持一致,但也可以不一样Student = namedtuple(\’Student\’, \’id name score\’)# 或者 Student = namedtuple(\’Student\’, [\’id\’, \’name\’, \’score\’]) students = [(1, \’Wu\’, 90), (2, \’Xing\’, 89), (3, \’Yuan\’, 98), (4, \’Wang\’, 95)] for s in students:    stu = Student._make(s)    print stu # Output:# Student(id=1, name=\’Wu\’, score=90)# Student(id=2, name=\’Xing\’, score=89)# Student(id=3, name=\’Yuan\’, score=98)# Student(id=4, name=\’Wang\’, score=95)

在上面的例子中,Student就是一个namedtuple,它和tuple的使用方法一样,可以通过index直接取,而且是只读的。这种方式比tuple容易理解多了,可以很清楚的知道每个值代表的含义。

参考文章:

再谈collections模块defaultdict()和namedtuple()

defaultdict 和 dict.__missing__

不可不知的Python模块: collections

Python collections 模块


相关内容

热门资讯

Mobi、epub格式电子书如... 在wps里全局设置里有一个文件关联,打开,勾选电子书文件选项就可以了。
定时清理删除C:\Progra... C:\Program Files (x86)下面很多scoped_dir开头的文件夹 写个批处理 定...
500 行 Python 代码... 语法分析器描述了一个句子的语法结构,用来帮助其他的应用进行推理。自然语言引入了很多意外的歧义,以我们...
scoped_dir32_70... 一台虚拟机C盘总是莫名奇妙的空间用完,导致很多软件没法再运行。经过仔细检查发现是C:\Program...
65536是2的几次方 计算2... 65536是2的16次方:65536=2⁶ 65536是256的2次方:65536=256 6553...
小程序支付时提示:appid和... [Q]小程序支付时提示:appid和mch_id不匹配 [A]小程序和微信支付没有进行关联,访问“小...
pycparser 是一个用... `pycparser` 是一个用 Python 编写的 C 语言解析器。它可以用来解析 C 代码并构...
微信小程序使用slider实现... 众所周知哈,微信小程序里面的音频播放是没有进度条的,但最近有个项目呢,客户要求音频要有进度条控制,所...
Apache Doris 2.... 亲爱的社区小伙伴们,我们很高兴地向大家宣布,Apache Doris 2.0.0 版本已于...
python清除字符串里非数字... 本文实例讲述了python清除字符串里非数字字符的方法。分享给大家供大家参考。具体如下: impor...