11

我用 Flask 和 Python 构建了一个简单的 Web 应用程序,我打算将它上传到 Heroku。

在本地启动我的应用程序时,使用以下脚本:

#!venv/bin/python
from app import app
app.run(debug = True)

我收到此错误消息:

Traceback (most recent call last):
File "./run.py", line 2, in <module>
    from app import app, mail
File "/home/ricardo/personalSite/app/__init__.py", line 3, in <module>
    from app import index
File "/home/ricardo/personalSite/app/index.py", line 6, in <module>
    from emails import send_email
File "/home/ricardo/personalSite/app/emails.py", line 2, in <module>
    from app import app, mail
ImportError: cannot import name mail

因此,它无法导入mail.

app我有这个的目录中__init__.py,这里是我创建的Mail对象,这让我难以导入:

from flask import Flask
app = Flask(__name__)
from app import index
from flask.ext.mail import Mail
mail = Mail(app)

这是emails.py我调用send_mail函数的文件:

from flask.ext.mail import Message
from app import app, mail
from flask import render_template
from config import ADMINS
from decorators import async

因此,根据错误消息,错误在此文件中,在from app import app, mail.

问题是什么?为什么不能导入mail

更新:

这是我的目录列表:

persSite\
  venv\
    <virtual environment files>
  app\
    static\
    templates\
    __init__.py
    index.py
    emails.py
    decorators.oy
  tmp\
  run.py
4

2 回答 2

23

你有一个循环依赖。您必须了解 Python 在导入文件时在做什么。

每当 Python 导入文件时,Python 都会查看该文件之前是否已经开始导入。因此,如果模块 A 导入模块 B,而模块 B 又导入模块 A,那么 Python 将执行以下操作:

  • 开始运行模块 A。
  • 当模块 A 尝试导入模块 B 时,暂时停止运行模块 A,并开始运行模块 B。
  • 当模块 B 尝试导入模块 A 时,Python 将不会继续运行模块 A 以完成;相反,模块 B 将只能从模块 A 导入在模块 B 开始运行之前已经在那里定义的属性。

这是app/__init__.py,这是要导入的第一个文件。

from flask import Flask
app = Flask(__name__)
from app import index # <-- See note below.
from flask.ext.mail import Mail
mail = Mail(app)

导入此文件时,它只是运行脚本的 Python。创建的任何全局属性都将成为模块属性的一部分。因此,当您点击第三行时,已经定义了属性“Flask”和“app”。但是,当您点击第三行时,Python 开始尝试indexapp. 因此,它开始运行app/index.py文件。

当然,这看起来如下所示:

from flask.ext.mail import Message
from app import app, mail # <-- Error here
from flask import render_template
from config import ADMINS
from decorators import async

请记住,当这个 Python 文件被导入时,到目前为止,您只在模块中定义了Flask和。因此,尝试导入是行不通的。appappmail

因此,您需要重新排列代码,以便如果app.index依赖于 中的属性app,则app在尝试导入之前定义该属性app.index

于 2013-10-19T15:02:09.050 回答
0

这大概就是问题所在:

from app import app, mail

在文件“app/emails.py”中,导入来自当前模块,而不是嵌套的应用模块。尝试:

from . import app, mail

如果它不起作用,您应该使用更详细的目录列表更新您的问题。

于 2013-10-19T12:45:09.323 回答