
在Python异步编程中,尤其是在使用FastAPI等框架时,开发者常面临在类构造函数`__init__`中执行异步操作的挑战。由于`__init__`方法不能直接使用`await`,本文将探讨在构造器中处理异步代码的常见问题、不推荐的解决方案及其弊端,并重点介绍如何采用异步工厂模式或独立初始化方法,以实现高效、可维护且符合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的限制,一些开发者可能会考虑以下几种方案,但它们通常伴随着严重的性能或可维护性问题:
这种方法试图在同步的__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"))弊端:
另一种方法是让__init__保持同步且轻量,然后提供一个单独的异步方法(通常命名为create或from_client)来执行实际的异步初始化逻辑。
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")弊端:
解决上述问题的最佳实践是采用异步工厂模式。这种模式将对象的创建(同步部分)和异步初始化(异步部分)分离。__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)优点:
在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__应保持同步、轻量,并专注于设置对象的初始状态。对于需要异步初始化的类,推荐采用异步工厂模式:
遵循这些实践,不仅能解决异步构造器的问题,还能提升代码的可读性、可维护性和应用程序的整体性能。
以上就是深入理解Python中异步构造器与初始化模式的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号