我想将0x123
,012
或36#foo
转换为整数。
为此,我写了这个:
def str2int(s):
if re.findall('(?i)^0x',s):
return int(s, 16)
if re.findall('^(?i)0',s):
return int(s, 8)
m = re.findall('(?i)^(\d+)\#([0-9a-z]+)',s)
if m:
return int(m[0][1], m[0][0])
raise AssertionError, 'Unknown value'
我觉得它有点复杂。有内置方法吗?
答案 0 :(得分:2)
是的,您可以使用ast.literal_eval
。
>>> import ast
>>> ast.literal_eval("0x123")
291
>>> ast.literal_eval("012")
10
>>> ast.literal_eval("36#foo")
36
但请注意,literal_eval("012")
仅适用于2.7及更低版本,因为3.x不再支持该八进制文字样式。但这会奏效:
>>> ast.literal_eval("0o12")
10
答案 1 :(得分:2)
没有正则表达式的解决方案:
def convert (s):
if s.lower().startswith('0x'):
s = '16#' + s[2:]
elif s.startswith('0'):
s = '8#' + s[1:]
elif '#' not in s:
s = '10#' + s
base, num = s.split('#', 1)
return int(num, int(base))
>>> testcases = [('0x123', 291), ('012', 10), ('36#foo', 20328)]
>>> for s, n in testcases:
print(s, n, n == convert(s))
0x123 291 True
012 10 True
36#foo 20328 True
答案 2 :(得分:1)
int
作为第二个参数传递时, 0
会这样做:
int('0x123', 0)
=> 291
int('0o12', 0)
=> 10
如果您想支持评论,str.partition
是我能想到的最简单的方法:
int('36#foo'.partition('#')[0], 0)
=> 36