检查 'except ImportError' 中的模块名称
Checking module name inside 'except ImportError'
try:
import MySQLdb
# some action
except ImportError as err:
# fallback code
PyCharm 给出代码检查警告:
'MySQLdb' in try block with 'except ImportError' should also be defined in except block
This inspection detects names that should resolve but don't. Due to dynamic dispatch and duck typing, this is possible in a limited but useful number of cases. Top-level and class-level items are supported better than instance items.
好的,我认为这个警告是合理的,因为 fallback code
假设 'MySQLdb' 没有安装,而它可能是一些不同的错误,刚刚引发了 ImportError。所以我使用了类似的东西:
try:
import MySQLdb
# some action
except ImportError as err:
if "MySQLdb" in repr(err):
# fallback code
else:
raise
PyCharm 警报仍然存在,但这可能只是一个 PyCharm 问题(google 显示此类检查存在问题)
问题:
当你“except ImportError”时真的值得检查名称吗?即使在简单的情况下(import MySQLdb
之后没有 some action
)?
如果值得检查,上面的例子是正确的方法吗?如果不是 - 正确的方法是什么?
P.S. MySQLdb 只是系统中可能不存在的模块的一个示例。
在Python 3.3+ 中,ImportError
具有属性name
,它告诉导入失败的模块的名称。那么当然 MySQLdb 会暗示你被困在 Python 2.
我认为您误解了警告,如果您没有在 except 块中定义一个名为 MySQLdb
的变量,那么稍后当您尝试使用该模块时,您会得到一个 NameError
:
try:
import foo
except ImportError:
pass
foo.say_foo() #foo may or may not be defined at this point!
如果模块仅在 try:
子句中使用,则这不是问题。但对于更一般的情况,检查器希望您在 except 块中定义变量:
try:
import foo
except ImportError:
foo = None #now foo always exists
if foo: #if the module is present
foo.say_foo()
else:
print("foo") #backup use
如果该模块仅在 try 块中使用,那么您可以通过将其从命名空间中删除来向检查员(和您自己)表明您以后不能使用该模块:
try:
import foo
except ImportError:
pass
else:
# if it was able to import use it then get rid of the local variable
foo.do_thing()
del foo #optional module should not be relied on outside
# now foo never exists here, checker is happy.
试试这个:
try:
import libname
except ImportError as e:
e = e[0][16:]
print("\n[!] Error: Please Install Module Name:[ {} ] And Try Again !!!".format(e))
exit(1)
输出:
[!] Error: Please Install Module Name:[ libname ] And Try Again !!!
已经提到的一种获取名称的方法是使用 name
属性:
try:
import pty
except ImportError as e:
print(e.name)
但是,请务必注意,您可能无法获得您开始使用的模块名称;即,运行 Windows 上面的 termios
作为输出,而不是 pty
.
为了保持与 python 2.7 的兼容性(对于那些还没有进行转换的人,您在 clock)
try:
import pty
except ImportError as e:
print(e.args[0].rsplit(' ',1)[-1]) # `msg` attribute is `message` on python2...stick with args
# prints the same result: termios
一些奖金花絮:
Python 3.6 于 2016 年底左右发布。由于是 2019+,您可以选择更清晰的例外 ModuleNotFoundError
:
A subclass of ImportError which is raised by import when a module could not be located. It is also raised when None is found in sys.modules.
如果您使用的是 Python 3.3+,请不要忘记 path
属性(除了已经提到的 name
之外),它提供引发异常的任何文件的路径(仅在更复杂的情况下有用——在我给出的简单示例中 path
将 return None
.)
try:
import MySQLdb
# some action
except ImportError as err:
# fallback code
PyCharm 给出代码检查警告:
'MySQLdb' in try block with 'except ImportError' should also be defined in except block
This inspection detects names that should resolve but don't. Due to dynamic dispatch and duck typing, this is possible in a limited but useful number of cases. Top-level and class-level items are supported better than instance items.
好的,我认为这个警告是合理的,因为 fallback code
假设 'MySQLdb' 没有安装,而它可能是一些不同的错误,刚刚引发了 ImportError。所以我使用了类似的东西:
try:
import MySQLdb
# some action
except ImportError as err:
if "MySQLdb" in repr(err):
# fallback code
else:
raise
PyCharm 警报仍然存在,但这可能只是一个 PyCharm 问题(google 显示此类检查存在问题)
问题:
当你“except ImportError”时真的值得检查名称吗?即使在简单的情况下(
import MySQLdb
之后没有some action
)?如果值得检查,上面的例子是正确的方法吗?如果不是 - 正确的方法是什么?
P.S. MySQLdb 只是系统中可能不存在的模块的一个示例。
在Python 3.3+ 中,ImportError
具有属性name
,它告诉导入失败的模块的名称。那么当然 MySQLdb 会暗示你被困在 Python 2.
我认为您误解了警告,如果您没有在 except 块中定义一个名为 MySQLdb
的变量,那么稍后当您尝试使用该模块时,您会得到一个 NameError
:
try:
import foo
except ImportError:
pass
foo.say_foo() #foo may or may not be defined at this point!
如果模块仅在 try:
子句中使用,则这不是问题。但对于更一般的情况,检查器希望您在 except 块中定义变量:
try:
import foo
except ImportError:
foo = None #now foo always exists
if foo: #if the module is present
foo.say_foo()
else:
print("foo") #backup use
如果该模块仅在 try 块中使用,那么您可以通过将其从命名空间中删除来向检查员(和您自己)表明您以后不能使用该模块:
try:
import foo
except ImportError:
pass
else:
# if it was able to import use it then get rid of the local variable
foo.do_thing()
del foo #optional module should not be relied on outside
# now foo never exists here, checker is happy.
试试这个:
try:
import libname
except ImportError as e:
e = e[0][16:]
print("\n[!] Error: Please Install Module Name:[ {} ] And Try Again !!!".format(e))
exit(1)
输出:
[!] Error: Please Install Module Name:[ libname ] And Try Again !!!
已经提到的一种获取名称的方法是使用 name
属性:
try:
import pty
except ImportError as e:
print(e.name)
但是,请务必注意,您可能无法获得您开始使用的模块名称;即,运行 Windows 上面的 termios
作为输出,而不是 pty
.
为了保持与 python 2.7 的兼容性(对于那些还没有进行转换的人,您在 clock)
try:
import pty
except ImportError as e:
print(e.args[0].rsplit(' ',1)[-1]) # `msg` attribute is `message` on python2...stick with args
# prints the same result: termios
一些奖金花絮:
Python 3.6 于 2016 年底左右发布。由于是 2019+,您可以选择更清晰的例外
ModuleNotFoundError
:A subclass of ImportError which is raised by import when a module could not be located. It is also raised when None is found in sys.modules.
如果您使用的是 Python 3.3+,请不要忘记
path
属性(除了已经提到的name
之外),它提供引发异常的任何文件的路径(仅在更复杂的情况下有用——在我给出的简单示例中path
将 returnNone
.)