Использование распаковки для исключения лишних элементов
Допустим, вам необходимо распаковать N элементов из итерируемого объекта (список, кортеж), исключив при этом лишние элементы. Выражение со звёздочкой поможет это сделать.
Первая функция в примере исключит первые 2 элемента, а остальные вернет в качестве результата:
Код на GitHub
#std
Допустим, вам необходимо распаковать N элементов из итерируемого объекта (список, кортеж), исключив при этом лишние элементы. Выражение со звёздочкой поможет это сделать.
Первая функция в примере исключит первые 2 элемента, а остальные вернет в качестве результата:
>>>record = ('kiriharu', 'me@kiriha.ru', '8-800-555-35-35', '1337-1337')
>>>get_phones(record)
['8-800-555-35-35', '1337-1337']
Второй пример показывает, что распаковку можно использовать даже в середине присваивания. Функция отбросит первый и последний элемент в списке, а остальное запишет в grades:>>>grades = ('kiriharu', 1, 1, 1, 1, 'A')
>>>get_grades(grades)
[1, 1, 1, 1]
Выражение со звездочкой можно использовать как угодно - можно наоборот отбросить всё то, что попадает в него, а остальное вернуть. Попробуйте сами написать такую функцию :)Код на GitHub
#std
Несколько методов для эмуляции числовых типов
В питоне есть магические методы, которые позволяют складывать или умножать между собой объекты. Рассмотрим их.
Например, мы можем сделать класс для представления двухмерных векторов, которые могут складываться между собой создавая новый вектор или умножать вектор на скаляр.
#std
В питоне есть магические методы, которые позволяют складывать или умножать между собой объекты. Рассмотрим их.
Например, мы можем сделать класс для представления двухмерных векторов, которые могут складываться между собой создавая новый вектор или умножать вектор на скаляр.
>>> v1 = Vector2D(1, 1)
>>> v2 = Vector2D(5, 6)
>>> v1 + v2
Vector(6, 7)
>>> v1 * 3
Vector(3, 3)
GitHub#std
Коробка с питоном
Несколько методов для эмуляции числовых типов В питоне есть магические методы, которые позволяют складывать или умножать между собой объекты. Рассмотрим их. Например, мы можем сделать класс для представления двухмерных векторов, которые могут складываться…
Возвращайте NotImplemented для неподдерживаемых бинарных операций с вашим типом
В комментариях к предыдущему посту возник вопрос, что если вместо целочисленного scalar к нам придет float или какой-то другой тип?
Естественно, код в посте выше будет работать с ошибками, поэтому явно было бы выбросить исключение, но оказывается что так делать не нужно!
Оказывается, в Python есть следующий механизм: если метод представляющий бинарную операцию (
GitHub | Объяснение "на пальцах"
#std
В комментариях к предыдущему посту возник вопрос, что если вместо целочисленного scalar к нам придет float или какой-то другой тип?
Естественно, код в посте выше будет работать с ошибками, поэтому явно было бы выбросить исключение, но оказывается что так делать не нужно!
Оказывается, в Python есть следующий механизм: если метод представляющий бинарную операцию (
__add__, __mul__ и т.д.) возвращает NotImplemented, то Python попытается произвести отраженную операцию (для x.__mul__(1) вызовет 1.__rmul__(x), например) или другие операции, в зависимости от оператора. Как только все возможные методы вернут NotImplemented он сам возбудит исключение на месте вызова метода в вашем коде, а не в коде класса:>>> v1 = Vector2D(1, 2)Списки магических методов и рекомендации по их реализации можно найти в разделе Data model
>>> v1 * 1.1
TypeError: unsupported operand type(s) for *: 'Vector2D' and 'float'
GitHub | Объяснение "на пальцах"
#std
Как работает bool()?
Python может принимать любой объект в булевом контексте. Чтобы определить, является ли выражение истинным или ложным, применяется функция
По умолчанию, объект попытается вызвать метод
Резюмируя, получается такая цепочка:
Python может принимать любой объект в булевом контексте. Чтобы определить, является ли выражение истинным или ложным, применяется функция
bool(x), которая должна вернуть булево значение (True или False).По умолчанию, объект попытается вызвать метод
__bool__, который должен вернуть булево значение. Если реализованного метода нет, то Python попытается вызвать __len__, который должен вернуть 0 или значение больше ноля - при ноле результатом будет False, соответственно при значении больше ноля - True. Если ни один метод не реализован, то автоматически вернется True.Резюмируя, получается такая цепочка:
__bool__ → __len__ → True
#stdНемного фактов про str() и repr()
-
-
- Если метод
А что вы можете сказать про
#std
-
repr() вызывает метод __repr__ объекта, а метод str(), собственно __str__.-
repr() вызывается интерактивной оболочкой, при попытке вывести объект.- Если метод
__str__ в объекте не определен, то вызывается __repr__. А что вы можете сказать про
str() и repr()? Пишите ниже, в комментарии 👇#std
Коробка с питоном
Немного фактов про str() и repr() - repr() вызывает метод __repr__ объекта, а метод str(), собственно __str__. - repr() вызывается интерактивной оболочкой, при попытке вывести объект. - Если метод __str__ в объекте не определен, то вызывается __repr__. …
Когда использовать __str__ или __repr__?
Оба магических метода используются для получения строкового представления объекта. Но как их лучше использовать - можно увидеть на картинке.
Таким образом имеем, что
#std
Оба магических метода используются для получения строкового представления объекта. Но как их лучше использовать - можно увидеть на картинке.
Таким образом имеем, что
__repr__ должен использоваться для предоставлении информации об объекте разработчику, а __str__ должен быть читаемым и использоваться для представления информации пользователю.#std
В канале недавно уже был пост про namedtuple, но мне ещё есть что про него рассказать.
Первое, что я хотел бы вам показать - именованный кортеж можно создать с помощью любой структуры данных, которая поддерживает итерирование:
Именованные кортежи неизменяемые, но мы можем использовать метод
Первое, что я хотел бы вам показать - именованный кортеж можно создать с помощью любой структуры данных, которая поддерживает итерирование:
>>> lst = ['Mining', 'Excavating', 'Boiling']Ещё мы можем создать кортеж через метод
>>> Skills = namedtuple('Skills', lst)
>>> skills = Skills(1, 2, 3)
>>> skills
Skills(Mining=1, Excavating=2, Boiling=3)
_make():>>> skills = Skills._make([1, 2, 3])В примере используется список, но никто не мешает вам сделать такое же с кортежем, множеством и даже словарем (но вот только значения которые вы установите для ключа игнорируются).
Skills(Mining=1, Excavating=2, Boiling=3)
Именованные кортежи неизменяемые, но мы можем использовать метод
_replace(), чтобы изменить данные. Под капотом он использует метод _make для создания нового кортежа, поэтому, по сути, мы просто пересоздадим кортеж:>>> skills._replace(Mining=100)#std
Skills(Mining=100, Excavating=2, Boiling=3)
Типизация именованных кортежей.
Один из комментаторов указал, что
Именованный кортеж можно создать 2 путями - через наследование
Все методы, которые были в
#std
Один из комментаторов указал, что
namedtuple() объявлен устаревшим и в 3.10 его поддержка прекратится. Что же делать? Использовать NamedTuple, если в вашем проекте питон выше версии 3.6.Именованный кортеж можно создать 2 путями - через наследование
NamedTuple (класс Skills) или старым способом (AnotherSkills). Оба способа содержат аннотации типов, повышающие (как минимум) читаемость кода.Все методы, которые были в
namedtuple (_make(), _replace(), _asdict()) работают:>>> s = Skills(0)
>>> s._asdict()
{'mining': 0, 'excavating': 1}
Теперь мы можем посмотреть на аннотации:s.__annotations__Так же, ничего не мешает добавлять новые методы или оверрайтить существующие. В примере у
{'mining': <class 'int'>, 'excavating': <class 'int'>}
Skills переопределен метод для str():>>>str(s)
'Какой качок, уровень копания: 0\nуровень лесорубства: 100'
GitHub#std
Про применение all()
Стандартная библиотека Python полна полезных вещей, но полезные вещи есть даже среди встроенных функций. Одна из таких встроенных функций - это
Функция
К примеру,
Стандартная библиотека Python полна полезных вещей, но полезные вещи есть даже среди встроенных функций. Одна из таких встроенных функций - это
all().Функция
all() возвращает True, если все элементы истинные (или объект пустой). К примеру,
all() очень удобно использовать с генераторами. Например, мы можем проверить, являются ли все октеты в IP-адресе числами:>>> all(i.isdigit() for i in '127.0.0.b'.split('.'))
False
>>> all(i.isdigit() for i in '127.0.0.1'.split('.'))
True
#stdПоиск N максимальных или минимальных элементов.
Если вы хотите создать список из N максимальных или минимальных элементов, модуль
У этого модуля есть две функции:
Если вы хотите создать список из N максимальных или минимальных элементов, модуль
heapq вам поможет в этом.У этого модуля есть две функции:
nlargest() и nsmallest(), которые, соответственно ищут максимальные и минимальные элементы. Например:>>> nums = [10, 20, 30, 40, 55, 632, -3, 98321, 82, 0, 8]
>>> heapq.nlargest(4, nums)
[98321, 632, 82, 55]
>>> heapq.nsmallest(4, nums)
[-3, 0, 8, 10]
Обе функции принимают параметр key, который позволяет их использовать с сложными структурами данных. Например:currency = [
dict(name="Etherium", price=3323),
dict(name="Bitcoin", price=45538),
dict(name="ZCash", price=132),
dict(name="Litecoin", price=184),
dict(name="OmiseGo", price=8.933)
]
>>> heapq.nsmallest(2, currency, key=lambda s: s['price'])
[{'name': 'OmiseGo', 'price': 8.933}, {'name': 'ZCash', 'price': 132}]
>>> heapq.nlargest(2, currency, key=lambda s: s['price'])
[{'name': 'Bitcoin', 'price': 45538}, {'name': 'Etherium', 'price': 3323}]
#stdОпределяем наиболее часто встречающиеся элементы в последовательности.
Допустим, у нас есть некоторая последовательность из слов (words), и мы хотим узнать, какие элементы в ней встречаются чаще остальных.
Для этого можно использовать класс
Самому
#std
Допустим, у нас есть некоторая последовательность из слов (words), и мы хотим узнать, какие элементы в ней встречаются чаще остальных.
Для этого можно использовать класс
Counter из модуля collections в котором есть метод most_common(), который и выдаст список элементов, которые встречаются чаще остальных.Самому
Counter можно скормить любую последовательность элементы которой хешируются. В основе Counter лежит словарь который отображает количество значений, поэтому мы можем узнать количество элементов по ключу:>>>counts['test']
2
Кстати, эта задача часто встречается на собеседованиях и просят её реализовать без модуля collections. Как бы вы её решили? Ответы можете писать в комментарии, обсудим.#std
Ручной проход по списку
Задачка: нужно пройти по списку не используя цикл for.
Решение у этой задачи достаточно простое - можно использовать функцию
Здесь мы получаем итератор - это такой объект, который облегчает навигацию по коллекциям. Дальше в цикле вызываем
#std
Задачка: нужно пройти по списку не используя цикл for.
Решение у этой задачи достаточно простое - можно использовать функцию
next() и ловить исключение StopIteration. На скриншоте можно увидеть пример с использованием цикла while.Здесь мы получаем итератор - это такой объект, который облегчает навигацию по коллекциям. Дальше в цикле вызываем
next(), который получает следующий элемент и так до тех пор пока элементы не закончатся - индикатором этого будет вызов исключения StopIteration.#std
Я заметил, что редко использую функции из
Сам модуль itertools это набор из эффективных и быстрых по памяти инструментов, возвращающие итераторы. Сами по себе итераторы можно комбинировать с такими функциями как
А сегодня мы начнем c самых простых - c бесконечных итераторов. Всего по документации их три:
Ниже я буду указывать аргументы по умолчанию и возможные передаваемые типы как тайп-хинты. Надеюсь ни у кого не возникнет проблем с пониманием.
#itertools #std
itertools, в основном когда нужно произвести какие-то "красивые" шаманства. Теперь будем вспоминать саму библиотеку вместе, естественно с примерами :)Сам модуль itertools это набор из эффективных и быстрых по памяти инструментов, возвращающие итераторы. Сами по себе итераторы можно комбинировать с такими функциями как
map(), list() лямбдами или же использовать их с помощью цикла for. А сегодня мы начнем c самых простых - c бесконечных итераторов. Всего по документации их три:
count(), cycle() и repeat().Ниже я буду указывать аргументы по умолчанию и возможные передаваемые типы как тайп-хинты. Надеюсь ни у кого не возникнет проблем с пониманием.
#itertools #std