77

以下产生NameError: name 'Client' is not defined. 我该如何解决?

class Server():
    def register_client(self, client: Client)
        pass


class Client():
    def __init__(self, server: Server):
        server.register_client(self)
4

2 回答 2

117

您可以通过对尚未定义的类使用字符串名称来使用前向引用:Client

class Server():
    def register_client(self, client: 'Client')
        pass

从 Python 3.7 开始,您还可以通过在模块顶部添加以下导入来推迟注解的所有运行时解析:__future__

from __future__ import annotations

此时注释被存储为表达式的抽象语法树的字符串表示;您可以使用typing.get_type_hints()来解决这些问题(并解决上面使用的前向引用)。

有关详细信息,请参阅PEP 563 - 延迟评估注释;此行为将是 Python 4.0 中的默认行为。

于 2015-11-21T15:06:56.340 回答
5

如果您使用的是 Python 3.7+,请from __future__ import annotations 按照另一个答案中的说明使用。但是,如果由于操作系统限制(如 Cygwin 截至 2019 年 6 月 3 日)还不能使用 3.7,则可以使用前向引用模块来满足这些类型的前向/循环依赖问题。

请原谅这个人为的例子,但这应该说明这种方法的有用性。

class Server():
    clients: list = None

    def __init__(self):
        self.clients=[]

    def register_client(self, client: 'Client') -> None:
        self.clients.append(client)
        print('Client `%s` registered with server' % client.name)

    def print_clients(self) -> None:
        for i, client in enumerate(self.clients):
            print('client %i: %s' % (i, client.name))

    @staticmethod
    def build_clone(server: 'Server') -> 'Server':
        svr_new: Server = Server()
        for client in server.clients:
            svr_new.register_client(client)
        return svr_new

class Client():
    name: str = None
    def __init__(self, name: str, server: 'Server'):
        self.name = name
        server.register_client(self)


svr = Server()
cli = Client('foo', svr)
svr.print_clients()

svr_clone = Server.build_clone(svr)
svr_clone.print_clients()
于 2019-06-05T17:12:09.457 回答