python-学习-ORM中遇到的 mapping 详解并再总结字典dict

python里的字典dict(mapping)如何使用

Python字典的构造函数有三个,dict()dict(**args)dict(mapping),其中第一个、第二个构造函数比较好理解也比较容易使用,
dict(mapping)这个构造函数的使用比较难理解。

1. dict()构造函数可以返回一个空的字典

In [7]: d = dict()
In [8]: print (d)
{}

2. dict(**arg)构造函数,传入参数用赋值表达式,可多个复制表达式,用逗号间隔即可。

In [9]: d = dict(a = 12, b = 13, c = 15)

In [10]: print d
{'a': 12, 'c': 15, 'b': 13}

3. 映射类型 dict(mapping)构造python字典构造函数

一个mapping对象将可哈希的值映射为任意对象。映射是可变对象。目前Python中只有一种标准映射类型,就是字典(dict)。dcit和Set集合一样也是用花括号表示,但是花括号中的每个元素都是一个键值对(key:value)。字典中的键值对也是无序的,且key必须是可哈希的不可变类型,如字符串、数字、布尔值和不包含可变类型的tuple。而list和包含可变类型的tuple是不能做字典的key的。另外,同一个字典中,key不能重复,否则会覆盖之前的值(key最好不用数字)
dict类构建函数

class dict(**kwarg)
class dict(mapping, **kwarg)
class dict(iterable, **kwarg)

上面的dict构建函数会返回一个新的字典,这个新的字典是从一个可选的位置参数和一组可能为空关键字参数初始化的。

dict的构建方式

  1. 将以逗号分隔的 key:value (键值对)列表放置在一对花括号中:{ },{'Tom':1, 'Jerry':2, 'Peter':3}

  2. 使用不传递位置参数的构建函数,创建空字典:dict(),等同于{ }

  3. 如果给定了位置参数,并且它是映射对象,则使用与该映射对象相同的键值对创建字典;否则,位置参数必须是可迭代对象。可迭代对象中的每个条目本身必须是具有两个对象的可迭代对象。每个条目的第一个对象将会成为新字典中的键,第二个对象将会成为对应的值。如果键多次出现,则该键的最后一个值将成为新字典中的相应值。

  4. 如果给出了关键字参数,则将关键字参数及其值添加到从位置参数创建的字典中。如果要添加的键已经存在,则来自关键字参数的值将替换位置参数中的值。

>>> a = {'one': 1, 'two': 2, 'three': 3}  # 将以逗号分隔的键值对列表放置在一对花括号中

>>> b = dict(one=1, two=2, three=3)  # 使用dict构建函数,并传递关键字参数

>>> c = dict([('two', 2), ('one', 1), ('three', 3)])  # 使用dict构建函数,并传递可迭代对象为位置参数

>>> d = dict({'three': 3, 'two': 2, 'one': 1})  # 使用dict构建函数,并传递可迭代对象作为位置参数

>>> e = dict({'three': 3, 'two': 2, 'one': 1}, four=4, five=5)  # 使用dict构建函数,同时传递可迭代位置参数和关键字参数
>>> e
{'five': 5, 'two': 2, 'three': 3, 'one': 1, 'four': 4}
>>> a == b == c == d
True

dict支持的操作

len(d)  # 返回字典d中的条目数
key in d  # 如果字典d中包含key这个键则返回True, 否则返回False
key not in d  # 与上面刚好相反

d[key]  # 返回key对应的条目的值,如果key不存在会抛出KeyError异常
get(key[, default])  # 如果key存在则返回其对应的值,否则返回default参数指定的值。如果default没有被指定,则default取None,因此该方法永远不会抛出KeyError异常。
setdefault(key[,default])  # 如果key存在则返回这个key对应的值,如果key不存在则插入一个键值对key:default并返回default的值。default默认值为None。
classmethod fromkeys(seq[,value]) # 这是个类方法,用于创建一个新的字典。seq序列中的元素将作为新字典中的key,而value将会是这些key共同的值,value默认为None。

copy()  # 返回该dict的一个浅拷贝

keys()  # 返回一个字典键的新视图
values()  # 返回一个字典值的新视图
items()  # 返回一个字典条目的新视图:(key, value)
iter(d)  # 返回字典键的一个迭代器,这是iter(d.keys())的简写方式

d[key] = value # 如果key不存在则表示新增一个条目,如果key已经存在则表示修改该条目的值
update([other])  # 用other中的键值对更新字典的内容,覆盖现有的key。other可以是一个字典对象或键/值对(比如长度为2的元组或其它可迭代对象)。如果指定了关键字参数,字典将会使用关键字参数对字典的键/值对更新,如:d.update(红色=1, 蓝色=2)

del d[key]  # 删除key对应的表木,如果key不存在会抛出KeyError异常
popitem()  # 从字典中删除并返回一个任意(key, value)对,如果字典为空会抛出KeyError异常。该方法对于破坏性的遍历字典很有用,常用于集合算法。
pop(key[,default])  # 如果key存在就删除它并返回它的值,否则返回default的值。如果default没有被给出并且key不存在,则抛出KeyError异常。
clear()  # 清空dict中的所有条目

操作实例 (简单,参照即可)

>>> d = {'one': 1, 'two': 2, 'three': 3}
>>> len(d)
3
>>> 'two' in d
True
>>> 'two' not in d
False
>>>
>>> d['three']
3
>>> d['four']
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
KeyError: 'four'
>>> d.get('three')
3
>>> d.get('four', 'default')
'default'
>>> d
{'one': 1, 'two': 2, 'three': 3}
>>> d.setdefault('three', 'default')
3
>>> d
{'one': 1, 'two': 2, 'three': 3}
>>> d.setdefault('four', 'default')
'default'
>>> d
{'one': 1, 'two': 2, 'four': 'default', 'three': 3}
>>> dict.fromkeys(['one', 'two', 'three'])
{'one': None, 'two': None, 'three': None}
>>> dict.fromkeys(['one', 'two', 'three'], 'default')
{'one': 'default', 'two': 'default', 'three': 'default'}
>>>
>>> d.copy()
{'one': 1, 'two': 2, 'four': 'default', 'three': 3}
>>> d['four'] = 4
>>> d
{'one': 1, 'two': 2, 'four': 4, 'three': 3}
>>> d['five'] = 5
>>> d
{'one': 1, 'two': 2, 'five': 5, 'four': 4, 'three': 3}
>>> d.update({'two': 'II', 'six': 6})
>>> d
{'one': 1, 'two': 'II', 'four': 4, 'five': 5, 'six': 6, 'three': 3}
>>> d.update(two=2, seven=7)
>>> d
{'one': 1, 'two': 2, 'four': 4, 'five': 5, 'six': 6, 'seven': 7, 'three': 3}
>>> d.update([('three', 'III'), ('eight', 8)])
>>> d
{'one': 1, 'two': 2, 'four': 4, 'five': 5, 'eight': 8, 'six': 6, 'seven': 7, 'three': 'III'}
>>>
>>> d.keys()
dict_keys(['one', 'two', 'four', 'five', 'eight', 'six', 'seven', 'three'])
>>> d.values()
dict_values([1, 2, 4, 5, 8, 6, 7, 'III'])
>>> d.items()
dict_items([('one', 1), ('two', 2), ('four', 4), ('five', 5), ('eight', 8), ('six', 6), ('seven', 7), ('three', 'III')])
>>> iter(d)
<dict_keyiterator object at 0x000001EA21A179A8>
>>> iter(d.keys())
<dict_keyiterator object at 0x000001EA21A175E8>
>>>
>>> del d['eight']
>>> d
{'one': 1, 'two': 2, 'four': 4, 'five': 5, 'six': 6, 'seven': 7, 'three': 'III'}
>>> d.popitem()
('one', 1)
>>> d
{'two': 2, 'four': 4, 'five': 5, 'six': 6, 'seven': 7, 'three': 'III'}
>>> d.pop('three')
'III'
>>> d
{'two': 2, 'four': 4, 'five': 5, 'six': 6, 'seven': 7}
>>> d.pop('three')
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
KeyError: 'three'
>>> d.pop('three',None)
>>> d.pop('three','default')
'default'
>>> d.clear()
>>> d.popitem()
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
KeyError: 'popitem(): dictionary is empty'

字典视图对象(Dictionary view objects)

对于dict.keys()dict.values()dict.items()这些函数,在Python 2.x中返回的是list对象,而在Python 3.x中返回的是一个叫做字典视图的对象。

>>> d = {'one': 1, 'two': 2, 'three': 3}
>>> dkeys = d.keys()
>>> dkeys
dict_keys(['one', 'three', 'two'])
>>> dvalues = d.values()
>>> dvalues
dict_values([1, 3, 2])
>>> ditems = d.items()
>>> ditems
dict_items([('one', 1), ('three', 3), ('two', 2)])

字典视图在字典条目上提供了一个动态视图,这意味着当字典发生变化时,相应的视图也会发生变化。要保存固定的值需要将字典视图强制转换为其他类型的对象进行保存,通常保存为list类型。

>>> d['four'] = 4
>>> d
{'one': 1, 'three': 3, 'two': 2, 'four': 4}
>>> dkeys
dict_keys(['one', 'three', 'two', 'four'])
>>> dvalues
dict_values([1, 3, 2, 4])
>>> ditems
dict_items([('one', 1), ('three', 3), ('two', 2), ('four', 4)])

总结

回看廖神的元类里一段代码

class Model(dict, metaclass=ModelMetaclass):

    def __init__(self, **kw):
        super(Model, self).__init__(**kw)

    def __getattr__(self, key):
        try:
            return self[key]
        except KeyError:
            raise AttributeError(r"'Model' object has no attribute '%s'" % key)

    def __setattr__(self, key, value):
        self[key] = value

    def save(self):
        fields = []
        params = []
        args = []    #  Model继承了元类ModelMetaclass里的mappings定义!
        for k, v in self.__mappings__.items():    # 这里mappings不是函数,而是ModelMetaclass里的 mappings = dict() !
            fields.append(v.name)
            params.append('?')
            args.append(getattr(self, k, None))
        sql = 'insert into %s (%s) values (%s)' % (self.__table__, ','.join(fields), ','.join(params))
        print('SQL: %s' % sql)
        print('ARGS: %s' % str(args))
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 222,183评论 6 516
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 94,850评论 3 399
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 168,766评论 0 361
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,854评论 1 299
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,871评论 6 398
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 52,457评论 1 311
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,999评论 3 422
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,914评论 0 277
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 46,465评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,543评论 3 342
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,675评论 1 353
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 36,354评论 5 351
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 42,029评论 3 335
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,514评论 0 25
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,616评论 1 274
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 49,091评论 3 378
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,685评论 2 360

推荐阅读更多精彩内容

  • 个人笔记,方便自己查阅使用 Py.LangSpec.Contents Refs Built-in Closure ...
    freenik阅读 67,715评论 0 5
  • http://python.jobbole.com/85231/ 关于专业技能写完项目接着写写一名3年工作经验的J...
    燕京博士阅读 7,583评论 1 118
  • 最近在慕课网学习廖雪峰老师的Python进阶课程,做笔记总结一下重点。 基本变量及其类型 变量 在Python中,...
    victorsungo阅读 1,701评论 0 5
  • 哎!心情沉重!于是,反思学生错误较普遍出现的原因,并逐个将问题归纳并做了梳理,又思考了解决的方案。 刚...
    唯美儿阅读 252评论 0 2
  • 不柔弱要坚强,这是我就是想要的。 很多人说女孩子就应该被爱护,被疼爱,然后做一个听话地,顺从地,安安静静地姑娘,可...
    骑着电动的小毛驴阅读 221评论 0 0