Python实现访问者模式
假設要實現一個存放多種類型數據結構的對象,比如一個存放算術操作數和操作符的樹結點,需要存放包含一元操作符、二元操作符和數字類型的結點
class Node:passclass UnaryOperator(Node):def __init__(self, operand):self.operand = operandclass BinaryOperator(Node):def __init__(self, left, right):self.left = leftself.right = rightclass Add(BinaryOperator):passclass Sub(BinaryOperator):passclass Mul(BinaryOperator):passclass Div(BinaryOperator):passclass Negative(UnaryOperator):passclass Number(Node):def __init__(self, value):self.value = value執行運算需要這樣調用
# 假設運算式子:2 - (2+2) * 2 / 1 = 2-(8) = -6.0 t1 = Add(Number(2), Number(2)) t2 = Mul(t1, Number(2)) t3 = Div(t2, Number(1)) t4 = Sub(Number(2), t3)或者這樣調用
t5 = Sub(Number(2), Div(Mul(Add(Number(2), Number(2)), Number(2)), Number(1)))這樣子需要執行多次類的調用,極不易讀寫且冗長,有沒有一種方法讓調用更加通用,訪問變得簡單呢。這里使用訪問者模式可以達到這樣的目的。
訪問者模式能夠在不改變元素所屬對象結構的情況下操作元素,讓調用或調用者(caller)的方式變得簡單,這種操作常見于的士公司操作,當一個乘客叫了一輛的士時,的士公司接收到了一個訪問者,并分配一輛的士去接這個乘客。
首先定義一個訪問者結點類VisitorNode,實現最基本的訪問入口,任何訪問的方式都需要繼承這個訪問者結點類,并通過這個訪問者結點類的visit()方法來訪問它的各種操作
# 訪問者節點的基類 class NodeVisitor:def visit(self, node):if not isinstance(node, Node): # 不是Node對象時當做一個值返回,如果有其他情況可以根據實際來處理return nodeself.meth = "visit_" + type(node).__name__.lower() # type(node)也可以換成node.__class__(只要node.__class__不被篡改)meth = getattr(self, self.meth, None) if meth is None:meth = self.generic_visitreturn meth(node)def generic_visit(self, node):raise RuntimeError(f"No {self.meth} method")# (一種)訪問者對應的類 class Visitor(NodeVisitor):"""方法的名稱定義都要與前面定義過的結點類(Node)的名稱保證一致性"""def visit_add(self, node):return self.visit(node.left) + self.visit(node.right)def visit_sub(self, node):return self.visit(node.left) - self.visit(node.right)def visit_mul(self, node):return self.visit(node.left) * self.visit(node.right)def visit_div(self, node):return self.visit(node.left) / self.visit(node.right)def visit_negative(self, node): # 如果class Negative 命名-> class Neg,那么 def visit_negative 命名-> def visit_negreturn -self.visit(node.operand)def visit_number(self, node):return node.value這里的meth = getattr(self, self.meth, None)使用了字符串調用對象方法,self.meth動態地根據各類Node類(Add, Sub, Mul…)的名稱定義了對應于類Visitor中的方法(visit_add, visit_sub, visit_mul…)簡化了訪問入口的代碼,當沒有獲取到對應的方法時會執行generic_visit()并拋出RuntimeError的異常提示訪問過程中的異常
如果需要添加一種操作,比如取絕對值,只需要定義一個類class Abs(Unaryoperator): pass并在類Visitor中定義一個visit_abs(self, node)方法即可,不需要做出任何多余的修改,更不需要改變存儲的結構
這里visit()方法調用了visit_xxx()方法,而visit_xxx()可能也調用了visit(),本質上是visit()的循環遞歸調用,當數據量變大時,效率會變得很慢,且遞歸層次過深時會導致超過限制而失敗,而下面介紹的就是利用棧和生成器來消除遞歸提升效率的實現訪問者模式的方法
import typesclass Node:passclass BinaryOperator(Node):def __init__(self, left, right):self.left = leftself.right = rightclass UnaryOperator(Node):def __init__(self, operand):self.operand = operandclass Add(BinaryOperator):passclass Sub(BinaryOperator):passclass Mul(BinaryOperator):passclass Div(BinaryOperator):passclass Negative(UnaryOperator):passclass Number(Node):def __init__(self, value): # 與UnaryOperator區別僅命名不同self.value = valueclass NodeVisitor:def visit(self, node):# 使用棧+生成器來替換原來visit()的遞歸寫法stack = [node]last_result = None # 執行一個操作最終都會返回一個值while stack:last = stack[-1]try:if isinstance(last, Node):stack.append(self._visit(stack.pop()))elif isinstance(last, types.GeneratorType): # GeneratorType會是上一個if返回的對象,這個對象會返回兩個node執行算術之后的結果# 如果是生成器,不pop掉,而是不斷send,直到StopIteration# 如果last_result不是None,這個值會給回到生成器(例如2被visit_add()的左值接收到)stack.append(last.send(last_result))last_result = Noneelse: # 計算結果是一個值last_result = stack.pop()except StopIteration: # 生成器yield結束stack.pop()return last_resultdef _visit(self, node):self.method_name = "visit_" + type(node).__name__.lower()method = getattr(self, self.method_name, None)if method is None:self.generic_visit(node)return method(node)def generic_visit(self, node):raise RuntimeError(f"No {self.method_name} method")class Visitor(NodeVisitor):def visit_add(self, node):yield (yield node.left) + (yield node.right) # node.left和node.right都可能是Nodedef visit_sub(self, node):yield (yield node.left) - (yield node.right)def visit_mul(self, node):yield (yield node.left) * (yield node.right)def visit_div(self, node):yield (yield node.left) / (yield node.right)def visit_negative(self, node):yield -(yield node.operand)def visit_number(self, node):return node.value測試是否還會引起超過遞歸層數的異常
def test_time_cost():import times = time.perf_counter()a = Number(0)for n in range(1, 100000):a = Add(a, Number(n))v = Visitor()print(v.visit(a))print(f"time cost:{time.perf_counter() - s}")輸出正常,沒有問題
4999950000 time cost:0.9547078最后琢磨出了一個似乎可以作為替代的方法
class Node:passclass UnaryOperator(Node):def __init__(self, operand):self.operand = operandclass BinaryOperator(Node):def __init__(self, left, right):self.left = leftself.right = rightclass Add(BinaryOperator):def __init__(self, left, right):super().__init__(left, right)self.value = self.left.value + self.right.valuepassclass Sub(BinaryOperator):def __init__(self, left, right):super().__init__(left, right)self.value = self.left.value - self.right.valuepassclass Mul(BinaryOperator):def __init__(self, left, right):super().__init__(left, right)self.value = self.left.value * self.right.valuepassclass Div(BinaryOperator):def __init__(self, left, right):super().__init__(left, right)self.value = self.left.value / self.right.valuepassclass Negative(UnaryOperator):def __init__(self, operand):super().__init__(operand)self.value = -self.operand.valuepassclass Number(Node):def __init__(self, value):self.value = value運行測試
def test_time_cost():import times = time.perf_counter()a = Number(0)for n in range(1, 100000):a = Add(a, Number(n))print(a.value)print(time.perf_counter() - s)輸出
4999950000 0.2506986比前面的訪問者模式還快而且不用遞歸,哈哈,焯,算了,寫都寫了
總結
以上是生活随笔為你收集整理的Python实现访问者模式的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 笔记本AutoCAD启动时闪退怎么办_戴
- 下一篇: linux帮助命令和用法,Linux命令