0

0

深入理解Python中异步构造器与初始化模式

霞舞

霞舞

发布时间:2025-12-04 14:14:23

|

526人浏览过

|

来源于php中文网

原创

深入理解python中异步构造器与初始化模式

在Python异步编程中,尤其是在使用FastAPI等框架时,开发者常面临在类构造函数`__init__`中执行异步操作的挑战。由于`__init__`方法不能直接使用`await`,本文将探讨在构造器中处理异步代码的常见问题、不推荐的解决方案及其弊端,并重点介绍如何采用异步工厂模式或独立初始化方法,以实现高效、可维护且符合Python异步编程范式的类初始化。

异步构造器的挑战与Python限制

Python的__init__方法是一个同步方法,它在对象实例化时被调用,用于设置对象的初始状态。这意味着在__init__内部无法直接使用await关键字来执行异步操作,例如连接数据库、网络请求或文件I/O。尝试在__init__中直接使用await会导致语法错误。

考虑一个典型的场景,例如在FastAPI项目中需要一个与Azure Cosmos DB交互的异步类。这个类在实例化时需要确保数据库和容器已经创建。一个直观但错误尝试可能如下:

from azure.cosmos.aio import CosmosClient, Database, Container # 假设这些类型已定义

class CosmosCRUD:
    def __init__(self, client: CosmosClient):
        self.client = client
        # 以下代码会报错,因为__init__不能await
        # self.database = await self.client.create_database_if_not_exists("MY_DATABASE_NAME")
        # self.container = await self.database.create_container_if_not_exists("MY_CONTAINER_NAME", partition_key="/id")

上述代码由于__init__不是async def函数,因此无法使用await,直接运行会抛出SyntaxError。

立即学习Python免费学习笔记(深入)”;

不推荐的解决方案及其弊端

为了规避__init__不能await的限制,一些开发者可能会考虑以下几种方案,但它们通常伴随着严重的性能或可维护性问题:

1. 在__init__中创建并运行新的事件循环

这种方法试图在同步的__init__方法内部,通过手动创建和运行一个新的异步事件循环来执行异步代码。

import asyncio
from azure.cosmos.aio import CosmosClient # 假设CosmosClient已导入

class CosmosCRUDBad:
    def __init__(self, client: CosmosClient):
        self.client = client
        loop = asyncio.get_event_loop()
        # 这种方式会阻塞当前的线程,直到异步操作完成
        self.database = loop.run_until_complete(self.client.create_database_if_not_exists("MY_DATABASE_NAME"))
        self.container = loop.run_until_complete(self.database.create_container_if_not_exists("MY_CONTAINER_NAME", partition_key="/id"))

弊端:

  • 性能瓶颈 loop.run_until_complete()会阻塞当前线程,直到异步操作完成。在FastAPI等异步Web框架中,如果每个请求都实例化此类,这将导致请求处理被串行化,严重损害应用程序的并发性能。这违背了异步编程的初衷。
  • 事件循环管理复杂: 在__init__中管理事件循环容易出错,可能导致资源泄露或不可预测的行为,尤其是在多个实例创建时。
  • 非惯用模式: 这种模式在Python异步社区中被视为反模式,应尽量避免。

2. 忽略__init__,使用独立的异步构造函数

另一种方法是让__init__保持同步且轻量,然后提供一个单独的异步方法(通常命名为create或from_client)来执行实际的异步初始化逻辑。

Text-To-Song
Text-To-Song

免费的实时语音转换器和调制器

下载
from typing import Optional
from azure.cosmos.aio import CosmosClient, Database, Container

class CosmosCRUDPartial:
    def __init__(self, client: CosmosClient):
        self.client = client
        self.database: Optional[Database] = None  # 声明类型以满足IDE
        self.container: Optional[Container] = None # 声明类型以满足IDE

    async def create(self, db_name: str, container_name: str, partition_key: str):
        self.database = await self.client.create_database_if_not_exists(db_name)
        self.container = await self.database.create_container_if_not_exists(container_name, partition_key=partition_key)
        return self # 返回自身以便链式调用或直接使用

# 使用方式
# crud_instance = CosmosCRUDPartial(client)
# await crud_instance.create("MY_DATABASE_NAME", "MY_CONTAINER_NAME", "/id")

弊端:

  • IDE警告: 尽管在__init__中声明了实例变量的类型,但由于这些变量的实际赋值发生在create方法中,IDE(如PyCharm)可能会错误地认为这些变量在对象创建后尚未被初始化,从而发出警告。这会影响开发体验和代码的可读性。
  • 使用不直观: 用户必须记住先调用__init__,然后手动调用create方法,这使得类的使用模式不够简洁和“原子化”。

推荐解决方案:异步工厂模式

解决上述问题的最佳实践是采用异步工厂模式。这种模式将对象的创建(同步部分)和异步初始化(异步部分)分离。__init__方法保持同步,只负责最基本的属性设置。一个单独的async classmethod(即异步工厂方法)负责执行所有异步初始化逻辑,并返回一个完全初始化的实例。

from typing import Optional
from azure.cosmos.aio import CosmosClient, Database, Container

class CosmosCRUD:
    def __init__(self, client: CosmosClient, database: Database, container: Container):
        """
        构造函数只进行同步赋值,确保所有必要的依赖都已传入。
        """
        self.client = client
        self.database = database
        self.container = container

    @classmethod
    async def create(cls, client: CosmosClient, db_name: str, container_name: str, partition_key: str):
        """
        异步工厂方法,负责执行所有异步初始化逻辑,并返回一个完全初始化的实例。
        """
        database = await client.create_database_if_not_exists(db_name)
        container = await database.create_container_if_not_exists(container_name, partition_key=partition_key)
        return cls(client, database, container) # 调用__init__创建实例

    async def get_item(self, item_id: str, partition_key: str):
        # 示例CRUD操作
        return await self.container.read_item(item_id, partition_key=partition_key)

    async def create_item(self, item: dict):
        # 示例CRUD操作
        return await self.container.create_item(item)

优点:

  • 清晰的分离: __init__保持同步和轻量,仅用于属性赋值。所有异步操作都在create工厂方法中完成。
  • 原子化创建: CosmosCRUD.create()方法返回一个已经完全初始化并准备好使用的对象,避免了分两步初始化的复杂性。
  • IDE友好: 实例变量在__init__中直接赋值,IDE能够正确识别它们的存在和类型,不会发出不必要的警告。
  • 符合异步范式: 整个对象创建流程是异步的,不会阻塞事件循环。

集成到FastAPI应用

在FastAPI中,这种异步工厂模式可以与应用程序的生命周期事件(lifespan)或依赖注入(Depends)完美结合。通常,像CosmosCRUD这样的数据库客户端只需要在应用程序启动时创建一次,并在整个应用程序生命周期中复用。

示例:使用FastAPI的lifespan事件

from fastapi import FastAPI, Depends
from contextlib import asynccontextmanager
from azure.cosmos.aio import CosmosClient # 假设CosmosClient已导入

# 假设您的Cosmos DB连接字符串和密钥
COSMOS_DB_ENDPOINT = "YOUR_COSMOS_DB_ENDPOINT"
COSMOS_DB_KEY = "YOUR_COSMOS_DB_KEY"
DB_NAME = "MY_DATABASE_NAME"
CONTAINER_NAME = "MY_CONTAINER_NAME"
PARTITION_KEY = "/id"

# 全局变量,用于存储Cosmos客户端和CRUD实例
cosmos_client: Optional[CosmosClient] = None
cosmos_crud_instance: Optional[CosmosCRUD] = None

@asynccontextmanager
async def lifespan(app: FastAPI):
    global cosmos_client, cosmos_crud_instance
    # 启动时创建CosmosClient和CosmosCRUD实例
    cosmos_client = CosmosClient(COSMOS_DB_ENDPOINT, COSMOS_DB_KEY)
    cosmos_crud_instance = await CosmosCRUD.create(
        cosmos_client, DB_NAME, CONTAINER_NAME, PARTITION_KEY
    )
    print("Cosmos DB client and CRUD instance initialized.")
    yield # 应用启动,开始处理请求
    # 关闭时清理资源
    if cosmos_client:
        await cosmos_client.close()
    print("Cosmos DB client closed.")

app = FastAPI(lifespan=lifespan)

# 依赖注入函数
async def get_cosmos_crud() -> CosmosCRUD:
    if cosmos_crud_instance is None:
        raise RuntimeError("CosmosCRUD instance not initialized.")
    return cosmos_crud_instance

@app.get("/items/{item_id}")
async def read_item(item_id: str, crud: CosmosCRUD = Depends(get_cosmos_crud)):
    # 假设partition_key与item_id相同,或者根据业务逻辑获取
    item = await crud.get_item(item_id, partition_key=item_id)
    if item:
        return {"item": item}
    return {"message": "Item not found"}

@app.post("/items/")
async def create_new_item(item_data: dict, crud: CosmosCRUD = Depends(get_cosmos_crud)):
    # 确保item_data包含partition_key
    if PARTITION_KEY.strip('/') not in item_data:
        item_data[PARTITION_KEY.strip('/')] = item_data.get('id', 'default_id') # 示例,实际应根据业务逻辑
    new_item = await crud.create_item(item_data)
    return {"message": "Item created", "item": new_item}

通过lifespan事件,我们确保了CosmosCRUD实例在应用启动时异步创建一次,并在应用关闭时优雅地清理资源。get_cosmos_crud依赖注入函数则负责在路由处理函数中提供这个预先创建好的实例。

总结与最佳实践

在Python异步编程中,避免在__init__中执行异步操作是核心原则。__init__应保持同步、轻量,并专注于设置对象的初始状态。对于需要异步初始化的类,推荐采用异步工厂模式

  1. __init__保持同步: 仅接收已经准备好的依赖或基本参数,并进行同步赋值。
  2. 提供异步工厂方法: 创建一个async classmethod(例如create),在该方法中执行所有异步初始化逻辑(如数据库连接、资源分配),然后使用cls(...)调用__init__来创建并返回一个完全初始化的实例。
  3. 集成到应用生命周期: 在FastAPI等框架中,利用lifespan事件在应用启动时创建一次异步实例,并通过依赖注入在整个应用中复用,从而实现高效、非阻塞的资源管理。

遵循这些实践,不仅能解决异步构造器的问题,还能提升代码的可读性、可维护性和应用程序的整体性能。

相关专题

更多
python开发工具
python开发工具

php中文网为大家提供各种python开发工具,好的开发工具,可帮助开发者攻克编程学习中的基础障碍,理解每一行源代码在程序执行时在计算机中的过程。php中文网还为大家带来python相关课程以及相关文章等内容,供大家免费下载使用。

758

2023.06.15

python打包成可执行文件
python打包成可执行文件

本专题为大家带来python打包成可执行文件相关的文章,大家可以免费的下载体验。

639

2023.07.20

python能做什么
python能做什么

python能做的有:可用于开发基于控制台的应用程序、多媒体部分开发、用于开发基于Web的应用程序、使用python处理数据、系统编程等等。本专题为大家提供python相关的各种文章、以及下载和课程。

761

2023.07.25

format在python中的用法
format在python中的用法

Python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

618

2023.07.31

python教程
python教程

Python已成为一门网红语言,即使是在非编程开发者当中,也掀起了一股学习的热潮。本专题为大家带来python教程的相关文章,大家可以免费体验学习。

1264

2023.08.03

python环境变量的配置
python环境变量的配置

Python是一种流行的编程语言,被广泛用于软件开发、数据分析和科学计算等领域。在安装Python之后,我们需要配置环境变量,以便在任何位置都能够访问Python的可执行文件。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

548

2023.08.04

python eval
python eval

eval函数是Python中一个非常强大的函数,它可以将字符串作为Python代码进行执行,实现动态编程的效果。然而,由于其潜在的安全风险和性能问题,需要谨慎使用。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

579

2023.08.04

scratch和python区别
scratch和python区别

scratch和python的区别:1、scratch是一种专为初学者设计的图形化编程语言,python是一种文本编程语言;2、scratch使用的是基于积木的编程语法,python采用更加传统的文本编程语法等等。本专题为大家提供scratch和python相关的文章、下载、课程内容,供大家免费下载体验。

708

2023.08.11

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

41

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 2.8万人学习

Django 教程
Django 教程

共28课时 | 3.2万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号