【三十四】springboot+easyRule初识规则引擎

2024-08-31 17:20

本文主要是介绍【三十四】springboot+easyRule初识规则引擎,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

        代码场景:厂里有几个员工,现在厂长颁布了新的厂规关于薪资发放,如下:

  • 1、加班时长超过80小时的,一个小时10块钱;不满80小时的,不算加班。
  • 2、上班打卡迟到3次以下的不扣钱,3次以上的一次扣100。

        针对如上需求,是不是就可以通过写if-if判断来处理,但是如果规则变化呢,老板想只要迟到1次就扣1000,或者只要加班就100块钱一个小时呢,是不是只有改代码升级。 本章针对这个问题,通过规则引擎实现这个场景,实现规则配置化。

目录

一、准备工作

二、注解方式实现

三、yml配置方式实现


一、准备工作

1、表

2、依赖
<dependency><groupId>org.jeasy</groupId><artifactId>easy-rules-core</artifactId><version>4.1.0</version></dependency><dependency><groupId>org.jeasy</groupId><artifactId>easy-rules-mvel</artifactId><version>4.1.0</version></dependency><dependency><groupId>org.jeasy</groupId><artifactId>easy-rules-spel</artifactId><version>4.1.0</version></dependency><dependency><groupId>org.jeasy</groupId><artifactId>easy-rules-jexl</artifactId><version>4.1.0</version></dependency>

二、注解方式实现

1、编写目的场景的规则
@Rule(name = "无效加班", description = "加班费的结算", priority = 1)
public class Rule1 {@Conditionpublic boolean when(@Fact("time") double time) {return time<=80;}@Actionpublic void then(@Fact("reason") StringBuffer reason) {reason.append("加班少于80小时,不发加班费;");}}
@Rule(name = "有效加班", description = "加班费的结算", priority = 2)
public class Rule2 {@Conditionpublic boolean when(@Fact("time") double time) {return time > 80;}@Actionpublic void then(@Fact("time") double time,@Fact("reason") StringBuffer reason,@Fact("money") AtomicDouble money) {money.set(money.get()+10*(time-80));reason.append("加班费:").append(10*(time-80)).append(";");}}
@Rule(name = "迟到警告", description = "迟到的惩罚", priority = 3)
public class Rule3 {@Conditionpublic boolean when(@Fact("count") int count) {return count<=3;}@Actionpublic void then(@Fact("count") int count, @Fact("money") AtomicDouble money, @Fact("reason") StringBuffer reason) {reason.append("迟到小于3次,暂时不扣钱;");}}
@Rule(name = "迟到扣钱", description = "迟到的惩罚", priority = 3)
public class Rule4 {@Conditionpublic boolean when(@Fact("count") int count) {return count>3;}@Actionpublic void then(@Fact("count") int count, @Fact("money") AtomicDouble money, @Fact("reason") StringBuffer reason) {money.set(money.get() - (count-3)*100);reason.append("迟到大于3次,扣钱:").append((count - 3) * 100).append(";");}}
  • @Rule标识该类为规则类
  • @Condition标识该方法为条件(一个rule类只能有一个) 
  • @Action标识该方法为条件满足为true后的执行方法
  • @Fact代表事实,标识入参,可以通过Fact的key值获取值
  • priority标识该rule的执行顺序
2、编写接口
@Slf4j
@Api(tags = "牛马管理接口")
@RestController
@RequestMapping("/staffController")
@AllArgsConstructor
public class StaffController {private final StaffMapper staffMapper;@ApiOperation(value = "计算工资")@PostMapping("/getSalary")public BaseResponse<Integer> getSalary() {// 初始化规则引擎RulesEngineParameters parameters = new RulesEngineParameters().skipOnFirstAppliedRule(false);DefaultRulesEngine engine = new DefaultRulesEngine(parameters);engine.registerRuleListener(new MyRuleListener());// 注册规则进入引擎Rules rules = new Rules();rules.register(new Rule1());rules.register(new Rule2());rules.register(new Rule3());rules.register(new Rule4());//UnitRuleGroup unitRuleGroup = new UnitRuleGroup("myUnitRuleGroup", "myUnitRuleGroup");//unitRuleGroup.addRule(new Rule1());//unitRuleGroup.addRule(new Rule2());//unitRuleGroup.addRule(new Rule3());//unitRuleGroup.addRule(new Rule4());//rules.register(unitRuleGroup);List<Staff> list = staffMapper.selectList(new QueryWrapper<>());for (Staff staff : list) {AtomicDouble money = new AtomicDouble((Double.parseDouble(staff.getMoney())));double beforeMoney = money.get();StringBuffer reason = new StringBuffer();Facts facts = new Facts();facts.put("time", Double.parseDouble(staff.getTime()));facts.put("count", staff.getCount());facts.put("money", money);facts.put("reason", reason);engine.fire(rules, facts);Staff staffNew = staffMapper.selectById(staff.getId());staffNew.setFinalMoney(facts.get("money").toString());staffNew.setDetail(reason.toString());staffMapper.updateById(staffNew);}return RespGenerator.returnOK("成功");}}
3、测试

根据结果可以看到数据正确。

现在假设我们厂长更改了需求:现在只要迟到超过3次,都没有加班费

        如上代码可以根据实际情况改造一下,当有需求所有规则要么全部执行,要么全部不执行时(只要有一个不满足就全部跳过执行),可以选用另一种方式UnitRuleGroup,改造如下:

@ApiOperation(value = "计算工资")@PostMapping("/getSalary")public BaseResponse<Integer> getSalary() {// 初始化规则引擎RulesEngineParameters parameters = new RulesEngineParameters().skipOnFirstAppliedRule(false);DefaultRulesEngine engine = new DefaultRulesEngine(parameters);engine.registerRuleListener(new MyRuleListener());// 注册规则进入引擎Rules rules = new Rules();//rules.register(new Rule1());//rules.register(new Rule2());//rules.register(new Rule3());//rules.register(new Rule4());UnitRuleGroup unitRuleGroup = new UnitRuleGroup("myUnitRuleGroup", "myUnitRuleGroup");//unitRuleGroup.addRule(new Rule1());unitRuleGroup.addRule(new Rule2());unitRuleGroup.addRule(new Rule3());//unitRuleGroup.addRule(new Rule4());rules.register(unitRuleGroup);List<Staff> list = staffMapper.selectList(new QueryWrapper<>());for (Staff staff : list) {AtomicDouble money = new AtomicDouble((Double.parseDouble(staff.getMoney())));double beforeMoney = money.get();StringBuffer reason = new StringBuffer();Facts facts = new Facts();facts.put("time", Double.parseDouble(staff.getTime()));facts.put("count", staff.getCount());facts.put("money", money);facts.put("reason", reason);engine.fire(rules, facts);Staff staffNew = staffMapper.selectById(staff.getId());staffNew.setFinalMoney(facts.get("money").toString());staffNew.setDetail(reason.toString());staffMapper.updateById(staffNew);}return RespGenerator.returnOK("成功");}

结果如下:

 可以看到赵四加班了一个小时也没有加班费了。

三、yml配置方式实现

1、增加配置文件
---
name: '无效加班'
description: '加班费的结算'
priority: 1
condition: "time<=80"
actions:- "reason.append('加班少于80小时,不发加班费;');"
---
name: '有效加班'
description: '加班费的结算'
priority: 2
condition: "time>80"
actions:- "money.set(money.get()+10*(time-80));reason.append('加班费:').append(10*(time-80)).append(';');"
---
name: '迟到警告'
description: '迟到的惩罚'
priority: 1
condition: "count<=3"
actions:- "reason.append('迟到小于3次,暂时不扣钱;');"
---
name: '迟到扣钱'
description: '迟到的惩罚'
priority: 2
condition: "count>3"
actions:- "money.set(money.get() - (count-3)*1000);reason.append('迟到大于3次,扣钱:').append((count - 3) * 1000).append(';');"

每个rule之间通过---进行分割,可以在condition和actions下写java代码。 

2、改造接口
@ApiOperation(value = "计算工资")@PostMapping("/getSalary")public BaseResponse getSalary() throws Exception {// 初始化规则引擎RulesEngineParameters parameters = new RulesEngineParameters().skipOnFirstAppliedRule(false);DefaultRulesEngine engine = new DefaultRulesEngine(parameters);engine.registerRuleListener(new MyRuleListener());// 注册规则进入引擎MVELRuleFactory ruleFactory = new MVELRuleFactory(new YamlRuleDefinitionReader());Rules rules = ruleFactory.createRules(new BufferedReader(new InputStreamReader(Objects.requireNonNull(this.getClass().getClassLoader().getResourceAsStream("rules.yml")))));List<Staff> list = staffMapper.selectList(new QueryWrapper<>());for (Staff staff : list) {AtomicDouble money = new AtomicDouble((Double.parseDouble(staff.getMoney())));double beforeMoney = money.get();StringBuffer reason = new StringBuffer();Facts facts = new Facts();facts.put("time", Double.parseDouble(staff.getTime()));facts.put("count", staff.getCount());facts.put("money", money);facts.put("reason", reason);engine.fire(rules, facts);Staff staffNew = staffMapper.selectById(staff.getId());staffNew.setFinalMoney(facts.get("money").toString());staffNew.setDetail(reason.toString());staffMapper.updateById(staffNew);}return RespGenerator.returnOK("成功");}

 

假设当我们调整加班费时(1块钱一个小时),修改配置即可。

name: '有效加班'
description: '加班费的结算'
priority: 2
condition: "time>80"
actions:- "money.set(money.get()+1*(time-80));reason.append('加班费:').append(1*(time-80)).append(';');"

这篇关于【三十四】springboot+easyRule初识规则引擎的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1124519

相关文章

Java NoClassDefFoundError运行时错误分析解决

《JavaNoClassDefFoundError运行时错误分析解决》在Java开发中,NoClassDefFoundError是一种常见的运行时错误,它通常表明Java虚拟机在尝试加载一个类时未能... 目录前言一、问题分析二、报错原因三、解决思路检查类路径配置检查依赖库检查类文件调试类加载器问题四、常见

Java注解之超越Javadoc的元数据利器详解

《Java注解之超越Javadoc的元数据利器详解》本文将深入探讨Java注解的定义、类型、内置注解、自定义注解、保留策略、实际应用场景及最佳实践,无论是初学者还是资深开发者,都能通过本文了解如何利用... 目录什么是注解?注解的类型内置注编程解自定义注解注解的保留策略实际用例最佳实践总结在 Java 编程

Java 实用工具类Spring 的 AnnotationUtils详解

《Java实用工具类Spring的AnnotationUtils详解》Spring框架提供了一个强大的注解工具类org.springframework.core.annotation.Annot... 目录前言一、AnnotationUtils 的常用方法二、常见应用场景三、与 JDK 原生注解 API 的

Java controller接口出入参时间序列化转换操作方法(两种)

《Javacontroller接口出入参时间序列化转换操作方法(两种)》:本文主要介绍Javacontroller接口出入参时间序列化转换操作方法,本文给大家列举两种简单方法,感兴趣的朋友一起看... 目录方式一、使用注解方式二、统一配置场景:在controller编写的接口,在前后端交互过程中一般都会涉及

Java中的StringBuilder之如何高效构建字符串

《Java中的StringBuilder之如何高效构建字符串》本文将深入浅出地介绍StringBuilder的使用方法、性能优势以及相关字符串处理技术,结合代码示例帮助读者更好地理解和应用,希望对大家... 目录关键点什么是 StringBuilder?为什么需要 StringBuilder?如何使用 St

使用Java将各种数据写入Excel表格的操作示例

《使用Java将各种数据写入Excel表格的操作示例》在数据处理与管理领域,Excel凭借其强大的功能和广泛的应用,成为了数据存储与展示的重要工具,在Java开发过程中,常常需要将不同类型的数据,本文... 目录前言安装免费Java库1. 写入文本、或数值到 Excel单元格2. 写入数组到 Excel表格

Java并发编程之如何优雅关闭钩子Shutdown Hook

《Java并发编程之如何优雅关闭钩子ShutdownHook》这篇文章主要为大家详细介绍了Java如何实现优雅关闭钩子ShutdownHook,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起... 目录关闭钩子简介关闭钩子应用场景数据库连接实战演示使用关闭钩子的注意事项开源框架中的关闭钩子机制1.

Maven中引入 springboot 相关依赖的方式(最新推荐)

《Maven中引入springboot相关依赖的方式(最新推荐)》:本文主要介绍Maven中引入springboot相关依赖的方式(最新推荐),本文给大家介绍的非常详细,对大家的学习或工作具有... 目录Maven中引入 springboot 相关依赖的方式1. 不使用版本管理(不推荐)2、使用版本管理(推

Java 中的 @SneakyThrows 注解使用方法(简化异常处理的利与弊)

《Java中的@SneakyThrows注解使用方法(简化异常处理的利与弊)》为了简化异常处理,Lombok提供了一个强大的注解@SneakyThrows,本文将详细介绍@SneakyThro... 目录1. @SneakyThrows 简介 1.1 什么是 Lombok?2. @SneakyThrows

在 Spring Boot 中实现异常处理最佳实践

《在SpringBoot中实现异常处理最佳实践》本文介绍如何在SpringBoot中实现异常处理,涵盖核心概念、实现方法、与先前查询的集成、性能分析、常见问题和最佳实践,感兴趣的朋友一起看看吧... 目录一、Spring Boot 异常处理的背景与核心概念1.1 为什么需要异常处理?1.2 Spring B