python中使用自定义类实例作为字典的key

python中dict类型的key值要求是不可变类型,通常来说,我们一般采用int或者str类型来作为字典的key,但是在某些场景中,会造成一定的麻烦。

如我们有一个处理http Request的规则类名为Rule,其定义如下,它由一个Request path和其支持的Request methods数组组成:

class Rule(object):

    def __init__(self, path, methods):
        assert(isinstance(path, str))
        assert(isinstance(methods, list))
        self.path = path
        self.methods = [method.upper() for method in methods]

现在我们想为每一种Rule(非每一个Rule实例)关联一个对应的Handler对象,使用一个dict来保存对应关系。

r1 = Rule("/index", ["GET"])
r2 = Rule("/index", ["GET"])
d = {r1: handler}
print d[r2]
# 两个不同的对象,打印出None

r1和r2虽然是两个不同的对象实例,但是在业务逻辑上是一致的,因此如果我们想让两个在逻辑上一致的对象被认为是同一个key,可以通过一些手段达到这个效果。

为Rule添加两个方法__hash__和__eq__,其意义可以查看python官方文档。

class Rule(object):

    def __init__(self, path, methods):
        assert(isinstance(path, str))
        assert(isinstance(methods, list))
        self.path = path
        self.methods = [method.upper() for method in methods]

    def __hash__(self):
        return hash((self.path, str(self.methods)))

    def __eq__(self, other):
        return (self.path, self.methods) == (other.path, other.methods)

然后再执行一下上面的测试代码,发现可以顺利取到handler了:

r1 = Rule("/index", ["GET"])
r2 = Rule("/index", ["GET"])
d = {r1: handler}
print d[r2] == handler# 打印出True

 

你可能感兴趣的:(python中使用自定义类实例作为字典的key)