我正在使用参数解析器来允许将端口号传递给我的程序。然后我尝试验证该值,并且第一个测试之一是is int
测试:
parser = argparse.ArgumentParser(description='Provides XML-RPC API.')
parser.add_argument('--port', dest='port', default=9666, type=int,
help='port to listen on, 9666 by default.');
args = parser.parse_args()
if args.port is not int:
raise TypeError("Port must be integer (%s given), use -h argument to get help."%(type(args.port).__name__))
if args.port>=65536 or args.port<=0:
raise IndexError("Port must be lower than 65536, 65535 is the maximum port number. Also, port can only be positive, non-zero number. Use -h argument to get help.")
if args.port<=1204:
print("Notice: ports up to 1024 are reserved to well known services, using this port numbers often leads to conflicts. Use -h argument to get help.")
现在无论我做什么,我都会收到错误:
Port must be integer (int given), use -h argument to get help.
我的意思是,到底是什么?即使错误说它是int,所以还在进行什么?
答案 0 :(得分:6)
args.port is not int
设置为False
类型, args.port
将返回int
:
args.port = int
args.port is int # True
或者您将int
重新绑定到已被绑定的同一对象args.port
:
int = args.port
args.port is int # True
这是因为is
测试两个引用是否指向完全相同的对象。
也许您对如何测试对象类型感到困惑,因为您使用isinstance()
function或测试type()
的输出:
type(args.port) is not int # False if args.port is a different type
not isinstance(args.port, int) # False if args.port is a different type and not a subclass either
这也不会是这种情况,因为您将ArgumentParser
选项配置为仅接受整数:
parser.add_argument('--port', dest='port', default=9666, type=int,
help='port to listen on, 9666 by default.')
type=int
参数很重要,因为这意味着任何无法解释为整数的内容都将被解析器拒绝。这里永远不会超过parser.parse_args()
:
>>> import argparse
>>> parser = argparse.ArgumentParser(description='Provides XML-RPC API.')
>>> parser.add_argument('--port', dest='port', default=9666, type=int,
... help='port to listen on, 9666 by default.')
_StoreAction(option_strings=['--port'], dest='port', nargs=None, const=None, default=9666, type=<type 'int'>, choices=None, help='port to listen on, 9666 by default.', metavar=None)
>>> parser.parse_args(['--port', 'not an integer'])
usage: [-h] [--port PORT]
: error: argument --port: invalid int value: 'not an integer'
这是因为'not an integer'
功能无法转换int()
;引发ValueError
,argparse
将其转换为错误消息:
>>> int('not an integer')
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
ValueError: invalid literal for int() with base 10: 'not an integer'
您可以提供限制范围的自己的会话功能:
def portnumber(value):
port = int(value) # may raise ValueError
if not 0 <= port <= 65536:
raise ValueError('Not a valid port number')
return port
然后将其用作type
:
parser.add_argument('--port', dest='port', default=9666, type=portnumber,
help='port to listen on, 9666 by default.')
答案 1 :(得分:3)
一般情况下,int
测试某件事是isinstance(args.port, int)
。但是,这里不需要这个检查,因为argparse会保证它是一个整数。如果您没有指定默认值,那么None
也是可能的,但这不是这种情况,因为您确实指定了默认值。
答案 2 :(得分:1)
&#39; INT&#39;指的是班级本身。 &#39;不是int&#39;如果你通过了班级&#39; int&#39;本身,这不太可能发生。你可能想要&#39; isinstance(args.port,int)&#39;代替。
答案 3 :(得分:0)
is not int
将始终返回True
,除非您执行int is not int
。这是因为is
是一个身份测试,这意味着它检查两个参数是否是同一个对象。
假设你试试
>>> 1 is not int
这将返回True
,因为1
与int
不是同一个对象。我们可以更深入地检查这一点。
>>> type(1)
<class 'int'>
正如您所看到的,1
确实是int
,但int
本身实际上是一个类。
要解决此问题,您可以执行
not isinstance(args.port, int)
正如其他答案所示。
您也可以
type(args.port) is not int
检查args.port
的类型是否为int
。