在 Python 中自动类型转换参数
背景:
我主要从管道中的命令行运行 python 脚本,因此我的参数始终是需要类型转换为适当类型的字符串。我每天都会制作很多小脚本,并且为每个脚本对每个参数进行类型转换会花费比应有的时间更多的时间。
问题:
是否有一种规范的方法可以自动为函数键入强制类型转换参数?
我的方式:
如果没有更好的方法,我开发了一个装饰器来做我想做的事情。装饰器是下面的 autocast fxn。示例中,修饰后的 fxn 为 fxn2。请注意,在代码块的末尾,我将 1 和 2 作为字符串传递,如果您运行脚本,它会自动添加它们。这是一个好方法吗?
def estimateType(var):
#first test bools
if var == 'True':
return True
elif var == 'False':
return False
else:
#int
try:
return int(var)
except ValueError:
pass
#float
try:
return float(var)
except ValueError:
pass
#string
try:
return str(var)
except ValueError:
raise NameError('Something Messed Up Autocasting var %s (%s)'
% (var, type(var)))
def autocast(dFxn):
'''Still need to figure out if you pass a variable with kw args!!!
I guess I can just pass the dictionary to the fxn **args?'''
def wrapped(*c, **d):
print c, d
t = [estimateType(x) for x in c]
return dFxn(*t)
return wrapped
@autocast
def fxn2(one, two):
print one + two
fxn2('1', '2')
编辑:对于任何来到这里并想要更新且简洁的工作版本的人,请转到此处:
https: //github.com/sequenceGeek/cgAutoCast
这里也是基于上述的快速工作版本:
def boolify(s):
if s == 'True' or s == 'true':
return True
if s == 'False' or s == 'false':
return False
raise ValueError('Not Boolean Value!')
def estimateType(var):
'''guesses the str representation of the variables type'''
var = str(var) #important if the parameters aren't strings...
for caster in (boolify, int, float):
try:
return caster(var)
except ValueError:
pass
return var
def autocast(dFxn):
def wrapped(*c, **d):
cp = [estimateType(x) for x in c]
dp = dict( (i, estimateType(j)) for (i,j) in d.items())
return dFxn(*cp, **dp)
return wrapped
######usage######
@autocast
def randomFunction(firstVar, secondVar):
print firstVar + secondVar
randomFunction('1', '2')
Background:
I mostly run python scripts from the command line in pipelines and so my arguments are always strings that need to be type casted to the appropriate type. I make a lot of little scripts each day and type casting each parameter for every script takes more time than it should.
Question:
Is there a canonical way to automatically type cast parameters for a function?
My Way:
I've developed a decorator to do what I want if there isn't a better way. The decorator is the autocast fxn below. The decorated fxn is fxn2 in the example. Note that at the end of the code block I passed 1 and 2 as strings and if you run the script it will automatically add them. Is this a good way to do this?
def estimateType(var):
#first test bools
if var == 'True':
return True
elif var == 'False':
return False
else:
#int
try:
return int(var)
except ValueError:
pass
#float
try:
return float(var)
except ValueError:
pass
#string
try:
return str(var)
except ValueError:
raise NameError('Something Messed Up Autocasting var %s (%s)'
% (var, type(var)))
def autocast(dFxn):
'''Still need to figure out if you pass a variable with kw args!!!
I guess I can just pass the dictionary to the fxn **args?'''
def wrapped(*c, **d):
print c, d
t = [estimateType(x) for x in c]
return dFxn(*t)
return wrapped
@autocast
def fxn2(one, two):
print one + two
fxn2('1', '2')
EDIT: For anyone that comes here and wants the updated and concise working version go here:
https://github.com/sequenceGeek/cgAutoCast
And here is also quick working version based on above:
def boolify(s):
if s == 'True' or s == 'true':
return True
if s == 'False' or s == 'false':
return False
raise ValueError('Not Boolean Value!')
def estimateType(var):
'''guesses the str representation of the variables type'''
var = str(var) #important if the parameters aren't strings...
for caster in (boolify, int, float):
try:
return caster(var)
except ValueError:
pass
return var
def autocast(dFxn):
def wrapped(*c, **d):
cp = [estimateType(x) for x in c]
dp = dict( (i, estimateType(j)) for (i,j) in d.items())
return dFxn(*cp, **dp)
return wrapped
######usage######
@autocast
def randomFunction(firstVar, secondVar):
print firstVar + secondVar
randomFunction('1', '2')
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(6)
如果您想自动转换值:
如果您愿意,可以调整
boolify
以接受其他布尔值。If you want to auto-convert values:
You can adjust
boolify
to accept other boolean values if you like.如果您信任源,您可以只使用纯 eval 来输入字符串:
但是如果您不信任源,您可以使用 ast 模块中的literal_eval。
编辑:
正如 Ned Batchelder 的评论,它不接受非引号字符串,所以我添加了一个解决方法,也是一个关于带有关键字参数的 autocaste 装饰器的示例。
You could just use plain eval to input string if you trust the source:
But if you don't trust the source, you could use literal_eval from ast module.
Edit:
As Ned Batchelder's comment, it won't accept non-quoted strings, so I added a workaround, also an example about autocaste decorator with keyword arguments.
我想你可以用函数装饰器创建一个类型签名系统,就像你所拥有的那样,只有一个需要参数的系统。例如:
这样的装饰器可以相当容易地构建。像这样的东西(编辑 - 有效!):
就像这样,您现在可以为函数注入类型签名!
基本上,这是 @utdemir 方法的类型显式版本。
I'd imagine you can make a type signature system with a function decorator, much like you have, only one that takes arguments. For example:
Such a decorator can be built rather easily. Something like this (EDIT -- works!):
And just like that, you can now imbue functions with type signatures!
Basically, this is an type-explicit version of @utdemir's method.
如果您要从命令行解析参数,则应使用
argparse
模块(如果您使用的是 Python 2.7)。每个参数都可以有一个预期的类型,因此知道如何处理它应该相对简单。您甚至可以定义自己的类型。
If you're parsing arguments from the command line, you should use the
argparse
module (if you're using Python 2.7).Each argument can have an expected type so knowing what to do with it should be relatively straightforward. You can even define your own types.
您的代码片段中有几个问题。
这将始终检查
True
,因为您正在针对字符串'True'
和'False'
进行测试。python 中没有自动类型强制转换。当您通过
*args
和**kwargs
接收时,您的参数可以是任何内容。首先将查找值列表(每个值可以是任何数据类型,原始数据类型和复杂数据类型),其次将查找映射(可能具有任何有效的映射)。因此,如果您编写一个装饰器,您最终将得到一个很好的错误检查列表。通常,如果您希望以 str 形式发送,则在调用函数时,通过 (
str
) 将其类型转换为字符串并发送。There are couple of problems in your snippet.
This would always check for
True
because you are testing against the strings'True'
and'False'
.There is not an automatic coercion of types in python. Your arguments when you receive via
*args
and**kwargs
can be anything. First will look for list of values (each of which can be any datatype, primitive and complex) and second will look for a mapping (with any valid mapping possible). So if you write a decorator, you will end up with a good list of error checks.Normally, if you wish to send in str, just when the function is invoked, typecast it to string via (
str
) and send it.我知道我在这场比赛中迟到了,但是评估怎么样?
或者(更安全):
I know I arrived late at this game, but how about eval?
or alternatively (and more safely):