Понимание собственности в Python - PullRequest
3 голосов
/ 15 февраля 2012

Я хочу спросить о Python Property, я пишу код ниже ...

любой может объяснить, что выводит мой код, и объяснить простым способом, для чего это свойство и когда мы должны его использовать?

class C(object):
    def __init__(self):
        self.x = 'sulthan'
        self.y = 'ahnaf'
    @property
    def name(self):
        print self.x
        print self.y

Теперь, когда я запускаю код:

>>> c = C()
>>> c.name
sulthan
ahnaf
sulthan
ahnaf

Почему он печатается 2 раза?Извините за вопрос, я просто нуб, кто хочет понять Python ООП ...

Ответы [ 4 ]

3 голосов
/ 15 февраля 2012

Кажется, проблема с ipython , если вы используете приглашение python , оно показывает правильный вывод (единовременно)

[avasal@avasal]$ python
Python 2.7 (r27:82500, Sep 16 2010, 18:02:00) 
[GCC 4.5.1 20100907 (Red Hat 4.5.1-3)] on linux2
Type "help", "copyright", "credits" or "license" for more information.
>>> class C(object):
...     def __init__(self):
...         self.x = 'sulthan'
...         self.y = 'ahnaf'
...     @property
...     def name(self):
...         print self.x
...         print self.y
... 
>>> c1 = C()
>>> c1.name
sulthan
ahnaf
>>> 
[avasal@avasal]$ ipython 
Python 2.7 (r27:82500, Sep 16 2010, 18:02:00) 
Type "copyright", "credits" or "license" for more information.

IPython 0.10.2 -- An enhanced Interactive Python.
?         -> Introduction and overview of IPython's features.
%quickref -> Quick reference.
help      -> Python's own help system.
object?   -> Details about 'object'. ?object also works, ?? prints more.

In [1]: class C(object):
   ...:         def __init__(self):
   ...:             self.x = 'sulthan'
   ...:         self.y = 'ahnaf'
   ...:     @property
   ...:     def name(self):
   ...:             print self.x
   ...:         print self.y
   ...: 

In [2]: c1 = C()

In [3]: c1.name
sulthan
ahnaf
sulthan
ahnaf

In [4]:
2 голосов
/ 15 февраля 2012

Свойства особенно полезны, когда вы хотите получить прямой доступ к атрибутам объекта и в некоторых случаях вам нужно иметь дополнительный код вокруг методов получения / установки.

Представьте, что у вас есть автомобильный объект.

class Car(object):
    def __init__(self, make, model, year, vin):
        self.make = make
        self.model = model
        self.year = year
        self.vin = vin


my_car = Car('Ford', 'Taurus', 2005, '123ABCVINSARELONG999')

А теперь представьте, что мне нужно поменять год на машине.Я мог бы сделать это несколькими разными способами.

my_car.year = 2006

my_car.year = '2007'

Второй способ доставляет мне серьезные проблемы, если я хочу предположить, что год - это число.См. Следующее:

if(my_car.year > 2010):
    print('Shiney new!')

Проблема в том, что установка атрибутов напрямую очень чистая и удобная, но не дает нам возможности проводить какую-либо проверку данных (возможно, я хочу убедиться, что марка находится в спискенапример, известных производителей) или преобразование типов данных (в примере года).

Свойства приходят на помощь, позволяя нам настраивать код оболочки вокруг назначения и / или извлечения атрибута, но не требуя, чтобы мы это делалидля всех атрибутов или даже требует от нас заранее знать, что мы хотим сделать дополнительную проверку.После этого я могу изменить свой объект Car следующим образом, и он все равно будет использоваться точно таким же образом.

class Car(object):
    def __init__(self, make, model, year, vin):
        self.make = make
        self.model = model
        self._year = year
        self.vin = vin

    def get_year(self):
        return self._year

    def set_year(self, val):
        self._year = int(val)

    year = property(get_year, set_year)

Вы по-прежнему используете год как my_car.year, и вы по-прежнему устанавливаете его с * 1018.*, но теперь он конвертирует значение в int, где это необходимо, и выдает исключение, если значение, предоставленное пользователем, не преобразуется в int должным образом.

2 голосов
/ 15 февраля 2012

Это ошибка% autocall в ipython.Похожие билеты на панель запуска и github .Это, по-видимому, разрешено в текущей версии ipython.


Цель свойств - скрыть причудливую логику за тем, что выглядит как обычное присвоение / доступ к атрибуту.Это следует делать только в том случае, если у вас есть необходимая причина проектирования, поскольку python не является java, поэтому написание шаблонных методов получения и установки всего не требуется.

То, что Игнасио пытается сказать вам, - это то, что, похоже, нет никаких причин, почему у вас есть эти операторы печати в определении name().

Вот более обычная реализация вашего примера со свойствами.

class Person(object):
  def __init__(self):
    self._name = 'sulthan ahnaf'

  @property
  def name(self):
    return self._name

  @name.setter
  def name(self, value):
    self._name = value

Теперь у вас есть атрибут _name, доступ к которому вы можете контролировать с помощью «атрибута» name.Согласные взрослые оставят ваш _name в покое, потому что к нему добавлен знак подчеркивания, но учтите, что это не более чем конвенция.

0 голосов
/ 15 февраля 2012
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...