为了账号安全,请及时绑定邮箱和手机立即绑定

如何正确地将dict子类化并覆盖__getitem__和__setitem__

如何正确地将dict子类化并覆盖__getitem__和__setitem__

30秒到达战场 2019-10-17 14:58:47
我正在调试一些代码,我想找出何时访问特定词典。好吧,它实际上是一个类的子类,dict并实现了一些额外的功能。无论如何,我想做的是dict自己创建子类并添加重写__getitem__并__setitem__生成一些调试输出。现在,我有class DictWatch(dict):    def __init__(self, *args):        dict.__init__(self, args)    def __getitem__(self, key):        val = dict.__getitem__(self, key)        log.info("GET %s['%s'] = %s" % str(dict.get(self, 'name_label')), str(key), str(val)))        return val    def __setitem__(self, key, val):        log.info("SET %s['%s'] = %s" % str(dict.get(self, 'name_label')), str(key), str(val)))        dict.__setitem__(self, key, val)' name_label'是最终要设置的键,我想用它来标识输出。然后,我将要检测的类更改为子类,DictWatch而不是子类,dict并更改了对超级构造函数的调用。不过,似乎什么也没发生。我以为自己很聪明,但是我想知道我是否应该朝另一个方向发展。谢谢您的帮助!
查看完整描述

3 回答

?
慕工程0101907

TA贡献1887条经验 获得超5个赞

您正在做的事情绝对应该有效。我测试了您的类,除了在日志语句中缺少左括号之外,它还可以正常工作。我只能想到两件事。首先,您的log语句的输出设置正确吗?您可能需要logging.basicConfig(level=logging.DEBUG)在脚本的顶部放置一个。


其次,__getitem__和__setitem__只在所谓的[]访问。因此,请确保您仅DictWatch通过d[key]而不是d.get()和访问d.set()


查看完整回答
反对 回复 2019-10-17
?
Cats萌萌

TA贡献1805条经验 获得超9个赞

子类化时的另一个问题dict是,内置__init__函数不会调用update,而内置update函数不会调用__setitem__。因此,如果您希望所有setitem操作都通过您的__setitem__函数,则应确保自己调用该函数:


class DictWatch(dict):

    def __init__(self, *args, **kwargs):

        self.update(*args, **kwargs)


    def __getitem__(self, key):

        val = dict.__getitem__(self, key)

        print 'GET', key

        return val


    def __setitem__(self, key, val):

        print 'SET', key, val

        dict.__setitem__(self, key, val)


    def __repr__(self):

        dictrepr = dict.__repr__(self)

        return '%s(%s)' % (type(self).__name__, dictrepr)


    def update(self, *args, **kwargs):

        print 'update', args, kwargs

        for k, v in dict(*args, **kwargs).iteritems():

            self[k] = v


查看完整回答
反对 回复 2019-10-17
?
萧十郎

TA贡献1815条经验 获得超13个赞

那并不能真正改变结果(对于良好的日志记录阈值,它应该可以工作):您的init应该是:


def __init__(self,*args,**kwargs) : dict.__init__(self,*args,**kwargs) 

相反,因为如果使用DictWatch([(1,2 ,,(2,3)])或DictWatch(a = 1,b = 2)调用方法,则此操作将失败。


(或者,最好不要为此定义一个构造函数)


查看完整回答
反对 回复 2019-10-17
  • 3 回答
  • 0 关注
  • 634 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信