当前位置: 首页
编程语言
如何在 Java 中利用 面向对象的组合模式 构建支持无限递归的动态规则判别引擎

如何在 Java 中利用 面向对象的组合模式 构建支持无限递归的动态规则判别引擎

热心网友 时间:2026-05-01
转载

如何在 Ja va 中利用面向对象的组合模式构建支持无限递归的动态规则判别引擎

如何在 Ja va 中利用 面向对象的组合模式 构建支持无限递归的动态规则判别引擎

免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈

理解组合模式在规则引擎中的核心价值

组合模式的价值,从来不是为了“套用”某个设计模式。它的真正用武之地,在于解决一类非常典型的问题:当规则本身可以嵌套、分组、条件化,并且结构完全不确定时——比如用户自定义的“所有条件都满足”、“任意一个满足”或者“非此即彼”的复杂逻辑,同时还需要一套统一的执行逻辑。这时候,用树形结构来建模就显得再自然不过了。

在 Ja va 中实现这一点,其实并不依赖任何重型框架。关键在于一个巧妙的抽象:把“单个规则”和“规则容器”统一到同一个接口下。这样一来,调用方就完全无需区分眼前的是叶子节点还是分支节点,只管调用即可,整个系统的扩展性和灵活性也就随之而来。

定义统一的 Rule 接口与基础实现

一切的基础,是定义一个顶层的、统一的行为契约。这个契约非常简单:

Rule.ja va

boolean evaluate(Context context);

有了这个契约,我们就可以提供两类具体的实现:

  • 原子规则(Leaf):比如 EqualsRule("status", "active") 或者 GreaterThanRule("age", 18)。它们直接读取上下文中的字段,进行判断,然后返回一个布尔结果,干净利落。
  • 组合规则(Composite):比如 AndRule(List children)OrRule(List children) 或者 NotRule(Rule inner)。它们的 evaluate() 方法会递归地调用所有子规则,再根据逻辑运算符(与、或、非)来聚合最终结果。

你看,通过这样的设计,任意深度的嵌套表达式——比如 AND(OR(A, B), NOT(C))——都能用一个清晰的对象树来表示。更妙的是,未来如果需要增加新的运算符,你只需要新增一个 Composite 类,完全不用修改已有的任何逻辑,这完美符合了“开闭原则”。

支持动态构建与运行时解析

理论模型有了,接下来就得让它“活”起来,能处理用户动态配置的规则。用户通常会用 JSON 或者一套自定义的 DSL 来描述规则,我们的任务就是把这些配置解析成一棵活的 Rule 对象树。

举个例子,解析下面这段 JSON:

{ "type": "and", "children": [ { "type": "eq", "field": "role", "value": "admin" }, { "type": "or", "children": [ { "type": "gt", "field": "score", "value": 90 }, { "type": "lt", "field": "age", "value": 35 } ] } ] }

实现起来并不复杂。我们可以编写一个轻量的 RuleParser,利用工厂方法,根据 JSON 中的 type 字段来创建对应的 Rule 实例。对于 children 这样的嵌套字段,就递归地调用解析器自身。这里有个细节值得注意:每个组合规则在构造时,最好不要立即校验子规则列表是否为空。允许空列表,并为其设定一个默认行为(比如 AND 规则遇到空列表时返回 true,OR 规则返回 false),可以有效避免运行时出现恼人的空指针异常。

执行时避免栈溢出与提升性能

“支持无限递归”听起来很强大,但在真实世界里,无限不等于无限制。嵌套层数一旦过深(比如上百层),Ja va 的调用栈就很可能撑不住,抛出 StackOverflowError。怎么解决?通常有两个思路:

  • 主动防御:在 evaluate() 方法中加入一个深度计数器,一旦超过预设的阈值(例如50层),就主动抛出像 RuleEvaluationException("max depth exceeded") 这样的业务异常,提前终止,避免系统崩溃。
  • 改变执行模型:将隐式的递归调用,改为使用显式的栈(比如 Deque)来手动控制遍历过程。每个栈帧记录当前正在执行的规则和下一个要处理的子规则索引。这种方法不仅完全避免了栈溢出,内存消耗可控,而且在调试和日志记录时会清晰得多。

除了稳定性,性能也是关键。对于那些被高频调用、且计算结果纯函数化的原子规则(即输出只依赖于输入上下文,没有副作用),完全可以引入缓存机制。用一个 CachedRule 包装器,内部通过 WeakHashMap 来缓存最近的评估结果,这里的 ContextKey 可以根据规则依赖的字段哈希值来生成。这能在规则复杂、计算成本高时,带来显著的性能提升。

来源:https://www.php.cn/faq/2400390.html

游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。

同类文章
更多
如何优化Apache2响应速度

如何优化Apache2响应速度

Apache2响应速度优化实操指南 想让你的Apache2服务器跑得更快?这事儿其实有章可循。下面这份实操指南,将从基础到进阶,帮你系统地提升响应速度。记住,所有优化都建立在不变动核心业务逻辑和架构的前提下。 一 基础与系统层面优化 优化得从地基开始。系统层面的几个关键设置,往往能以小成本换来大收益

时间:2026-05-01 22:39
git多人协作的工作流程【汇总】

git多人协作的工作流程【汇总】

多人协作必须禁用直接 push 到 main 分支:PR MR 流程是保障代码质量、自动化测试与冲突预判的核心机制;最佳实践包括语义化分支命名、启用分支保护规则,并规范 rebase 与 merge 的使用场景。 多人协作时,为什么禁止直接 push 到 main 分支? 直接向主分支推送代码,表面

时间:2026-05-01 22:39
CentOS上如何升级PHPStorm到最新版本

CentOS上如何升级PHPStorm到最新版本

在 CentOS 上升级 PhpStorm 的可选方案 说到在 CentOS 上升级 PhpStorm,其实路径很清晰。核心原则是:优先使用内置更新或 JetBrains Toolbox App 这类自动管理工具,其次才是手动下载安装包覆盖升级。下面,就按推荐顺序,把每种方式的操作步骤和关键要点给你

时间:2026-05-01 22:39
Atom如何设置自动保存?Atom自动保存功能开启教程

Atom如何设置自动保存?Atom自动保存功能开启教程

Atom如何设置自动保存?Atom自动保存功能开启教程 如果你还在为Atom的自动保存功能头疼,那很可能踩中了几个常见的“坑”。从1 27版本开始,autosa ve功能已经作为核心特性内置,不再依赖插件。但问题也随之而来:为什么设置了却不见效?答案往往藏在版本、配置层级,或者那些本该被清理的旧插件

时间:2026-05-01 22:39
如何在CentOS上备份PHPStorm的配置文件

如何在CentOS上备份PHPStorm的配置文件

在 CentOS 上备份 PhpStorm 配置文件:完整指南与最佳实践 一、备份前的准备工作 在开始备份 PhpStorm 配置之前,充分的准备工作至关重要。这能有效保障备份数据的完整性与安全性,避免因操作不当导致配置丢失或损坏。 彻底关闭 PhpStorm 应用程序:这是首要且必须的步骤。确保

时间:2026-05-01 22:38
热门专题
更多
刀塔传奇破解版无限钻石下载大全 刀塔传奇破解版无限钻石下载大全
洛克王国正式正版手游下载安装大全 洛克王国正式正版手游下载安装大全
思美人手游下载专区 思美人手游下载专区
好玩的阿拉德之怒游戏下载合集 好玩的阿拉德之怒游戏下载合集
不思议迷宫手游下载合集 不思议迷宫手游下载合集
百宝袋汉化组游戏最新合集 百宝袋汉化组游戏最新合集
jsk游戏合集30款游戏大全 jsk游戏合集30款游戏大全
宾果消消消原版下载大全 宾果消消消原版下载大全
  • 日榜
  • 周榜
  • 月榜
热门教程
更多
  • 游戏攻略
  • 安卓教程
  • 苹果教程
  • 电脑教程