我有一个具有同步和非同步代码的应用程序(因为它进行计算)。
Asyncpg 声称比 psycopg2 快得多,这里: https ://magic.io/blog/asyncpg-1m-rows-from-postgres-to-python/
我想出了在 DatabaseManager 上动态修补的代码,它本身使用 asyncpg 和协程方法。目标是使同步代码可以使用相同的界面。有一个更好的方法吗?
AsyncioWorkerThread 只是在守护线程中运行的事件循环。(见这个链接https://stackoverflow.com/questions/67199459/asyncio-loop-in-daemon-thread)
class DatabaseManager(DatabaseManager):
async def test():
asyncio.sleep(1)
print('pretend we are doing work with asyncpg')
class SyncDatabaseManager(DatabaseManager):
"""A sync version of the DatabaseManager."""
def __init__(self, *args, **kwargs):
"""Construct."""
super().__init__(*args, **kwargs)
self.workerThread = AsyncioWorkerThread(daemon=True)
self.workerThread.start()
self.__patch_all_coroutine_methods()
def __patch_all_coroutine_methods(self):
methods = inspect.getmembers(DatabaseManager, predicate=inspect.isfunction)
coro_methods = [(func_name, func_addr) for (func_name, func_addr) in methods if inspect.iscoroutinefunction(func_addr)]
def sync_method_factory(func_addr):
@functools.wraps(func_addr)
def sync_func(*args, **kwargs):
return self.workerThread.submit(func_addr(self, *args, **kwargs))
return sync_func
for func_name, func_addr in coro_methods:
setattr(self, func_name, sync_method_factory(func_addr))