Python __str__ и списки - PullRequest
       9

Python __str__ и списки

95 голосов
/ 08 апреля 2009

В Java, если я вызову List.toString (), он автоматически вызовет метод toString () для каждого объекта в списке. Например, если мой список содержит объекты o1, o2 и o3, list.toString () будет выглядеть примерно так:

"[" + o1.toString() + ", " + o2.toString() + ", " + o3.toString() + "]"

Есть ли способ получить подобное поведение в Python? Я реализовал метод __str __ () в своем классе, но когда я распечатывал список объектов, используя:

print 'my list is %s'%(list)

выглядит примерно так:

[<__main__.cell instance at 0x2a955e95f0>, <__main__.cell instance at 0x2a955e9638>, <__main__.cell instance at 0x2a955e9680>]

как я могу заставить python автоматически вызывать мой __str__ для каждого элемента в списке (или dict в этом отношении)?

Ответы [ 9 ]

123 голосов
/ 08 апреля 2009

Вызов строки в списке Python вызывает метод __repr__ для каждого элемента внутри. Для некоторых элементов __str__ и __repr__ одинаковы. Если вы хотите такое поведение, выполните:

def __str__(self):
    ...
def __repr__(self):
    return self.__str__()
20 голосов
/ 08 апреля 2009

Вы можете использовать понимание списка , чтобы автоматически создать новый список с каждым элементом str () 'd:

print([str(item) for item in mylist])
8 голосов
/ 08 апреля 2009

Две простые вещи, которые вы можете сделать, использовать функцию map или использовать понимание.

Но вы получите список строк, а не строку. Таким образом, вы также должны объединить строки.

s= ",".join( map( str, myList ) )

или

s= ",".join( [ str(element) for element in myList ] )

Затем вы можете распечатать этот составной строковый объект.

print 'my list is %s'%( s )
4 голосов
/ 08 апреля 2009

В зависимости от того, для чего вы хотите использовать этот вывод, возможно, более подходящим будет __repr__:

import unittest

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

    def __repr__(self):
        return repr(self.val)

class Test(unittest.TestCase):
    def testMain(self):
        l = [A('a'), A('b')]
        self.assertEqual(repr(l), "['a', 'b']")

if __name__ == '__main__':
    unittest.main()
3 голосов
/ 08 апреля 2009

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

def is_list(value):
    if type(value) in (list, tuple): return True
    return False

def list_str(value):
    if not is_list(value): return str(value)
    return [list_str(v) for v in value]

Ради интереса я сделал list_str () рекурсивно str () для всего содержимого списка.

0 голосов
/ 06 июня 2019

Этого должно хватить.

При печати списков, а также других контейнерных классов содержащиеся элементы будут напечатаны с использованием __repr__, поскольку __repr__ предназначено для использования для внутреннего представления объектов. Если мы позвоним: help(object.__repr__), он скажет нам:

Help on wrapper_descriptor:

__repr__(self, /)
    Return repr(self).

И если мы позвоним help(repr), то получится:

Help on built-in function repr in module builtins:

repr(obj, /)
    Return the canonical string representation of the object.

    For many object types, including most builtins, eval(repr(obj)) == obj.

Если для объекта реализовано __str__, а __repr__ не равно repr(obj), то будет выводиться вывод по умолчанию, как и print(obj), если не реализовано ни одно из них.

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

class C:           
    def __str__(self):
        return str(f"{self.__class__.__name__} class str ")

C.__repr__=C.__str__       
ci = C()    


print(ci)       #C class str 
print(str(ci))  #C class str 
print(repr(ci)) #C class str 
0 голосов
/ 26 января 2019

Выходные данные - это просто имя модуля объекта, имя класса, а затем адрес памяти в шестнадцатеричном формате, поскольку функция __repr__ не переопределяется.

__str__ используется для строкового представления объекта при использовании печати. Но так как вы печатаете список объектов, а не перебираете список для вызова метода str для каждого элемента, он выводит представление объектов.

Чтобы вызвать функцию __str__, вам нужно сделать что-то вроде этого:

'my list is %s' % [str(x) for x in myList]

Если вы переопределите функцию __repr__, вы можете использовать метод печати, как был раньше:

class cell:
    def __init__(self, id):
        self.id = id
    def __str__(self):
        return str(self.id) # Or whatever
    def __repr__(self):
        return str(self) # function invoked when you try and print the whole list.

myList = [cell(1), cell(2), cell(3)]
'my list is %s' % myList

Тогда вы получите "my list is [1, 2, 3]" в качестве вывода.

0 голосов
/ 21 апреля 2017

Вы можете использовать формат.

    def __str__(self):
       return("my list is".format(self.list1))
0 голосов
/ 08 апреля 2009

Как то так?

a = [1, 2 ,3]
[str(x) for x in a]
# ['1', '2', '3']
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...