目錄
SQLAlchemy數據庫連接的正確關閉方法及問題排查
首頁 後端開發 Python教學 為什麼Sqlalchemy數據庫連接無法正確關閉?如何解決這個問題?

為什麼Sqlalchemy數據庫連接無法正確關閉?如何解決這個問題?

Apr 01, 2025 pm 04:57 PM
mysql python ai 解決方法 為什麼

為什麼Sqlalchemy數據庫連接無法正確關閉?如何解決這個問題?

SQLAlchemy數據庫連接的正確關閉方法及問題排查

在使用Python的SQLAlchemy庫進行數據庫操作時,確保數據庫連接的正確關閉至關重要,以避免資源洩漏和性能問題。本文將分析一個常見的SQLAlchemy連接關閉問題,並提供解決方案。

以下代碼片段展示了一個可能存在連接關閉問題的示例:

 from sqlalchemy import create_engine, url, delete, update, select, exists
from sqlalchemy.orm import sessionmaker, scoped_session
from core.database.base import base # 假設這是你的數據庫基類from lib.type import type # 假設這是你的類型定義from typing import Any
from flask import g, current_app

import importlib
import re


class Database: # 類名改為首字母大寫,符合Python規範env = None

    def set(self, key: str, value: Any):
        """
        設置屬性值,根據環境變量設置到g.application或g.platform
        """
        if self.env == "application":
            g.application = self.container._replace(**{key: value})
        elif self.env == 'platform':
            g.platform = self.container._replace(**{key: value})

    @property
    def container(self):
        """
        返回g.application或g.platform容器"""
        if self.env == "application":
            if "application" not in g:
                g.application = type.application(None, None, None)
            return g.application
        elif self.env == 'platform':
            if "platform" not in g:
                g.platform = type.platform(None, None)
            return g.platform

    @property
    def database_conf(self):
        """
        獲取數據庫配置"""
        return base.setting(current_app.config["database"])

    @property
    def __database_core(self):
        """
        創建數據庫會話,並緩存到實例屬性"""
        if not hasattr(self, '_database_core'):
            self._database_core = self.__create_session(**self.database_conf)
        return self._database_core

    @property
    def __create_engine(self):
        """
        獲取數據庫引擎,並緩存到實例屬性"""
        return self.__database_core.engine

    @property
    def __create_database(self):
        """
        獲取數據庫會話,並緩存到實例屬性"""
        return self.__database_core.session

    def __create_session(self, **config):
        """
        創建數據庫會話"""
        engine = self.create_engine(**config)
        session = scoped_session(sessionmaker(bind=engine, autoflush=True))
        return type.database(engine=engine, session=session())

    @classmethod
    def create_engine(cls, **kwargs):
        """
        創建數據庫引擎"""
        return create_engine(url.create("mysql pymysql", **kwargs), echo=True, isolation_level="autocommit")

    @staticmethod
    def create_all(models: list, engine=None):
        """
        創建所有模型的表"""
        tables = [Database.get_model(model).__table__ for model in models]
        base.metadata.create_all(bind=engine, tables=tables)

    def create_table(self, tables: list):
        """
        創建指定模型的表"""
        Database.create_all(models=tables, engine=self.__create_engine)

    @staticmethod
    def get_model(model: str):
        """
        獲取模型對象"""
        module = importlib.import_module(f"model.{model.split('_')[0]}.{model}")
        class_name = ''.join(re.findall(r"[a-za-z] ", model.split(".")[-1].title()))
        return getattr(module, class_name)()

    @property
    def database(self):
        """
        獲取數據庫會話"""
        return self.__create_database

    def table_data_query_all(self, model: Any, condition: list = None, order: list = None, limit: int = 500,
                             fields: list = None) -> list[dict]:
        """
        查詢所有數據"""
        query = select(model)
        if fields:
            query = query.with_only_columns(*fields)
        if condition:
            query = query.filter(*condition)
        if order:
            query = query.order_by(*order)
        results = [row.dict() for row in self.database.execute(query.limit(limit)).scalars()]
        return results

    def table_data_query_one(self, model: Any, condition: list = None) -> dict:
        """
        查詢單條數據"""
        result = self.database.execute(select(model).filter(*condition).limit(1)).scalar_one_or_none()
        return None if result is None else result.dict()

    def table_data_query_exists(self, condition: list) -> bool:
        """
        查詢數據是否存在"""
        return self.database.query(exists().where(*condition)).scalar()

    def table_data_insert_all(self, models: list) -> None:
        """
        批量插入數據"""
        with self.database as db:
            db.add_all(models)
            db.commit()

    def table_data_insert_one(self, model, data: bool = False) -> int | dict:
        """
        插入單條數據"""
        with self.database as db:
            db.add(model)
            db.commit()
            return model.dict() if data else model.id

    def table_data_update(self, model: Any, condition: list, data: dict) -> None:
        """
        更新數據"""
        with self.database as db:
            db.execute(update(model).where(*condition).values(**data))
            db.commit() # 需要顯式提交def table_data_delete(self, model: Any, condition: list) -> None:
        """
        刪除數據"""
        with self.database as db:
            db.execute(delete(model).where(*condition))
            db.commit() # 需要顯式提交def close(self):
        """
        關閉數據庫連接"""
        if hasattr(self, '_database_core'):
            self._database_core.session.close()
            self._database_core.engine.dispose()
            del self._database_core

    def __del__(self):
        """
        析構函數,確保連接關閉"""
        self.close()
登入後複製

改進說明:

  1. 類名規範:database改為Database ,符合Python命名規範。
  2. 屬性緩存:使用@property和實例屬性緩存_database_core ,避免重複創建會話。
  3. 顯式提交:table_data_updatetable_data_delete中添加了db.commit() ,確保事務提交。
  4. 資源釋放: close()方法中顯式調用session.close()engine.dispose()來釋放資源。 del self._database_core刪除緩存的會話對象。
  5. 異常處理:可以考慮添加try...except塊來處理潛在的異常,例如數據庫連接錯誤。
  6. scoped_session的使用: scoped_session在Flask 應用中通常配合g對象使用,確保每個請求使用獨立的會話,並在請求結束時自動關閉。 但代碼中沒有體現Flask請求上下文管理,因此dispose()是必要的。如果使用Flask的上下文管理, dispose()可能不是必需的,但session.close()仍然是必要的。

解決方法:

主要問題在於scoped_session的使用和資源釋放的時機。 scoped_session本身並不保證連接的自動關閉,它只是管理會話的範圍。 self.database.get_bind().dispose()在某些情況下可能無效,因為它可能無法正確地關閉底層的數據庫連接。

因此,需要在合適的地方調用close()方法,或者在類的析構函數__del__中調用close()方法,確保連接被正確關閉。 但是,依賴__del__並非最佳實踐,因為Python 的垃圾回收機制不可預測。推薦在使用完Database實例後,顯式調用instance.close()

最佳實踐:

  • 使用上下文管理器( with語句) 來管理數據庫會話:這可以確保會話在代碼塊執行完畢後自動關閉。
  • 在Flask 應用中,利用Flask-SQLAlchemy 等擴展庫,可以更方便地管理數據庫連接和會話。 這些庫通常會自動處理連接的關閉和釋放。

通過以上改進,可以有效地解決SQLAlchemy 數據庫連接無法正確關閉的問題,並提高代碼的健壯性和可維護性。 記住,顯式地關閉連接是最佳實踐,避免依賴垃圾回收機制。

以上是為什麼Sqlalchemy數據庫連接無法正確關閉?如何解決這個問題?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

<🎜>:泡泡膠模擬器無窮大 - 如何獲取和使用皇家鑰匙
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
Mandragora:巫婆樹的耳語 - 如何解鎖抓鉤
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
北端:融合系統,解釋
3 週前 By 尊渡假赌尊渡假赌尊渡假赌

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

熱門話題

Java教學
1668
14
CakePHP 教程
1428
52
Laravel 教程
1329
25
PHP教程
1273
29
C# 教程
1256
24
2025最安全交易所TOP5:黑U避坑指南,資金100%保命法則 2025最安全交易所TOP5:黑U避坑指南,資金100%保命法則 May 08, 2025 pm 08:27 PM

在加密貨幣交易領域,交易所的安全性始終是用戶關注的重點。 2025年,經過多年的發展和演變,一些交易所憑藉其卓越的安全措施和用戶體驗脫穎而出。本文將詳細介紹2025年最安全的五大交易所,並提供如何避開黑U(黑客攻擊用戶)的實用指南,確保您的資金100%安全。

MySQL:SQL的實際應用 MySQL:SQL的實際應用 May 08, 2025 am 12:12 AM

MySQL受歡迎的原因是其性能卓越且易於使用和維護。 1.創建數據庫和表:使用CREATEDATABASE和CREATETABLE命令。 2.插入和查詢數據:通過INSERTINTO和SELECT語句操作數據。 3.優化查詢:使用索引和EXPLAIN語句提升性能。

AI和作曲家:增強代碼質量和開發 AI和作曲家:增強代碼質量和開發 May 09, 2025 am 12:20 AM

AI在Composer中主要通過依賴推薦、依賴衝突解決和代碼質量提升來提高開發效率和代碼質量。 1.AI可以根據項目需求推薦合適的依賴包。 2.AI提供智能解決方案來處理依賴衝突。 3.AI審查代碼並提供優化建議,提升代碼質量。通過這些功能,開發者可以更專注於業務邏輯的實現。

全球十大支持多鏈交易的加密貨幣平台2025年權威發布 全球十大支持多鏈交易的加密貨幣平台2025年權威發布 May 08, 2025 pm 07:15 PM

根據 2025 年權威機構的最新評估和行業趨勢,以下是全球十大支持多鏈交易的加密貨幣平台,結合交易量、技術創新、合規性及用戶口碑綜合分析:

2025年十大數字虛擬幣交易APP排行 十大數字幣交易所app匯總 2025年十大數字虛擬幣交易APP排行 十大數字幣交易所app匯總 May 08, 2025 pm 05:24 PM

2025年十大數字虛擬幣交易APP排行:1. Binance:全球領先,提供高效交易和多種金融產品。 2. OKX:創新多樣,支持多種交易類型。 3. Huobi:穩定可靠,服務優質。 4. Coinbase:新手友好,界面簡潔。 5. Kraken:專業交易者首選,工具強大。 6. Bitfinex:高效交易,交易對豐富。 7. Bittrex:安全合規,監管合作。

歐意OKX6.118.0版本最新下載教程 歐意OKX6.118.0版本最新下載教程 May 07, 2025 pm 06:51 PM

歐意OKX6.118.0版本最新下載教程:1、點擊文章中快捷鏈接;2、點擊下載即可(如果是網頁用戶請先進行信息註冊)。最新安卓版本v6.118.0優化了部分功能和體驗,讓交易更容易。立即更新App,感受更加極致的交易體驗。

Python vs. JavaScript:開發人員的比較分析 Python vs. JavaScript:開發人員的比較分析 May 09, 2025 am 12:22 AM

Python和JavaScript的主要區別在於類型系統和應用場景。 1.Python使用動態類型,適合科學計算和數據分析。 2.JavaScript採用弱類型,廣泛用於前端和全棧開發。兩者在異步編程和性能優化上各有優勢,選擇時應根據項目需求決定。

零基礎躺賺攻略:2025年必囤的5種山寨幣,穩賺50倍! 零基礎躺賺攻略:2025年必囤的5種山寨幣,穩賺50倍! May 08, 2025 pm 08:30 PM

在加密貨幣市場中,山寨幣(altcoins)常常被投資者視為潛在的高回報資產。雖然市場上存在許多山寨幣,但並非所有山寨幣都能帶來預期的收益。本文將為零基礎的投資者提供一份詳細的攻略,介紹2025年值得囤積的5種山寨幣,並解釋如何通過這些投資實現穩賺50倍的目標。

See all articles