具有多个值的属性设置器

时间:2013-09-10 08:41:18

标签: python setter

我有一个属性setter,它通过获取两个字符串并对其进行散列来生成唯一的id:

@id.setter
def id(self,value1,value2):
    self._id = sha512(value1+value2)

我有两个问题:

  1. 是否允许(考虑好的python编码实践)以这种方式编码
  2. 如何将两个值传递给setter?

5 个答案:

答案 0 :(得分:9)

  

如何将两个值传递给setter?

您可以将iterable(元组,列表)传递给setter,例如:

class A(object):
    def __init__(self, val):
        self.idx = val

    @property    
    def idx(self):
        return self._idx

    @idx.setter
    def idx(self, val):
        try:
            value1, value2 = val
        except ValueError:
            raise ValueError("Pass an iterable with two items")
        else:
            """ This will run only if no exception was raised """
            self._idx = sha512(value1+value2)

<强>演示:

>>> a = A(['foo', 'bar'])     #pass a list
>>> b = A(('spam', 'eggs'))   #pass a tuple
>>> a.idx
<sha512 HASH object @ 0xa57e688>
>>> a.idx = ('python', 'org')  #works
>>> b.idx = ('python',)         #fails
Traceback (most recent call last):
    ...
    raise ValueError("Pass an iterable with two items")
ValueError: Pass an iterable with two items

答案 1 :(得分:7)

setter只能获取一个值,因此请使用元组:(value1, value2)

@id.setter
def id(self,value):          
    self._id = sha512(str(value))

...

self.id = (value1, value2)

(您没有发布sha512的内容。我假设您使用的是hashlib.sha512,而sha512正在调用update方法,这需要一个字符串作为输入。)

答案 2 :(得分:5)

我对这里的编码实践有些怀疑。作为API的用户,我总是假设无论我将其设置为属性,我都可以从同一属性中获取它。除此之外,拥有一个名为id mutable的东西看起来很可疑。

至于传递两个值,如何:

@id.setter
def id(self, vals):
    value1, value2 = vals
    self._id = sha512(value1+value2)

然后分配一个元组:

myobj.id = (val1, val2)

答案 3 :(得分:0)

我不知道这是不是你想要的,但这是传递两个或更多值的另一种方式:

@idx.setter
def idx(self, args):
    self._idx = sha512(''.join([str(i) for i in args]))

您可以避免列表或元组的限制:

a.idx = 'zere', 'fezfz'
a.idx = ("eddez", "erez")
a.idx = 'deze', 'drezre', 'ezre'

您必须测试异常并在init中处理excpetion。

答案 4 :(得分:0)

您可以像这样使用property()函数。

from hashlib import sha512

class U:

    def __init__(self, value1, value2):
        self.id_setter(value1, value2)

    def id_getter(self):
        return self._id

    def id_setter(self, value1: str, value2: str):
    value1, value2 = value1.encode(), value2.encode()
    self._id = sha512(value1+value2)

    id = property(id_getter, id_setter) # <----

u = U("Foo", "Bar")
print(u.id.hexdigest())