Знакомство с классами в Python
Python — это высокоуровневый язык объектно-ориентированного программирования, созданный специально для того, чтобы помочь программистам в написании ясного логичного кода для проектов любого размера.
Чёткость, лёгкость в понимании и в то же время мощь — нигде больше не проявляются эти принципы так, как в классах Python.
Класс Python
Почти всё в Python можно назвать объектом. И у каждого объекта, естественно, есть свои характеристики, свойства и функции.
Мы можем считать класс неким «макетом» для создания объектов. Точнее сказать — нашим собственным специально настраиваемым макетом. А раз мы сами под себя его настраиваем, то и задавать ему можем любые характеристики, свойства и функции!
Начнём с простого примера. Создадим класс со свойством «x», где x=10. Вот определение нашего класса:
Вот и всё! Мы создали наш первый класс Python my_class со свойством x и значением 10. Чтобы использовать класс, вызываем нашу функцию. Дальше мы можем обращаться к любым свойствам класса по отдельности.
Данный код выводит число 10. Всё просто!
Ещё мы можем поменять переменную, просто присвоив ей новое значение. Вместо того, чтобы x был равен 10, пусть он равняется строке «Bob».
Функция __init__()
У всех классов есть функция __init__() . Её можно менять в любое время. Функция __init__() выполняется всякий раз, когда из класса создаётся объект. Она может использоваться для инициализации переменных класса. __init__() становится очень полезной, когда нам нужно, чтобы класс Python всегда начинался с тех или иных свойств.
В качестве примера возьмём следующий код:
Здесь у нас есть два человека с классом Person и именами Bob и Kate (типа мы их создали). В обоих классах исполнялась функция __init__() , инициализируя переменные класса для имени, пола и страны человека.
Что касается имени и пола, мы передали классу свои переменные, которые требовались в __init__() . Переменная «страна» инициализировалась при создании объекта в этой же функции, но с одним отличием: по причине того, что она не является переменной функции __init__() , значение не может быть задано извне. Поэтому у всех будет одна страна — США.
Результат этого кода будет такой:
Функции класса
Как любой объект, классы Python могут содержать функции! Находясь внутри класса, функции ведут себя точно так же, как вне его. Единственное отличие — способность функций класса обращаться непосредственно к переменным класса, не принимая их в качестве аргументов.
Первая функция класса встречается в строке 9 print_info() , которая выводит всю информацию о нашем объекте Person. Заметьте, что теперь, используя переменные класса с self , мы можем получить информацию о Бобе из любого места класса! Теперь, когда у нас есть прямой доступ ко всей информации, применять функции к объектам Python стало намного удобнее.
С другой стороны, трудно не заметить, как много кода нам требуется для отображения информации о Бобе с использованием print() в Python. Ну и наличие функций, специально предназначенных для того, чтобы конкретный тип-класс определялся в рамках этого класса, лучше организует код.
Вторая функция, которую мы здесь написали, называется grow_person() . Она увеличивает возраст человека на заданное пользователем количество лет. Логично сделать её функцией класса, так как она связана с нашим классом Person. Код в конечном итоге выглядит гораздо более чистым и удобным для восприятия!
Заключение
Вот и всё. Вы прошли вводный курс, посвящённый классам Python.
Если вам этого мало, займитесь изучением программирования. Отлично подойдёт для начала вот этот сайт GeeksForGeeks website (Eng). Если вам нравятся приложения или вы предпочли бы записаться на какие-нибудь курсы, в Coursera есть курс Python for Everybody (Eng), в котором больше внимания уделяется приложениям.
KotazIO
ООП, или объектно-ориентированное программирование, это подход к программированию, основанный на использовании объектов, которые взаимодействуют друг с другом для решения задач.
- ООП как язык — каждый язык имеет свой набор правил и способов выражения мыслей. Аналогично, ООП имеет свои правила и способы выражения кода.
- ООП как машина — каждый объект является отдельной частью, которая работает вместе с другими частями для выполнения задачи. Аналогично, каждый объект в ООП является отдельной частью, которая работает вместе с другими объектами для выполнения задачи.
- ООП как дом — каждый объект является отдельной комнатой в доме, которая имеет свою уникальную функцию. Аналогично, каждый объект в ООП является отдельной частью программы, которая имеет свою уникальную функцию.
Принципы ООП
- Инкапсуляция: объединение данных и методов, которые работают с этими данными, в одном классе и скрытие их от остального кода.
- Наследование: возможность создавать новые классы на основе уже существующих и наследовать их свойства и методы.
- Полиморфизм: возможность объектов с одинаковым интерфейсом иметь разные реализации методов.
Еще два принципа ООП, которые не были упомянуты ранее: абстракция и композиция.
- Абстракция: создание упрощенной модели объекта, которая позволяет сосредоточиться на основных характеристиках объекта, игнорируя мелкие детали. Абстракция позволяет разрабатывать более гибкие и масштабируемые программы.
- Композиция: создание нового объекта путем объединения уже существующих объектов. Композиция позволяет создавать объекты, которые могут выполнять сложные задачи, используя функциональность других объектов.
В Python поддерживаются все принципы ООП, включая инкапсуляцию, наследование, полиморфизм, абстракцию и композицию. Кроме того, в Python есть множество инструментов и библиотек, которые облегчают работу с объектами и позволяют создавать более эффективные и гибкие программы.
Преимущества и недостатки ООП
✅ Преимущества ООП
- Более структурированный и модульный код
- Повторное использование кода
- Улучшенная поддерживаемость
❎ Недостатки ООП
- Высокий уровень абстракции может усложнить понимание кода
- Добавление новых функций может потребовать много времени и усилий
Другие подходы к программированию
В дополнение к ООП, существуют и другие подходы к программированию:
- Процедурное программирование — это подход, который фокусируется на выполняемых процедурах и функциях.
- Функциональное программирование — это подход, который фокусируется на использовании функций.
- Логическое программирование — это подход, который фокусируется на использовании формальной логики для решения задач.
Каждый из этих подходов имеет свои преимущества и недостатки, и выбор подхода зависит от конкретных потребностей проекта.
Парадигма программирования | Общее использование | Основные характеристики | Преимущества | Недостатки |
---|---|---|---|---|
Объектно-ориентированное программирование | Широко используется в разработке программного обеспечения | Использует объекты для представления данных и поведения | Инкапсуляция, Наследование, Полиморфизм, Абстракция, Композиция | Высокий уровень абстракции может усложнить понимание кода |
Процедурное программирование | Используется для крупномасштабных проектов, где важна организация кода | Использует процедуры и функции для представления поведения | Модульность, Повторное использование, Легкость поддержки | Может быть менее эффективным, чем другие парадигмы |
Функциональное программирование | Используется для математических и научных вычислений | Сфокусирован на вычислении функций и избегает изменения состояния и изменяемых данных | Неизменяемость, Референциальная прозрачность, Функции высшего порядка | Может быть сложным для изучения и понимания |
Логическое программирование | Используется для искусственного интеллекта и систем на основе знаний | Использует формальную логику для представления и решения задач | Декларативность, Легкость отладки, Может обрабатывать сложные задачи | Ограниченная поддержка для нелогических задач, Может быть менее эффективным, чем другие парадигмы |
Базовые определения
Класс — это шаблон для создания объектов, а объект — это экземпляр класса.
Методы — это функции, принадлежащие определенному классу.
Атрибуты — это переменные, принадлежащие определенному объекту. Переменные — это временные хранилища для значений.
ООП в python
Как создать класс в Python
Для создания класса в Python используется ключевое слово class , за которым следует имя класса.
Другие примеры
Классы Car , Robot , School и Person являются примерами создания классов в Python. В блоке кода, представляющем каждый класс, описываются его атрибуты и методы.
Car — класс для создания объектов машин, у которых есть цвет, модель, скорость и позиция. Метод go задает скорость машины, а метод get_position возвращает текущую позицию машины.
Robot — класс для создания объектов роботов, у которых есть имя и список запомненных текстов. Метод say выводит заданный текст, метод remember добавляет текст в список запомненных, а метод retell выводит все запомненные тексты в одну строку.
School — класс для создания объектов школ, у которых есть название и список учеников. Метод add_student добавляет ученика в список.
Person — класс для создания объектов людей, у которых есть имя. Метод say_hello выводит приветствие с именем объекта.
Пример использования этих классов:
Результат выполнения кода:
В этом примере мы создали объекты классов Car , Robot , School и Person , и использовали их методы для задания скорости машины, запоминания текста роботом, добавления учеников в школу и приветствия человека.
Таблица всех dunder методов и их описания
В ООП классы используются для создания объектов, которые могут быть использованы в программе. Однако, что делать, если мы хотим, чтобы объекты могли складываться, вычитаться или выполнять код при вызове? В Python для этого существуют специальные методы, называемые “dunder methods”, которые позволяют определить поведение объектов при выполнении различных операций. Например, метод add позволяет определить, как складывать два объекта, а метод call позволяет определить, как выполнять код при вызове объекта. Решение этих задач и многое другое будет рассмотрено в данном курсе. Dunder-методы это методы, которые начинаются и заканчиваются на два символа подчеркивания. Они используются для реализации различных магических методов.
Метод | Описание |
---|---|
__init__(self, . ) | Инициализация объекта |
__str__(self) | Возвращает строковое представление объекта |
__repr__(self) | Возвращает строковое представление объекта для отладки |
__len__(self) | Возвращает длину объекта |
__add__(self, other) | Возвращает результат сложения двух объектов |
__sub__(self, other) | Возвращает результат вычитания двух объектов |
__mul__(self, other) | Возвращает результат умножения двух объектов |
__truediv__(self, other) | Возвращает результат деления двух объектов |
__floordiv__(self, other) | Возвращает результат целочисленного деления двух объектов |
__mod__(self, other) | Возвращает результат операции модуля двух объектов |
__pow__(self, other) | Возвращает результат возведения одного объекта в степень другого |
__lt__(self, other) | Возвращает True, если текущий объект меньше, чем объект other |
__le__(self, other) | Возвращает True, если текущий объект меньше или равен объекту other |
__eq__(self, other) | Возвращает True, если текущий объект равен объекту other |
__ne__(self, other) | Возвращает True, если текущий объект не равен объекту other |
__gt__(self, other) | Возвращает True, если текущий объект больше, чем объект other |
__ge__(self, other) | Возвращает True, если текущий объект больше или равен объекту other |
__hash__(self) | Возвращает хеш-значение объекта |
__bool__(self) | Возвращает False, если объект пустой, и True, если нет |
__getattr__(self, name) | Вызывается при обращении к несуществующему атрибуту объекта |
__setattr__(self, name, value) | Вызывается при установке значения атрибута объекта |
__delattr__(self, name) | Вызывается при удалении атрибута объекта |
__call__(self, . ) | Вызывается при вызове объекта как функции |
__enter__(self) | Вызывается при входе в блок контекстного менеджера |
__exit__(self, exc_type, exc_val, exc_tb) | Вызывается при выходе из блока контекстного менеджера |
__iter__(self) | Возвращает итератор для объекта |
__next__(self) | Возвращает следующий элемент итератора |
__getitem__(self, key) | Возвращает элемент по индексу или ключу |
__setitem__(self, key, value) | Устанавливает значение элемента по индексу или ключу |
__delitem__(self, key) | Удаляет элемент по индексу или ключу |
__contains__(self, item) | Возвращает True, если объект содержит элемент item |
__index__(self) | Возвращает целочисленное представление объекта |
__reversed__(self) | Возвращает обратный итератор для объекта |
__enter__(self) | Вызывается при входе в блок контекстного менеджера |
__exit__(self, exc_type, exc_val, exc_tb) | Вызывается при выходе из блока контекстного менеджера |
__dir__(self) | Возвращает список атрибутов объекта |
__format__(self, format_spec) | Форматирует объект с помощью указанной спецификации формата |
__bytes__(self) | Возвращает байтовое представление объекта |
__class__(self) | Возвращает ссылку на класс объекта |
__instancecheck__(self, instance) | Возвращает True, если объект является экземпляром указанного класса |
__subclasscheck__(self, subclass) | Возвращает True, если класс является наследником указанного класса |
__getattribute__(self, name) | Вызывается при обращении к атрибуту объекта |
__setattr__(self, name, value) | Вызывается при установке значения атрибута объекта |
__delattr__(self, name) | Вызывается при удалении атрибута объекта |
__dir__(self) | Возвращает список атрибутов объекта |
__enter__(self) | Вызывается при входе в блок контекстного менеджера |
__exit__(self, exc_type, exc_val, exc_tb) | Вызывается при выходе из блока контекстного менеджера |
__get__(self, instance, owner) | Возвращает значение атрибута |
__set__(self, instance, value) | Устанавливает значение атрибута |
__delete__(self, instance) | Удаляет значение атрибута |
В чем разница между init и new? __init__ — это магический метод, который вызывается при создании нового объекта класса. __new__ — это метод, который создает новый объект класса.
self — это параметр метода, который ссылается на экземпляр класса, для которого вызывается метод. Он используется для доступа к атрибутам и методам объекта. Когда мы вызываем метод на объекте, Python автоматически передает этот объект в качестве первого параметра метода, используя параметр self . Это позволяет нам оперировать данными конкретного объекта и изменять их внутри методов. Без параметра self мы бы не имели доступа к атрибутам и методам объекта и не могли бы изменять их значения.
self — это ссылка на текущий объект. self используется для доступа к атрибутам и методам объекта.
Декоратор @property
@property — это декоратор, который позволяет использовать метод как атрибут.
Для использования декоратора @property необходимо определить метод с этим декоратором.
Пример класса с использованием декоратора @property :
Здесь класс Rectangle имеет два атрибута — length и width , которые задают длину и ширину прямоугольника соответственно. Декоратор @property используется для создания атрибутов area и perimeter , которые вычисляют площадь и периметр прямоугольника соответственно.
Пример использования класса:
Здесь мы создаем экземпляр класса Rectangle с длиной 4 и шириной 5, и затем используем атрибуты length , width , area и perimeter для получения соответствующих значений. Затем мы меняем значения атрибутов length и width , и снова используем атрибуты area и perimeter , чтобы получить новые значения.
Наследование
Наследование — это механизм, позволяющий создавать новый класс на основе существующего.
Для использования наследования необходимо указать базовый класс в определении нового класса.
В этом примере мы определяем базовый класс Animal , который имеет два атрибута — name и sound — и метод make_sound() , который выводит сообщение со звуком, издаваемым животным.
Затем мы определяем два класса, Cat и Dog , которые наследуются от Animal . Каждый из этих классов переопределяет метод __init__() , чтобы задать соответствующий звук, и добавляет свой собственный метод — purr() для Cat и wag_tail() для Dog .
Затем мы создаем экземпляры классов Cat и Dog , и вызываем методы make_sound() , purr() и wag_tail() для каждого из них. Это демонстрирует, как наследование позволяет создавать новые классы на основе существующих и добавлять им собственные методы и атрибуты.
Советы и частые ошибки при создании классов
Советы при создании классов
- Создавайте классы, которые отвечают только за одну задачу, и следуйте принципу единственной ответственности.
- Используйте наследование, чтобы избежать дублирования кода и упростить его поддержку.
- Используйте полиморфизм, чтобы обеспечить гибкость кода и упростить его использование.
- Используйте инкапсуляцию, чтобы скрыть детали реализации и обеспечить безопасность кода.
- Помните, что классы должны моделировать реальные объекты и процессы, а не только выполнять задачи.
- Документируйте каждый класс и его методы, чтобы облегчить его использование другими разработчиками.
Частые ошибки при создании классов
Одной из частых ошибок при создании классов является создание слишком сложных и неструктурированных классов.
Чтобы избежать создания слишком сложных и неструктурированных классов, можно использовать следующие советы:
- Разбейте класс на более мелкие подклассы, каждый из которых решает отдельную задачу.
- Используйте наследование, чтобы избежать дублирования кода.
- Избегайте сильной связности между классами, чтобы классы были более переносимыми и гибкими.
- Не добавляйте в классы функциональность, которая не относится к их основной задаче.
- Избегайте создания слишком многих методов и атрибутов в классе, чтобы класс был проще в использовании и понимании.
- Используйте исключения, чтобы сообщать об ошибках внутри класса, вместо того, чтобы просто возвращать ошибочное значение.
- Разделяйте классы на слои, чтобы каждый слой решал свою задачу, и классы внутри слоя имели схожую функциональность.
Особенности ООП в Python
Одной из особенностей ООП в Python является отсутствие инкапсуляции в чистом виде. В Python нет специальных модификаторов доступа к атрибутам и методам объекта, и все они по умолчанию являются открытыми. Однако, существует соглашение об именовании, которое указывает на то, что атрибуты и методы, которые начинаются с символа подчеркивания, не должны использоваться вне класса. Это соглашение не является обязательным и может быть нарушено, но его соблюдение делает код более понятным и удобочитаемым.
Еще одной особенностью ООП в Python является наличие множественного наследования. В Python классы могут наследоваться от нескольких базовых классов, что позволяет создавать более гибкие и мощные иерархии классов. Однако, множественное наследование может привести к сложностям в понимании кода и конфликтам имен атрибутов и методов.
Еще одной особенностью ООП в Python является наличие магических методов, которые позволяют определять поведение объектов в различных ситуациях. Например, метод __str__ определяет строковое представление объекта, а метод __len__ определяет длину объекта. Магические методы позволяют создавать более гибкие и мощные объекты, но их неправильное использование может привести к сложностям в понимании кода.
В Python также есть возможность использовать декораторы, которые позволяют модифицировать поведение методов и классов. Например, декоратор @property позволяет использовать метод как атрибут, а декоратор @staticmethod позволяет определить статический метод, который не имеет доступа к атрибутам и методам объекта.
Кроме того, Python поддерживает динамическое создание атрибутов и методов объектов, что позволяет создавать более гибкие и динамичные программы. Однако, динамическое создание атрибутов и методов может привести к сложностям в понимании кода и его отладке.
Наконец, Python поддерживает возможность перегрузки операторов, что позволяет определять поведение объектов при использовании стандартных операторов, таких как + , — , * и т.д. Перегрузка операторов позволяет создавать более удобный и понятный код, но ее неправильное использование может привести к сложностям
Примеры использования ООП
Примеры использования ООП в Python
ООП может быть использован для создания различных программ, таких как игры, приложения и веб-сайты.
Итоги
Мы рассмотрели основы ООП в Python, включая классы и объекты, наследование, полиморфизм и инкапсуляцию. Также мы рассмотрели некоторые советы и частые ошибки при создании классов, а также особенности ООП в Python, такие как отсутствие инкапсуляции в чистом виде, наличие множественного наследования, магических методов, декораторов и возможности перегр
Надеемся, что эта статья была полезной для вас и помогла лучше понять основы ООП в Python. Желаем вам успехов в дальнейшем изучении языка Python и создании своих программ.
9. Classes¶
Classes provide a means of bundling data and functionality together. Creating a new class creates a new type of object, allowing new instances of that type to be made. Each class instance can have attributes attached to it for maintaining its state. Class instances can also have methods (defined by its class) for modifying its state.
Compared with other programming languages, Python’s class mechanism adds classes with a minimum of new syntax and semantics. It is a mixture of the class mechanisms found in C++ and Modula-3. Python classes provide all the standard features of Object Oriented Programming: the class inheritance mechanism allows multiple base classes, a derived class can override any methods of its base class or classes, and a method can call the method of a base class with the same name. Objects can contain arbitrary amounts and kinds of data. As is true for modules, classes partake of the dynamic nature of Python: they are created at runtime, and can be modified further after creation.
In C++ terminology, normally class members (including the data members) are public (except see below Private Variables ), and all member functions are virtual. As in Modula-3, there are no shorthands for referencing the object’s members from its methods: the method function is declared with an explicit first argument representing the object, which is provided implicitly by the call. As in Smalltalk, classes themselves are objects. This provides semantics for importing and renaming. Unlike C++ and Modula-3, built-in types can be used as base classes for extension by the user. Also, like in C++, most built-in operators with special syntax (arithmetic operators, subscripting etc.) can be redefined for class instances.
(Lacking universally accepted terminology to talk about classes, I will make occasional use of Smalltalk and C++ terms. I would use Modula-3 terms, since its object-oriented semantics are closer to those of Python than C++, but I expect that few readers have heard of it.)
9.1. A Word About Names and Objects¶
Objects have individuality, and multiple names (in multiple scopes) can be bound to the same object. This is known as aliasing in other languages. This is usually not appreciated on a first glance at Python, and can be safely ignored when dealing with immutable basic types (numbers, strings, tuples). However, aliasing has a possibly surprising effect on the semantics of Python code involving mutable objects such as lists, dictionaries, and most other types. This is usually used to the benefit of the program, since aliases behave like pointers in some respects. For example, passing an object is cheap since only a pointer is passed by the implementation; and if a function modifies an object passed as an argument, the caller will see the change — this eliminates the need for two different argument passing mechanisms as in Pascal.
9.2. Python Scopes and Namespaces¶
Before introducing classes, I first have to tell you something about Python’s scope rules. Class definitions play some neat tricks with namespaces, and you need to know how scopes and namespaces work to fully understand what’s going on. Incidentally, knowledge about this subject is useful for any advanced Python programmer.
Let’s begin with some definitions.
A namespace is a mapping from names to objects. Most namespaces are currently implemented as Python dictionaries, but that’s normally not noticeable in any way (except for performance), and it may change in the future. Examples of namespaces are: the set of built-in names (containing functions such as abs() , and built-in exception names); the global names in a module; and the local names in a function invocation. In a sense the set of attributes of an object also form a namespace. The important thing to know about namespaces is that there is absolutely no relation between names in different namespaces; for instance, two different modules may both define a function maximize without confusion — users of the modules must prefix it with the module name.
By the way, I use the word attribute for any name following a dot — for example, in the expression z.real , real is an attribute of the object z . Strictly speaking, references to names in modules are attribute references: in the expression modname.funcname , modname is a module object and funcname is an attribute of it. In this case there happens to be a straightforward mapping between the module’s attributes and the global names defined in the module: they share the same namespace! 1
Attributes may be read-only or writable. In the latter case, assignment to attributes is possible. Module attributes are writable: you can write modname.the_answer = 42 . Writable attributes may also be deleted with the del statement. For example, del modname.the_answer will remove the attribute the_answer from the object named by modname .
Namespaces are created at different moments and have different lifetimes. The namespace containing the built-in names is created when the Python interpreter starts up, and is never deleted. The global namespace for a module is created when the module definition is read in; normally, module namespaces also last until the interpreter quits. The statements executed by the top-level invocation of the interpreter, either read from a script file or interactively, are considered part of a module called __main__ , so they have their own global namespace. (The built-in names actually also live in a module; this is called builtins .)
The local namespace for a function is created when the function is called, and deleted when the function returns or raises an exception that is not handled within the function. (Actually, forgetting would be a better way to describe what actually happens.) Of course, recursive invocations each have their own local namespace.
A scope is a textual region of a Python program where a namespace is directly accessible. “Directly accessible” here means that an unqualified reference to a name attempts to find the name in the namespace.
Although scopes are determined statically, they are used dynamically. At any time during execution, there are 3 or 4 nested scopes whose namespaces are directly accessible:
the innermost scope, which is searched first, contains the local names
the scopes of any enclosing functions, which are searched starting with the nearest enclosing scope, contain non-local, but also non-global names
the next-to-last scope contains the current module’s global names
the outermost scope (searched last) is the namespace containing built-in names
If a name is declared global, then all references and assignments go directly to the next-to-last scope containing the module’s global names. To rebind variables found outside of the innermost scope, the nonlocal statement can be used; if not declared nonlocal, those variables are read-only (an attempt to write to such a variable will simply create a new local variable in the innermost scope, leaving the identically named outer variable unchanged).
Usually, the local scope references the local names of the (textually) current function. Outside functions, the local scope references the same namespace as the global scope: the module’s namespace. Class definitions place yet another namespace in the local scope.
It is important to realize that scopes are determined textually: the global scope of a function defined in a module is that module’s namespace, no matter from where or by what alias the function is called. On the other hand, the actual search for names is done dynamically, at run time — however, the language definition is evolving towards static name resolution, at “compile” time, so don’t rely on dynamic name resolution! (In fact, local variables are already determined statically.)
A special quirk of Python is that – if no global or nonlocal statement is in effect – assignments to names always go into the innermost scope. Assignments do not copy data — they just bind names to objects. The same is true for deletions: the statement del x removes the binding of x from the namespace referenced by the local scope. In fact, all operations that introduce new names use the local scope: in particular, import statements and function definitions bind the module or function name in the local scope.
The global statement can be used to indicate that particular variables live in the global scope and should be rebound there; the nonlocal statement indicates that particular variables live in an enclosing scope and should be rebound there.
9.2.1. Scopes and Namespaces Example¶
This is an example demonstrating how to reference the different scopes and namespaces, and how global and nonlocal affect variable binding:
The output of the example code is:
Note how the local assignment (which is default) didn’t change scope_test‘s binding of spam. The nonlocal assignment changed scope_test‘s binding of spam, and the global assignment changed the module-level binding.
You can also see that there was no previous binding for spam before the global assignment.
9.3. A First Look at Classes¶
Classes introduce a little bit of new syntax, three new object types, and some new semantics.
9.3.1. Class Definition Syntax¶
The simplest form of class definition looks like this:
Class definitions, like function definitions ( def statements) must be executed before they have any effect. (You could conceivably place a class definition in a branch of an if statement, or inside a function.)
In practice, the statements inside a class definition will usually be function definitions, but other statements are allowed, and sometimes useful — we’ll come back to this later. The function definitions inside a class normally have a peculiar form of argument list, dictated by the calling conventions for methods — again, this is explained later.
When a class definition is entered, a new namespace is created, and used as the local scope — thus, all assignments to local variables go into this new namespace. In particular, function definitions bind the name of the new function here.
When a class definition is left normally (via the end), a class object is created. This is basically a wrapper around the contents of the namespace created by the class definition; we’ll learn more about class objects in the next section. The original local scope (the one in effect just before the class definition was entered) is reinstated, and the class object is bound here to the class name given in the class definition header ( ClassName in the example).
9.3.2. Class Objects¶
Class objects support two kinds of operations: attribute references and instantiation.
Attribute references use the standard syntax used for all attribute references in Python: obj.name . Valid attribute names are all the names that were in the class’s namespace when the class object was created. So, if the class definition looked like this:
then MyClass.i and MyClass.f are valid attribute references, returning an integer and a function object, respectively. Class attributes can also be assigned to, so you can change the value of MyClass.i by assignment. __doc__ is also a valid attribute, returning the docstring belonging to the class: "A simple example class" .
Class instantiation uses function notation. Just pretend that the class object is a parameterless function that returns a new instance of the class. For example (assuming the above class):
creates a new instance of the class and assigns this object to the local variable x .
The instantiation operation (“calling” a class object) creates an empty object. Many classes like to create objects with instances customized to a specific initial state. Therefore a class may define a special method named __init__() , like this:
When a class defines an __init__() method, class instantiation automatically invokes __init__() for the newly created class instance. So in this example, a new, initialized instance can be obtained by:
Of course, the __init__() method may have arguments for greater flexibility. In that case, arguments given to the class instantiation operator are passed on to __init__() . For example,
9.3.3. Instance Objects¶
Now what can we do with instance objects? The only operations understood by instance objects are attribute references. There are two kinds of valid attribute names: data attributes and methods.
data attributes correspond to “instance variables” in Smalltalk, and to “data members” in C++. Data attributes need not be declared; like local variables, they spring into existence when they are first assigned to. For example, if x is the instance of MyClass created above, the following piece of code will print the value 16 , without leaving a trace:
The other kind of instance attribute reference is a method. A method is a function that “belongs to” an object. (In Python, the term method is not unique to class instances: other object types can have methods as well. For example, list objects have methods called append, insert, remove, sort, and so on. However, in the following discussion, we’ll use the term method exclusively to mean methods of class instance objects, unless explicitly stated otherwise.)
Valid method names of an instance object depend on its class. By definition, all attributes of a class that are function objects define corresponding methods of its instances. So in our example, x.f is a valid method reference, since MyClass.f is a function, but x.i is not, since MyClass.i is not. But x.f is not the same thing as MyClass.f — it is a method object, not a function object.
9.3.4. Method Objects¶
Usually, a method is called right after it is bound:
In the MyClass example, this will return the string ‘hello world’ . However, it is not necessary to call a method right away: x.f is a method object, and can be stored away and called at a later time. For example:
will continue to print hello world until the end of time.
What exactly happens when a method is called? You may have noticed that x.f() was called without an argument above, even though the function definition for f() specified an argument. What happened to the argument? Surely Python raises an exception when a function that requires an argument is called without any — even if the argument isn’t actually used…
Actually, you may have guessed the answer: the special thing about methods is that the instance object is passed as the first argument of the function. In our example, the call x.f() is exactly equivalent to MyClass.f(x) . In general, calling a method with a list of n arguments is equivalent to calling the corresponding function with an argument list that is created by inserting the method’s instance object before the first argument.
If you still don’t understand how methods work, a look at the implementation can perhaps clarify matters. When a non-data attribute of an instance is referenced, the instance’s class is searched. If the name denotes a valid class attribute that is a function object, a method object is created by packing (pointers to) the instance object and the function object just found together in an abstract object: this is the method object. When the method object is called with an argument list, a new argument list is constructed from the instance object and the argument list, and the function object is called with this new argument list.
9.3.5. Class and Instance Variables¶
Generally speaking, instance variables are for data unique to each instance and class variables are for attributes and methods shared by all instances of the class:
As discussed in A Word About Names and Objects , shared data can have possibly surprising effects with involving mutable objects such as lists and dictionaries. For example, the tricks list in the following code should not be used as a class variable because just a single list would be shared by all Dog instances:
Классы
И снова обратимся к Лутцу, он явно скажет лучше чем я: Классы – это основные инструменты объектно-ориентированного программирования ( ООП ) в языке Python , поэтому в этой статье мы попутно рассмотрим основы ООП . ООП предлагает другой, часто более эффективный подход к программированию, при котором мы разлагаем программный код на составляющие, чтобы уменьшить его избыточность, и пишем новые программы, адаптируя имеющийся программный код, а не изменяя его.
Давайте еще определение ООП из Википедии:
Объе́ктно-ориенти́рованное программи́рование ( ООП ) — методология программирования, основанная на представлении программы в виде совокупности объектов, каждый из которых является экземпляром определённого класса, а классы образуют иерархию наследования.
- Объекты — данные + медоты(функции). Для создания объектов используются классы, все объекты являются экземпляром какого-то класса.
- Класс — объект, создающий новые объекты(экземпляры этого класса), объекты будут наследовать методы класса, которым они были порождены.
Некоторые утверждения#
- В python все является объектами. А значит и классы тоже являются объектами, поэтому нам важно разделять объект класса(сам класс) и экземпляры класса(объекты порожденные этим классом)
Класс как пространство имен#
Самому мне кажется что объяснение с этой стороны может путать, но давайте обозначим один момент, что классы чем-то похожи с модулями — они создают пространства имен, экземпляры классов тоже создают пространства имет, только пространство имен экземпляра связанно с пространством имен класса.
Создание классов#
Для создание класса в python используется инструкция class . Она сильно похожа на объявление функций( def ) и так же как и функция является исполняющейся инструкцией, помимо этого так же как и def , class создает объект, только в этот раз не объект функции, а объект класса.