2013-11-23 95 views
5

我有一個簡單的類,它擴展長時間接受值調節劑的字符串(如「10米」是1024 * 1024 * 10)爲什麼Python的通話__str__而不是返回長值

我有__str__功能該打印傳入的原始值(即,如果「10米」中傳遞,回報「10米」)

問題是,當我打電話的東西,如:

>>> printf("%d" % Size('10m')) 

我得到以下

SystemError: ../Objects/stringobject.c:4044: bad argument to internal function 

顯然,如果我打印"%s"我得到「10米」

所以現在的問題是,因爲我繼承長,爲什麼高級應召__str__當它應該得到的長期價值。

順便說一句,多一點測試表明,%x%f將打印整數值,讓我更加困惑。我也嘗試添加__format__,但似乎只在調用"...".format()時才被調用。

編輯#1,下面的代碼:

class Size(long): 
    '''Represents a size reflected bytes. Subclass of long. 
    Size passed in must be in the formats <int> or "0x<int>" or "0x<int><unit>" or "<int><unit>" or "<int><unit><int><unit>....". 
    "0x<int><unit>0x<int><unit>" or similar numbers are not supported as is "<int><unit><int>" 

    b = bytes 
    s = sectors (512-byte) 
    k = kilobytes 
    m = megabytes 
    g = gigabytes 
    t = terabytes 
    ''' 

    units = { 'b':1, 's':512, 'k':1024, 'm':1024 ** 2, 'g':1024 ** 3, 't':1024 ** 4 } 

    def __new__(cls, value): 
     '''Creates a Size object with the specified value. 

     Value can be a number or a string (optionally prefixed with '0x' or 
     postfixed with a type character). If using hex, the final character 
     will be treated as part of the value if it is a hex digit, regardless 
     of whether it is a valid unit character. 

     Examples: 
      Size(50) 
      Size("0x100s") # 256 sectors 
      Size("64") 
      Size("512k") 
      Size("0x1b") # this is 1b bytes, not 1 byte 
     ''' 
     self = _new_unit_number(value, cls.units, long, cls) 
     return self 

    def __init__(self, value): 
     self._orig_value = value 

    def __str__(self): 
     print "calling str" 
     return str(self._orig_value) # Convert to str in case the object was created w/an int 

    def __format__(self, format_spec): 
     print "calling format" 
     print format_spec 
     try: 
      value = format(str(self), format_spec) 
     except ValueError: 
      value = format(int(self), format_spec) 
     return value 

def _new_unit_number(value, unit_list, num_type, cls): 
    '''Converts a string of numbers followed by a unit character to the 
    requested numeric type (int or long for example). 
    ''' 
    base = 10 
    start = 0 
    digits = string.digits 
    try: 
     if value[0:2] == '0x': 
      start = 2 
      base = 16 
      digits = string.hexdigits 

     if value[-1] in digits: 
      return num_type.__new__(cls, value[start:], base) 
     else: 
      try: 
       # Use a regex to split the parts of the unit 
       regex_string = '(\d+[%s])' % (''.join(unit_list.keys())) 
       parts = [x for x in re.split(regex_string, value[start:]) if x] 

       if len(parts) == 1: 
        return num_type.__new__(cls, num_type(value[start:-1], base) * unit_list[value[-1]]) 
       else: 
        # Total up each part 
        # There's probably a better way to do this. 
        # This converts each unit to its base type, stores it in total, 
        # only to be converted back to the base type. 
        total = 0 
        for part in parts: 
         total += num_type(part[start:-1], base) * unit_list[part[-1]] 

        # Finally return the requested unit 
        return num_type.__new__(cls, total) 
      except KeyError: 
       raise ValueError("Invalid %s unit identifier: %s" 
        % (cls.__name__, unit_list[value[-1]])) 

    # not a string or empty, see if we can still use the class's constructor 
    except (TypeError, IndexError): 
     return num_type.__new__(cls, value) 
+5

「尺寸」是如何定義的? – Hyperboreus

+2

請注意,「m」是米的國際符號,或10^-3的前綴。如果你想10^6,使用M.如果你想要2^20,使用Mi。 –

+1

你從Python 2.7中獲得'printf()'函數的位置? –

回答

2

不是一個真正的答案,但過長的註釋。

我覺得這個問題非常有趣。我試圖複製使用此行爲:

#! /usr/bin/python2.7 

class Size (long): 
    def __new__ (cls, arg): 
     if arg and type (arg) == str: 
      if arg [-1] == 'm': 
       return super (Size, cls).__new__ (cls, long (arg [:-1]) * 2 ** 20) 
     return super (Size, cls).__new__ (cls, arg) 

    def __init__ (self, arg): 
     self.s = arg 

    def __str__ (self): 
     return self.s 

a = Size ('12m') 
print (a) 
print ('%s' % a) 
#The following fails horribly 
print ('%d' % a) 

行爲描述由OP。但現在來的有趣的部分:當我從int,而不是從長繼承,工作平穩:

class Size (int): 
    def __new__ (cls, arg): 
     if arg and type (arg) == str: 
      if arg [-1] == 'm': 
       return super (Size, cls).__new__ (cls, int (arg [:-1]) * 2 ** 20) 
     return super (Size, cls).__new__ (cls, arg) 

    def __init__ (self, arg): 
     self.s = arg 

    def __str__ (self): 
     return self.s 

也就是說,它在python2工作正常,但在python3失敗。奇怪,奇怪。

+1

你基本上有我寫的類(我只支持更多的修飾符像K,m,g等),那就是確切的問題。有趣的是,它失敗了很長時間,但不是int ...我需要很長的時間,因爲我們超越了int支持的32位。 – JasonAUnrein

2

請參閱Python的問題跟蹤,Issue 18780: SystemError when formatting int subclass

>>> class I(int): 
... def __str__(self): 
...  return 'spam' 
... 
>>> '%d' % I(42) 
Traceback (most recent call last): 
    File "<stdin>", line 1, in <module> 
SystemError: Objects/unicodeobject.c:13305: bad argument to internal function 

這個工作在3.4.0alpha4,但不是在3 [0123]。

+0

注意:該代碼適用於Python 2.7(用於'int'子類)。它打破了「長」的子類。 – jfs

相關問題