Flask 循环依赖

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)

leo*_*ovp 7

我提出以下结构:

# 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)

在这种情况下,databaseapp和 您的模型都在单独的模块中,并且没有冲突或循环导入。


Mik*_*ise 7

我追了几个小时,在这里登陆了几次,结果发现我在@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 用户来说是一个常见的错误,他们很可能会在这里遇到。在过去的几年里,我至少击中过两次。