在 Java 低代码平台中,Liquor 充当了运行时 JIT 编译器和规则引擎。它弥合了"配置"与"高性能 Java 运行时"之间的鸿沟。它解决了 Java 生态中热更新、高性能、强类型三者难以兼得的根本矛盾。
Liquor (动态编译后)就是 Java 原生运行,比一般的 JVM 脚本(或表达式)性能高 "20倍" 左右。
一、突破 Java 静态编译的限制:实现配置即代码的热生效
低代码平台的核心价值在于业务敏捷性,即用户在界面上的配置(流程、规则、公式、代码)必须实时生效,而传统的 Java 开发模式是强静态编译的,新代码需要重新编译、打包和重启。
1. 动态编译即服务 (Dynamic Compilation-as-a-Service)
Liquor 的 DynamicCompiler 模块将编译能力从开发工具链中解放出来,变为一个运行时的服务能力。
- 实现机制: 低代码平台的前端配置(如拖拽流程、设置数据源映射)经过后端解析器,被转化为标准的 Java 代码片段。Liquor 在运行时调用 Java Compiler API 将这些代码编译成字节码。
- 技术价值: 平台无需依赖外部的编译环境,也不需要进行耗时的重启。新业务逻辑直接在内存中生成、加载并执行,实现了业务逻辑的零停机热更新。
2. 代码示例:流程节点的动态实现
java
// 低代码平台解析用户配置,生成一个 Java 类或方法
String customLogicCode = """
import com.platform.utils.DataConverter;
import java.util.Map; // 增加 Map 导入,使代码更完整
public class CustomProcessor {
public Map<String, Object> process(Map<String, Object> inputData) {
if ((double)inputData.get("amount") > 1000) {
inputData.put("status", "HighValue");
}
// 调用平台内置工具类
inputData.put("convertedKey", DataConverter.hashKey(inputData.get("originalKey")));
return inputData;
}
}
""";
// Liquor 运行时编译和加载
DynamicCompiler compiler = new DynamicCompiler();
compiler.addSource("CustomProcessor", customLogicCode).build();
// 实例化并执行 (模拟流程引擎调用)
Class<?> clazz = compiler.getClassLoader().loadClass("CustomProcessor");
Object processor = clazz.getDeclaredConstructor().newInstance();
// ... 调用 processor.process(data) ...
二、利用字节码优势:高性能的表达式与规则引擎
低代码平台的核心逻辑往往是大量的表达式求值和业务规则判断。采用脚本解释器(如 JavaScript、Groovy 等脚本引擎)或基于反射的规则引擎(如 MVEL/OGNL)在性能上存在固有瓶颈。
1. 高性能的字节码缓存 (Bytecode Caching)
Liquor 的 liquor-eval 模块将复杂的表达式和脚本编译成 Java 字节码,并通过 LRU 缓存 存储编译结果。
- 实现机制: 对于重复出现的表达式(例如:a * (b + 1)),Liquor 只进行一次编译。后续求值直接加载缓存中的字节码,然后使用 LiquorEvaluator 执行。
- 技术价值: 在低代码平台的高并发场景中,大多数表达式都是重复的。Liquor 避免了重复编译和解释的开销,使得规则引擎的执行性能逼近原生 Java 方法调用,从根本上解决了性能瓶颈。
2. 代码示例:高性能表达式求值 (Exprs 应用)
在数据校验和计算字段场景中,保证计算速度至关重要:
java
// 1. 定义一个用于计算的表达式
CodeSpec calculationSpec = new CodeSpec("basePrice * (1 + taxRate) + shippingFee")
.parameters(
new ParamSpec("basePrice", Double.class),
new ParamSpec("taxRate", Double.class),
new ParamSpec("shippingFee", Double.class)
);
// 2. 首次执行(触发编译并缓存)
Map<String, Object> data1 = Map.of("basePrice", 100.0, "taxRate", 0.1, "shippingFee", 5.0);
Double total1 = Exprs.eval(calculationSpec, data1); // 首次编译
// 3. 第二次执行(直接命中缓存,极快)
Map<String, Object> data2 = Map.of("basePrice", 200.0, "taxRate", 0.1, "shippingFee", 5.0);
Double total2 = Exprs.eval(calculationSpec, data2); // 直接执行字节码
三、构建动态扩展点的基石:动态 Bean 与 IoC 集成
低代码平台需要提供灵活的扩展点,允许开发者嵌入自定义服务或覆写默认行为。Liquor 提供的动态加载机制是实现这一目标的关键。
1. 动态类加载与 IoC 容器集成 (Dynamic Class Loading & IoC)
Liquor 编译生成的类由其内部的 DynamicClassLoader 负责加载。
- 实现机制:
平台利用 Liquor 的集成特性(如对 Spring 或 Solon 的支持),将动态编译的类(例如一个带有 @Component 注解的 Service 类)注册到 IoC (Inversion of Control) 容器中。
- 技术价值:
动态依赖注入: 动态生成的代码可以直接 @Autowired 或 @Inject 平台中已有的服务 Bean。
动态 API 暴露: 动态编译的 Controller 类可以即时映射到 URL 路径,实现 动态 API 接口的发布,为低代码平台提供了灵活的微服务扩展能力。
2. 安全隔离与沙箱的配合
动态编译虽然强大,但也带来了安全风险。Liquor 的类加载机制为安全沙箱提供了基础。
- 实现机制:
平台可以通过拦截 Liquor 的 DynamicClassLoader,结合 Java 的 SecurityManager 或自定义的白名单/黑名单机制,限制动态代码能够访问的包、类和方法。
- 技术价值:
确保了用户编写或平台自动生成的代码不会恶意或意外地访问敏感系统资源(如文件系统、网络连接),保障了核心平台的安全与稳定。
结论
在 Java 低代码平台的底层实现中,Liquor 框架的意义:
它是连接配置层和高性能运行时的桥梁。 它将 Java 编译能力服务化、运行时化,使平台能够以 Java 级别的性能、强类型语言的健壮性,去实现 脚本语言的热更新和灵活性。这对于追求高性能、高并发、且基于 Java 技术的低代码平台来说,是一个不可或缺的动态引擎。