В Python, как представить класс так же, как это делает именованный кортеж?

Сначала я написал некоторый код, используя именованный кортеж:

from collections import namedtuple


Count = namedtuple('Count', 'input_number multiple_of count remainder')


def get_count(input_number: int, multiple_of: int) -> Count:
    '''
    >>> get_count(100, 10)
    Count(input_number=100, multiple_of=10, count=10, remainder=0)
    >>> get_count(5, 6)
    Count(input_number=5, multiple_of=6, count=0, remainder=5)
    >>> get_count(999, 90)
    Count(input_number=999, multiple_of=90, count=11, remainder=9)
    '''
    count = input_number // multiple_of
    remainder = input_number % multiple_of
    return Count(
        input_number=input_number,
        multiple_of=multiple_of,
        count=count,
        remainder=remainder
    )

ПРИМЕЧАНИЕ. Когда объект печатается, он имеет формат: например, количество (входное_число = 100, множественное_из = 10, количество = 10, остаток = 0).

Есть ли Pythonic/лучший способ представления обычных классов таким же образом?

Затем я хотел включить функции набора текста python3.6+ и обнаружил, что не могу сделать это с namedtuple из коллекций, поэтому я отредактировал класс Count следующим образом:

from typing import NamedTuple


class Count(NamedTuple):
    input_number: int
    multiple_of: int
    count: int
    remainder: int

Меня не совсем устраивало, что мне нужны две функции, поэтому я начал с новой:

class GetCount():
    '''
    >>> GetCount(100, 10)
    GetCount(input_number=100, multiple_of=10, count=10, remainder=0)
    >>> GetCount(5, 6)
    GetCount(input_number=5, multiple_of=6, count=0, remainder=5)
    >>> GetCount(999, 90)
    GetCount(input_number=999, multiple_of=90, count=11, remainder=9)
    '''

    def __init__(self, input_number: int, multiple_of: int):
        self.input_number: int = input_number
        self.multiple_of: int = multiple_of
        self.count: int = input_number // multiple_of
        self.remainder: int = input_number % multiple_of

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


person Dean Kayton    schedule 10.01.2020    source источник


Ответы (1)


Решение проблемы следующее:

class GetCount():
    '''
    >>> GetCount(100, 10)
    GetCount(input_number=100, multiple_of=10, count=10, remainder=0)
    >>> GetCount(5, 6)
    GetCount(input_number=5, multiple_of=6, count=0, remainder=5)
    >>> GetCount(999, 90)
    GetCount(input_number=999, multiple_of=90, count=11, remainder=9)
    '''

    def __init__(self, input_number: int, multiple_of: int):
        self.input_number: int = input_number
        self.multiple_of: int = multiple_of
        self.count: int = input_number // multiple_of
        self.remainder: int = input_number % multiple_of

    def __repr__(self):
        return f"{self.__class__.__name__}({', '.join([f'{k}={v}' for k, v in self.__dict__.items()])})"

Вы можете увидеть, что предоставленные тесты проходят, выполнив следующее:

import doctest
doctest.testmod(verbose=True)
person Dean Kayton    schedule 10.01.2020