sqlalchemy mysql连接没有关闭烧瓶API

时间:2015-03-14 06:51:44

标签: python mysql flask sqlalchemy database-connection

我有一个用烧瓶写的api。它使用sqlalchemy来处理mysql数据库。我不使用flask-sqlalchemy,因为我不喜欢模块强制你进入某种模式来声明模型的方式。

我遇到了数据库连接没有关闭的问题。表示连接的对象超出了范围,因此我假设它正在被垃圾收集。我还明确地在会话上调用close()。尽管存在这些事实,但在api呼叫返回其响应之后,连接仍然保持打开状态。

sqlsession.py:这是我用于会话的包装器。

class SqlSession:
    def __init__(self, conn=Constants.Sql):
        self.db = SqlSession.createEngine(conn)

        Session = sessionmaker(bind=self.db)
        self.session = Session()
    @staticmethod
    def createEngine(conn):
        return create_engine(conn.URI.format(user=conn.USER, password=conn.PASS, host=conn.HOST, port=conn.PORT, database=conn.DATABASE, poolclass=NullPool))

    def close(self):
        self.session.close()

flaskroutes.py:这是一个烧瓶应用实例化并使用包装器对象的示例。请注意,它在api调用范围内的开头实例化它,然后在结束时关闭会话,并且可能在返回响应后进行垃圾收集。

def commands(self, deviceId):
    sqlSession = SqlSession(self.sessionType) <---

    commandsQueued = getCommands()
    jsonCommands = []
    for command in commandsQueued:
     jsonCommand = command.returnJsonObject()
     jsonCommands.append(jsonCommand)
     sqlSession.session.delete(command)
    sqlSession.session.commit()
    resp = jsonify({'commands': jsonCommands})
    sqlSession.close() <---  
    resp.status_code = 200
    return resp

我希望在建立http响应后立即清除连接,而是连接以“SLEEP”状态结束(在mysql命令行界面'show processlist'中查看时)。

1 个答案:

答案 0 :(得分:3)

我最终使用了这篇SO帖子的建议: How to close sqlalchemy connection in MySQL

我强烈建议您向有此问题的人阅读该帖子。基本上,我添加了一个dispose()调用close方法。这样做会导致整个连接被破坏,而关闭只会返回到可用池的连接(但保持打开状态)。

def close(self):
    self.session.close()
    self.db.dispose()

这对我来说有点混乱,但至少现在我对连接池了解得更多。