Python实现访问者模式详情

作者:Moelimoe 时间:2021-02-16 05:37:51 

假设要实现一个存放多种类型数据结构的对象,比如一个存放算术操作数和操作符的树结点,需要存放包含一元操作符、二元操作符和数字类型的结点

class Node:
? ? pass

class UnaryOperator(Node):
? ? def __init__(self, operand):
? ? ? ? self.operand = operand

class BinaryOperator(Node):
? ? def __init__(self, left, right):
? ? ? ? self.left = left
? ? ? ? self.right = right

class Add(BinaryOperator):
? ? pass

class Sub(BinaryOperator):
? ? pass

class Mul(BinaryOperator):
? ? pass

class Div(BinaryOperator):
? ? pass

class Negative(UnaryOperator):
? ? pass

class 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 node
? ? ? ? self.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_visit
? ? ? ? return 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_neg
? ? ? ? return -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 types

class Node:
? ? pass

class BinaryOperator(Node):
? ? def __init__(self, left, right):
? ? ? ? self.left = left
? ? ? ? self.right = right

class UnaryOperator(Node):
? ? def __init__(self, operand):
? ? ? ? self.operand = operand

class Add(BinaryOperator):
? ? pass

class Sub(BinaryOperator):
? ? pass

class Mul(BinaryOperator):
? ? pass

class Div(BinaryOperator):
? ? pass

class Negative(UnaryOperator):
? ? pass

class Number(Node):
? ? def __init__(self, value): ?# 与UnaryOperator区别仅命名不同
? ? ? ? self.value = value

class 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 = None
? ? ? ? ? ? ? ? else: ? # 计算结果是一个值
? ? ? ? ? ? ? ? ? ? last_result = stack.pop()
? ? ? ? ? ? except StopIteration: ? # 生成器yield结束
? ? ? ? ? ? ? ? stack.pop()
? ? ? ? return last_result

? ? def _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都可能是Node

? ? def 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 time
? ? s = 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

最后琢磨出了一个似乎可以作为替代的方法:

clas Node:
? ? psass

class UnaryOperator(Node):
? ? def __init__(self, operand):
? ? ? ? self.operand = operand

class BinaryOperator(Node):
? ? def __init__(self, left, right):
? ? ? ? self.left = left
? ? ? ? self.right = right

class Add(BinaryOperator):
? ? def __init__(self, left, right):
? ? ? ? super().__init__(left, right)
? ? ? ? self.value = self.left.value + self.right.value
? ? pass

class Sub(BinaryOperator):
? ? def __init__(self, left, right):
? ? ? ? super().__init__(left, right)
? ? ? ? self.value = self.left.value - self.right.value
? ? pass

class Mul(BinaryOperator):
? ? def __init__(self, left, right):
? ? ? ? super().__init__(left, right)
? ? ? ? self.value = self.left.value * self.right.value
? ? pass

class Div(BinaryOperator):
? ? def __init__(self, left, right):
? ? ? ? super().__init__(left, right)
? ? ? ? self.value = self.left.value / self.right.value
? ? pass

class Negative(UnaryOperator):
? ? def __init__(self, operand):
? ? ? ? super().__init__(operand)
? ? ? ? self.value = -self.operand.value
? ? pass

class Number(Node):
? ? def __init__(self, value):
? ? ? ? self.value = value

运行测试:

def test_time_cost():
? ? import time
? ? s = 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

来源:https://blog.csdn.net/Moelimoe/article/details/123755397

标签:Python,实现,访问者,模式
0
投稿

猜你喜欢

  • 利用标准库fractions模块让Python支持分数类型的方法详解

    2023-06-06 05:23:47
  • Python Pygame实战之水果忍者游戏的实现

    2021-07-25 03:22:37
  • Python实现的HMacMD5加密算法示例

    2022-01-15 06:05:31
  • JupyterNotebook设置Python环境的方法步骤

    2023-01-15 12:22:14
  • 二级域名原理以及asp实现程序

    2007-08-03 13:08:00
  • ADO组件之分页程序详解

    2008-10-09 12:28:00
  • asp一个空间绑定N个域名的方法!

    2009-03-08 18:32:00
  • 常用的匹配正则表达式和实例

    2008-06-07 09:19:00
  • python语言中有算法吗

    2022-06-22 12:42:15
  • Oracle学习笔记(五)

    2024-01-25 04:12:09
  • Python之list对应元素求和的方法

    2022-12-30 20:18:00
  • SQL Server 磁盘请求超时的833错误原因及解决方法

    2024-01-14 00:14:43
  • 在Django的视图中使用数据库查询的方法

    2024-01-14 20:03:20
  • python 数据挖掘算法的过程详解

    2022-11-17 09:09:19
  • adox 的vbs类,提取表名,列名等

    2008-07-02 12:37:00
  • python新式类和经典类的区别实例分析

    2023-07-28 10:01:02
  • python 使用pandas计算累积求和的方法

    2021-05-22 19:46:16
  • Python实现SMTP发送邮件详细教程

    2021-05-10 03:01:01
  • python超时重新请求解决方案

    2022-04-22 00:16:12
  • python cv2图像质量压缩的算法示例

    2023-08-16 19:09:02
  • asp之家 网络编程 m.aspxhome.com