与通过IDLE' run module f5
命令运行相比,通过命令行运行时代码是否会引发错误?
最近,我一直在尝试提高代码的可读性和健壮性。因此,我一直试图删除所有from module import *
行。我以前使用from tkinter import *
,我的代码行完全正常:
self.path = filedialog.askdirectory()
但现在我已将from tkinter import *
更改为import tkinter as tk
,我已相应更改了代码:
self.path = tk.filedialog.askdirectory()
名为GUI.py的文件使用以下内容导入此文件:from lib.filesearch import *
(我提到的代码行位于filesearch文件中。)
我通过IDLE运行我的代码,一切都很好。我的GUI仍然有效,行self.path = tk.filedialog.askdirectory()
正常工作,但是,当我通过Windows命令行运行代码时,我收到错误:
AttributeError: 'module' object has no attribute 'filedialog'
以下是我的代码中的相关位:
来自filesearch.py
import tkinter as tk
def get_path(self):
"""Store user chosen path to search"""
self.paths = tk.filedialog.askdirectory(initialdir = FileSearch.DEFAULT)
return self.paths
来自GUI.py
from lib.filesearch import *
def Browse(self):
self.BrowseB['state']='disabled'
self.p=self.CrawlObj.get_path()
self.AddText('Searching from Path: ' + str(self.p))
self.BrowseB['state']='normal'
与此question不同,我只安装了一个版本的python。即,Python34。
答案 0 :(得分:10)
我想先说:如果您知道将使用子模块,请始终明确导入子模块。
由于tkinter
的结构,您必须明确导入要加载的子模块:
import tkinter as tk
print(hasattr(tk,"filedialog")) # in a standard interpreter will print false
import tkinter.filedialog
print(hasattr(tk,"filedialog")) # should always print true after explicit import
你不需要在IDLE中执行此操作的原因是,在运行代码之前,IDLE会在后台设置一些内容并最终导入一些tkinter库。其中一个维护者has commented,这实际上是IDLE中的一个错误。
在python 3.6.5中(可能更早,只检查过这个版本)这个特定的差异已得到修复,在任何版本中你都可以看到加载了一些代码的子模块列表:
# standard interpreter
Python 3.6.5 (v3.6.5:f59c0932b4, Mar 28 2018, 03:03:55)
[GCC 4.2.1 (Apple Inc. build 5666) (dot 3)] on darwin
Type "help", "copyright", "credits" or "license" for more information.
>>> import sys
>>> len(sys.modules) #total number of modules automatically loaded
71
>>> sorted(name for name in sys.modules.keys() if ("." in name)) #submodules loaded
['collections.abc', 'encodings.aliases', 'encodings.latin_1', 'encodings.utf_8', 'importlib._bootstrap', 'importlib._bootstrap_external', 'importlib.abc', 'importlib.machinery', 'importlib.util', 'os.path']
>>> len(_) #number of submodules
10
在IDLE:
Python 3.6.5 (v3.6.5:f59c0932b4, Mar 28 2018, 03:03:55)
[GCC 4.2.1 (Apple Inc. build 5666) (dot 3)] on darwin
Type "copyright", "credits" or "license()" for more information.
>>> import sys
>>> len(sys.modules)
152
>>> sorted(name for name in sys.modules.keys() if ("." in name and "idlelib" not in name))
['collections.abc', 'encodings.aliases', 'encodings.ascii', 'encodings.latin_1', 'encodings.utf_8', 'importlib._bootstrap', 'importlib._bootstrap_external', 'importlib.abc', 'importlib.machinery', 'importlib.util', 'os.path', 'tkinter.constants', 'urllib.parse']
>>> len(_) #number of submodules not directly related to idlelib.
13
只有tkinter.constants
才会加载 import tkinter
,因此,就我测试的版本而言,此问题仅适用于urllib.parse
和encodings.ascii
(以及idlelib
模块但通常生产代码不使用那个)
这不一定是IDLE特定的问题,更糟糕的问题是如果子模块是由您使用的另一个库加载的。以下面的代码为例:
>>> import pandas
>>> import http
>>> http.client
<module 'http.client' from '.../http/client.py'>
现在假设我们写了一些仍使用http.client
但没有使用pandas的其他代码:
>>> import http
>>> http.client
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
AttributeError: module 'http' has no attribute 'client'
通过这种方式,当使用它的代码可能通过使用碰巧使用它的库加载http.client
时,子模块可以正常工作,否则会失败。
这让我回到了我的初始点 - 始终明确导入子模块。
答案 1 :(得分:4)
实际上模块没有属性filedialog
,它是一个子模块,你应该在使用之前将其导入为import tkinter.filedialog
。您可以使用tk.filedialog
而无需在IDLE中明确导入filedialog
,因为它已经导入。
import sys
sys.modules['tkinter.filedialog']
上面的代码会在标准的python解释器中引发KeyError
,但它会在IDLE中返回类似<module 'tkinter.filedialog' from '/usr/lib/python3.5/tkinter/filedialog.py'>
的内容。