pav*_*163 3 python circular-dependency flask
我正在开发 Flask 应用程序。它仍然相对较小。我只有一个 app.py 文件,但因为我需要进行数据库迁移,我使用本指南将其分为 3 个:
https://realpython.com/blog/python/flask-by-example-part-2-postgres-sqlalchemy-and-alembic/
但是,我现在无法运行我的应用程序,因为应用程序和模型之间存在循环依赖关系。
应用程序.py:
from flask import Flask
from flask_sqlalchemy import SQLAlchemy
from flask import render_template, request, redirect, url_for
import os
app = Flask(__name__)
app.config['SQLALCHEMY_DATABASE_URI'] = os.environ['DB_URL']
app.config['SQLALCHEMY_TRACK_MODIFICATIONS'] = False
app.debug = True
db = SQLAlchemy(app)
from models import User
... routes ...
if __name__ == "__main__":
app.run()
Run Code Online (Sandbox Code Playgroud)
模型.py:
from app import db
class User(db.Model):
id = db.Column(db.Integer, primary_key=True)
username = db.Column(db.String(80), unique=True)
email = db.Column(db.String(120), unique=True)
def __init__(self, username, email):
self.username = username
self.email = email
def __repr__(self):
return self.username
Run Code Online (Sandbox Code Playgroud)
管理.py:
from flask_script import Manager
from flask_migrate import Migrate, MigrateCommand
from app import app, db
migrate = Migrate(app, db)
manager = Manager(app)
manager.add_command('db', MigrateCommand)
if __name__ == "__main__":
manager.run()
Run Code Online (Sandbox Code Playgroud)
它们都在同一个目录中。当我尝试运行python app.py以启动服务器时,我收到一个错误,它肯定显示了循环依赖(非常明显)。我在遵循指南时是否犯了任何错误,还是指南有误?我怎样才能重构这是正确的?
非常感谢。
编辑:追溯
Traceback (most recent call last):
File "app.py", line 14, in <module>
from models import User
File "/../models.py", line 1, in <module>
from app import db
File "/../app.py", line 14, in <module>
from models import User
ImportError: cannot import name User
Run Code Online (Sandbox Code Playgroud)
我提出以下结构:
# app/extensions.py
from flask_sqlalchemy import SQLAlchemy
db = SQLAlchemy()
...
# app/app.py
from app.extensions import db
def create_app(config_object=ProdConfig):
app = Flask(__name__.split('.')[0])
app.config.from_object(config_object)
register_extensions(app)
...
def register_extensions(app):
db.init_app(app)
...
# manage.py
from yourapp.app import create_app
app = create_app()
app.debug = True
...
Run Code Online (Sandbox Code Playgroud)
在这种情况下,database、app和 您的模型都在单独的模块中,并且没有冲突或循环导入。
我追了几个小时,在这里登陆了几次,结果发现我在@app.route创建应用程序的行之前导入了我的页面模块(包含命令的模块)。这很容易做到,因为导入命令往往放在最开始的地方,但在这种情况下不起作用。
所以这:
# app/__init__.py
print("starting __init__.py")
from flask import Flask
from flask import render_template
import matplotlib.pyplot as plt
import numpy as np
import mpld3
app = Flask(__name__, instance_relative_config=True)
app.config.from_object('config')
from . import index
from . import simple
app.run(threaded=False)
print("finished __init__.py")
Run Code Online (Sandbox Code Playgroud)
而不是将所有导入放在上面。
将其放在这里是因为这对于普通 Flask 用户来说是一个常见的错误,他们很可能会在这里遇到。在过去的几年里,我至少击中过两次。