在Python中,`不是int`是否会返回false?

时间:2015-09-07 21:07:00

标签: python argparse

我正在使用参数解析器来允许将端口号传递给我的程序。然后我尝试验证该值,并且第一个测试之一是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,所以还在进行什么?

4 个答案:

答案 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();引发ValueErrorargparse将其转换为错误消息:

>>> 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,因为1int不是同一个对象。我们可以更深入地检查这一点。

>>> type(1)
<class 'int'>

正如您所看到的,1确实是int,但int本身实际上是一个类。

要解决此问题,您可以执行

not isinstance(args.port, int)

正如其他答案所示。

您也可以

type(args.port) is not int

检查args.port的类型是否为int