1

假设我有BaseClass其中包含一些逻辑,其中和main_function()都是通用的。假设这两个类有其独特的实现,前者以同步方式获得,而后者以异步方式获得。我写了这样的东西,它似乎工作:SyncClassAsyncClassget_data()

class BaseClass:
    def get_data():
        pass

    @gen.coroutine
    def main_function():
        # some logic
        try:
            data = yield self.get_data()
        except:
            data = self.get_data()
        # some more logic

class SyncClass(BaseClass):
    def get_data():
        //makes sync call to Mongo and gets data (using Mongoengine)

class AsyncClass(BaseClass):
    @gen.coroutine
    def get_data():
        //makes async call to Mongo and gets data (using Motorengine)

我使用此代码作为解决方法,因为我已经以这种方式实现了 get_data() 的那些方法。有没有更优雅的解决方案?我的代码有两部分与我有关:

try:
    data = yield self.get_data()
except:
    data = self.get_data()

我不想在这里使用 try/except 。

另一件事是:我同时@gen.coroutine在.AsyncClass@gen.coroutineBaseClass

谢谢!

4

1 回答 1

3

同步和异步方法有不同的接口(这就是异步的意思)。AsyncClass.get_data返回一个FutureSyncClass.get_data才不是。如果这是在静态类型语言中,这两种方法将无法从基类实现相同的抽象方法。当然,Python 更灵活,不会以这种方式限制你,但是调用者仍然需要知道它正在处理哪种方法,或者准备好通过try/exceptisinstance检查等来查找(注意 try/except 是在这种情况下很危险,因为yield在龙卷风协程中会接受列表和字典之类的东西)

通常,您不能像在这里希望的那样在它们之间透明地切换。请记住,任何可能调用的函数yield self.get_data()也需要用 装饰@coroutine,因此一旦系统的一部分是异步的,它就会开始传播。通常最好接受这一趋势并让事情变得异步。

于 2017-04-02T14:36:14.853 回答