Python:未解析的静态变量类的引用

Str*_*tyk 0 python

所以我得到了这段代码:

class MyClass:

    ACTIONS = {
        "ACTION_A": MyClass.__a,
        "ACTION_B": MyClass.__b
    }

    @staticmethod
    def do(constant):
        ACTIONS[constant]()

    @staticmethod
    def __a():
        print("A")

    @staticmethod
    def __b():
        print("B")
Run Code Online (Sandbox Code Playgroud)

我正在尝试将私有__a__b函数映射到静态字典,因此我可以使用方法do执行函数.

尝试运行此代码时,我得到错误:ACTIONS字典的每一行上的" 未解析的引用'MyClass' " .

有关如何使其正常工作的任何想法?

Mar*_*ers 5

你不应该首先使用一个类.您所做的只是创建命名空间,为此使用模块.在包中创建一个新模块并将所有功能放在其中:

def _a():
    print("A")

def _b():
    print("B")


ACTIONS = {
    'ACTION_A': _a,
    'ACTION_B': _b,
}

def do(constant):
    ACTIONS[constant]()
Run Code Online (Sandbox Code Playgroud)

请注意,我使用了单个下划线名称.Python在类中使用双下划线名称来创建额外的每类命名空间.MyClass.__a变得MyClass._MyClass__a避免与子类冲突(因此他们可以自由地重用名称而不用担心破坏超类的实现),Python中没有隐私模型.

您可以使用装饰器来注册_a_b功能:

def do(constant):
    ACTIONS[constant]()

ACTIONS = {}

def action(name):
    def decorator(f):
        ACTIONS[name] = f
        return f

@action('ACTION_A')
def _a():
    print("A")

@action('ACTION_B')
def _b()
    print("B")
Run Code Online (Sandbox Code Playgroud)

您看到的具体错误是由于MyClass在整个class语句完成之前未设置名称.你必须设置该字典:

class MyClass:
    @classmethod
    def do(cls, constant):
        cls.ACTIONS[constant]()

    @staticmethod
    def _a():
        print("A")

    @staticmethod
    def _b():
        print("B")

MyClass.ACTIONS = {
    'ACTION_A': MyClass._a,
    'ACTION_B': MyClass._b,
}
Run Code Online (Sandbox Code Playgroud)

注意,这do是一个类方法,你不能只是ACTIONS作为全局访问,你需要使用MyClass.ACTIONS或者,如上所述,一个类方法然后引用对象cls.

您可以通过使用名称来设置语句ACTIONS之后class,并创建def do一个类方法:

class MyClass:
    ACTIONS = {
        'ACTION_A': '_a',
        'ACTION_B': '_b',
    }

    @classmethod
    def do(cls, constant):
        getattr(cls, cls.ACTIONS[constant])()

    @staticmethod
    def _a():
        print("A")

    @staticmethod
    def _b():
        print("B")
Run Code Online (Sandbox Code Playgroud)