7

这与这个关于 floats的问题大致相同。

当您有一个可以转换为整数的值时,旧的%d会转换它,但格式不会。

class MyIntegerTenClass:
    def __int__(self):
        return 10
    def __str__(self):
        return 'ten'
ten = MyIntegerTenClass()
print '%d, %02X, %s' % (ten, ten, ten) # ok
print  '{0}'.format(ten) # ok
print  '{0:d}, {0:02X}'.format(ten) # ValueError: Unknown format code 'd' for object of type 'str'

有没有办法修改格式的行为,而不触及要格式化的值的类(不向__format__该类添加方法)?

编辑:我的目标是让格式取决于格式字符串,而不是值。因此,如果格式字符串显示“d”或“x”,则将值转换为 int,然后转换为十进制或十六进制表示。如果格式字符串为“s”,则直接将其转换为字符串。就像老人%那样。

实际上,我什__format__至可以在值的类中添加一个方法。但是,如果给定的格式规范是整数格式规范,我该如何检查呢?无需重新实现内置格式的格式规范解析器。

编辑:这是一个有__format__例外的解决方案。有更好的想法吗?

class MyIntegerTenClass:
    def __int__(self):
        return 10
    def __str__(self):
        return 'ten'
    def __format__(self, spec):
        fmt = '{0:%s}'%spec
        try:
            return fmt.format(str(self))
        except:
            return fmt.format(int(self))
ten = MyIntegerTenClass()
print '%d, %02X, %s' % (ten, ten, ten) # ok, prints "10, 0A, ten"
print  '{0:d}, {0:02X}, {0}'.format(ten) # ok, prints "10, 0A, ten"
4

3 回答 3

1

传递一个整数而不是一个字符串,它会正常工作。

>>> print  '{0:d}'.format(1)
1
>>> print  '{0:d}'.format('1')

Traceback (most recent call last):
  File "<pyshell#57>", line 1, in <module>
    print  '{0:d}'.format('1')
ValueError: Unknown format code 'd' for object of type 'str'
于 2012-07-12T18:40:21.560 回答
1

解决这个问题的第一种方法可能很简单try

class MyIntegerTenClass:
    def __int__(self):
        return 10
    def __str__(self):
        return 'ten'
    def __format__(self, format_spec):
        try:
            s = format(str(self), format_spec)
        except ValueError:
            s = format(int(self), format_spec)
        return s

如果MyIntegerTenClass 可以作为字符串插入到格式字符串中,那就是。如果不是,它将被转换为 int 并重新提交到format.

>>> print '{0}, {0:s}, {0:d}, {0:02X}, {0:f}'.format(ten)
ten, ten, 10, 0A, 10.000000

如果您希望默认表示为10而不是ten,则只需交换转换行:

    def __format__(self, format_spec):
        try:
            s = format(int(self), format_spec)
        except ValueError:
            s = format(str(self), format_spec)
        return s

测试输出:

>>> print '{0}, {0:s}, {0:d}, {0:02X}, {0:f}'.format(ten)
10, ten, 10, 0A, 10.000000

作为附录,我不相信你能够在不定义的情况下获得你想要的行为__format__;但是,您可以使用对象开发更复杂的方法Formatter。尽管如此,我认为基于异常的方法为您提供了许多免费的内置功能。

于 2012-07-12T19:25:00.767 回答
0

它说十是一个字符串,您正在尝试将其格式化为数字。

尝试将其包装在演员表中:

>>> print  '{0:d}, {0:02X}'.format(int(ten))
10, 0A
于 2012-07-12T18:43:35.217 回答