Spring boot装载模板代码工程实践问题
Spring boot装载模板代码工程中,动态编译,在Rancher真实上线时,遇到了两大问题,第一,JDK基础镜像都是采用openJDK1.8的镜像,1.8还没有组合tools.jar工具包;第二,JavaCompiler将字符串文件流加入compiler队列进行编译时,出现程序包无法找到问题。幸甚,功夫不负有心啊,进过摸索终于解决了以上两大难题,使得项目如意上线并且运行良好,读取MySQL配置,根据字段映射规则完整自动入库,减少大量重复工作。
动态化补充
项目已经完成RocketMQ动态化配置,因成本问题,后期引入Kafka,需要对Kafka进行动态化配置,阅读了Spring-Kafka官方文档后,发现了@KafkaListener本身就支持动态化配置,👍👍👍!
@KafkaListener
@Autowired
KafkaConsumerService kafkaConsumerService;
@KafkaListener(topics = "#{__listener.topic}", groupId = "#{__listener.group}", containerFactory = "kafkaListenerContainerFactory")
public void listener(List<ConsumerRecord<String, String>> messages, Acknowledgment ack) throws Exception {
String infoMsg = String.format("【入库配置化流程】kafka消费 topic=%s,partitionOffset=%s,size=%s",
messages.get(0).topic(),
getPartitionOffsetListJsonStr(messages),
messages.size());
log.info(infoMsg);
try {
// 入库处理
ack.acknowledge();
} catch (Exception e) {
log.error("【入库配置化流程】消费异常={}", ExceptionUtils.getStackTrace(e));
throw e;
}
}
其它问题
JDK版本问题及解决
此问题会导致ToolProvider.getSystemJavaCompiler()返回NULL,无法调用Java的编译器,解决方案有两种:
1、修改jdk基础镜像,将tools.jar导入到基础镜像并重新上传,修改Dockerfile文件的FROM(运维允许基础镜像定制化的话)
2、查阅ToolProvider.getSystemJavaCompiler()源码,发现底层是直接读取lib下的tools.jar中的com.sun.tools.javadoc.api.JavadocTool的Class,那就简单了,直接在pom导入tools.jar,将关键代码ToolProvider.getSystemJavaCompiler()替换为(JavaCompiler) Class.forName(“com.sun.tools.javac.api.JavacTool”).newInstance();
tools.jar引入可以:
<dependency>
<groupId>com.perfma.wrapped</groupId>
<artifactId>com.sun.tools</artifactId>
<version>1.8.0_jdk8u275-b01_linux_x64</version>
</dependency>
记住在spring-boot-maven-plugin下添加如下配置:
<configuration>
<includeSystemScope>true</includeSystemScope>
</configuration>
同样也可以将tools.jar放入resources下:
<dependency>
<groupId>jdk.tools</groupId>
<artifactId>jdk.tools</artifactId>
<version>1.8</version>
<scope>system</scope>
<systemPath>${project.basedir}/src/main/resources/lib/tools.jar</systemPath>
</dependency>
动态编译失败,出现程序包无法找到问题
一开始,代码在本地启动运行都没什么问题,但是打包之后,用java -jar运行就无法实现动态编译。在网上搜罗的一大堆,都没有解决办法,想着本地每次运行都很正常,就查看了一下本地启动的java -jar参数,发现了-classpath里面将所有的第三方jar都写上去了,瞬间觉醒,当spring boot打成jar运行时,是有JarLaunch的编译约定,也就是说在spring boot运行时会自动读取\BOOT-INF\lib下的所有jar,这样编译就能通过并且运行,compiler.getTask(null, classJavaFileManager, null, null, null,
Collections.singletonList(stringObject));正好提供了指定classpath的参数(Iterable options),然后将IDEA的java -jar启动参数复制到optins上,即:
List<String> options = new ArrayList<>();
options.add("-classpath");
options.add("D:\\repo\\org\\springframework\\boot\\spring-boot-starter-web\\2.3.0.RELEASE\\spring-boot-starter-web-2.3.0.RELEASE.jar;D:\\repo\\org\\springframework\\boot\\spri...");
JavaCompiler.CompilationTask task = compiler.getTask(null, classJavaFileManager, null, options, null,
Collections.singletonList(stringObject));
打包,java -jar启动,成功!验证了解决方案的可行性,接下来就是读取jar内文件并生成文件列表的问题。
PS.参考文档已忘…感谢为此解决方案提供帮助的推文和博客