改進型 Easydict

https://github.com/makinacorpus/easydict
可惜有個問題 如果 key 不存在會爆出錯誤 我覺得返回 None 比較合理

#_*_ coding: utf-8 _*_

#  https://github.com/makinacorpus/easydict

class EasyDict(dict):
    """
    Get attributes

    >>> d = EasyDict({'foo':3})
    >>> d['foo']
    3
    >>> d.foo
    3
    >>> d.bar
    Traceback (most recent call last):
    ...
    AttributeError: 'EasyDict' object has no attribute 'bar'

    Works recursively

    >>> d = EasyDict({'foo':3, 'bar':{'x':1, 'y':2}})
    >>> isinstance(d.bar, dict)
    True
    >>> d.bar.x
    1

    Bullet-proof

    >>> EasyDict({})
    {}
    >>> EasyDict(d={})
    {}
    >>> EasyDict(None)
    {}
    >>> d = {'a': 1}
    >>> EasyDict(**d)
    {'a': 1}

    Set attributes

    >>> d = EasyDict()
    >>> d.foo = 3
    >>> d.foo
    3
    >>> d.bar = {'prop': 'value'}
    >>> d.bar.prop
    'value'
    >>> d
    {'foo': 3, 'bar': {'prop': 'value'}}
    >>> d.bar.prop = 'newer'
    >>> d.bar.prop
    'newer'


    Values extraction

    >>> d = EasyDict({'foo':0, 'bar':[{'x':1, 'y':2}, {'x':3, 'y':4}]})
    >>> isinstance(d.bar, list)
    True
    >>> from operator import attrgetter
    >>> map(attrgetter('x'), d.bar)
    [1, 3]
    >>> map(attrgetter('y'), d.bar)
    [2, 4]
    >>> d = EasyDict()
    >>> d.keys()
    []
    >>> d = EasyDict(foo=3, bar=dict(x=1, y=2))
    >>> d.foo
    3
    >>> d.bar.x
    1

    Still like a dict though

    >>> o = EasyDict({'clean':True})
    >>> o.items()
    [('clean', True)]

    And like a class

    >>> class Flower(EasyDict):
    ...     power = 1
    ...
    >>> f = Flower()
    >>> f.power
    1
    >>> f = Flower({'height': 12})
    >>> f.height
    12
    >>> f['power']
    1
    >>> sorted(f.keys())
    ['height', 'power']
    """
    def __init__(self, d=None, **kwargs):
        if d is None:
            d = {}
        if kwargs:
            d.update(**kwargs)
        for k, v in d.items():
            setattr(self, k, v)
        # Class attributes
        for k in self.__class__.__dict__.keys():
            if not (k.startswith('__') and k.endswith('__')):
                setattr(self, k, getattr(self, k))

    def __setattr__(self, name, value):
        if isinstance(value, (list, tuple)):
            def get_EasyDict_list(value):
                return [EasyDict(x) if isinstance(x, dict) else x if not isinstance(x, (list, tuple)) else get_EasyDict_list(x) for x in value]
            value = get_EasyDict_list(value)
        else:
            value = EasyDict(value) if isinstance(value, dict) else value
        super(EasyDict, self).__setattr__(name, value)
        self[name] = value

    def __getattr__(self, name):
        """
        如果 key 不存在返回 None
        """
        try:
            return super(EasyDict, self).__getattr__(name)
        except AttributeError:
            try:
                return self[name]
            except KeyError:
                key = name or ""
                if key.startswith('__') and key.endswith('__'):
                    raise AttributeError
                return None


if __name__ == "__main__":
    import doctest
    doctest.testmod()

    import json
    d = EasyDict({'a': 1, 'b': [1,2], 'c': {'x': 1, 'y': 2},'d':[[{'x':3, 'y':4}]]})

    print d.a
    print d.b
    print d.c
    print d.c.x, d.c.y
    print d.d
    print d.e

    d.e = 100
    print d.e
    d.c.z = 4
    d.b.append(10)
    print d.d[0][0].x
    print d.d[0][0].y

    d.f = [1,2,3,4,5]
    print d.f
    print d['f']
    print json.dumps(d)

    print(hasattr(d, 'a'))
    print(hasattr(d, 'b'))
    print(hasattr(d, 'adfasfc'))
    print(hasattr(d, 'd'))
    print(hasattr(d[0][0], 'x'))
    print(hasattr(d[0][0], 'y'))
    print(hasattr(d[0][0], 'z'))

    x = EasyDict()
    x['abcd'] = 10
    print(x['abcd'])
    print(x.abcd)



?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市肥印,隨后出現(xiàn)的幾起案子憎妙,更是在濱河造成了極大的恐慌柠傍,老刑警劉巖擒抛,帶你破解...
    沈念sama閱讀 219,539評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件饱溢,死亡現(xiàn)場離奇詭異遏匆,居然都是意外死亡叁征,警方通過查閱死者的電腦和手機尸红,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,594評論 3 396
  • 文/潘曉璐 我一進店門吱涉,熙熙樓的掌柜王于貴愁眉苦臉地迎上來刹泄,“玉大人,你說我怎么就攤上這事怎爵⊙” “怎么了?”我有些...
    開封第一講書人閱讀 165,871評論 0 356
  • 文/不壞的土叔 我叫張陵疙咸,是天一觀的道長县匠。 經(jīng)常有香客問我,道長撒轮,這世上最難降的妖魔是什么乞旦? 我笑而不...
    開封第一講書人閱讀 58,963評論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮题山,結(jié)果婚禮上兰粉,老公的妹妹穿的比我還像新娘。我一直安慰自己顶瞳,他們只是感情好玖姑,可當我...
    茶點故事閱讀 67,984評論 6 393
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著慨菱,像睡著了一般焰络。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上符喝,一...
    開封第一講書人閱讀 51,763評論 1 307
  • 那天闪彼,我揣著相機與錄音,去河邊找鬼协饲。 笑死畏腕,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的茉稠。 我是一名探鬼主播描馅,決...
    沈念sama閱讀 40,468評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼而线!你這毒婦竟也來了铭污?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,357評論 0 276
  • 序言:老撾萬榮一對情侶失蹤吞获,失蹤者是張志新(化名)和其女友劉穎况凉,沒想到半個月后谚鄙,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體各拷,經(jīng)...
    沈念sama閱讀 45,850評論 1 317
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,002評論 3 338
  • 正文 我和宋清朗相戀三年闷营,在試婚紗的時候發(fā)現(xiàn)自己被綠了烤黍。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片知市。...
    茶點故事閱讀 40,144評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖速蕊,靈堂內(nèi)的尸體忽然破棺而出嫂丙,到底是詐尸還是另有隱情,我是刑警寧澤规哲,帶...
    沈念sama閱讀 35,823評論 5 346
  • 正文 年R本政府宣布跟啤,位于F島的核電站,受9級特大地震影響唉锌,放射性物質(zhì)發(fā)生泄漏隅肥。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,483評論 3 331
  • 文/蒙蒙 一袄简、第九天 我趴在偏房一處隱蔽的房頂上張望腥放。 院中可真熱鬧,春花似錦绿语、人聲如沸秃症。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,026評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽种柑。三九已至,卻和暖如春匹耕,著一層夾襖步出監(jiān)牢的瞬間莹规,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,150評論 1 272
  • 我被黑心中介騙來泰國打工泌神, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留良漱,地道東北人。 一個月前我還...
    沈念sama閱讀 48,415評論 3 373
  • 正文 我出身青樓欢际,卻偏偏與公主長得像母市,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子损趋,可洞房花燭夜當晚...
    茶點故事閱讀 45,092評論 2 355

推薦閱讀更多精彩內(nèi)容