一灰灰blog 一灰灰blog
首页
  • InfluxDB
  • MongoDB
  • MySql
  • 基础系列
  • DB系列
  • 搜索系列
  • MQ系列
  • WEB系列
  • 中间件
  • 运维
  • SpringSecurity
  • SpringCloud
  • QuickAlarm
  • QuickCrawer
  • QuickFix
  • QuickMedia
  • QuickSpi
  • QuickTask
  • 分类
  • 标签
  • 归档
  • 收藏
  • 关于
GitHub (opens new window)

一灰灰blog

资深搬运工
首页
  • InfluxDB
  • MongoDB
  • MySql
  • 基础系列
  • DB系列
  • 搜索系列
  • MQ系列
  • WEB系列
  • 中间件
  • 运维
  • SpringSecurity
  • SpringCloud
  • QuickAlarm
  • QuickCrawer
  • QuickFix
  • QuickMedia
  • QuickSpi
  • QuickTask
  • 分类
  • 标签
  • 归档
  • 收藏
  • 关于
GitHub (opens new window)
  • QuickAlarm

  • QuickCrawer

  • QuickFix

  • QuickMedia

  • QuickSpi

  • QuickTask

    • QuickTask动态脚本支持框架整体介绍篇
    • Quick-Task 动态脚本支持框架之使用介绍篇
    • Quick-Task 动态脚本支持框架之结构设计篇
    • Quick-Task 动态脚本支持框架之任务动态加载
    • Quick-Task 动态脚本支持框架之Groovy脚本加载执行
      • I. Groovy文件动态加载
        • 1. 依赖
        • 2. 加载Groovy
        • 3. 调用包装
        • 4. 小结
      • II. 其他
        • 0. 相关
        • 1. 一灰灰Blog: https://liuyueyi.github.io/hexblog
        • 2. 声明
        • 3. 扫描关注
  • Quick开源系列
  • QuickTask
一灰灰
2018-08-07

Quick-Task 动态脚本支持框架之Groovy脚本加载执行

上一篇简答说了如何判断有任务动态添加、删除或更新,归于一点就是监听文件的变化,判断目录下的Groovy文件是否有新增删除和改变,从而判定是否有任务的变更;

接下来的问题就比较明显了,当任务变更之后,就需要重新加载任务了,即如何动态的编译并执行Groovy文件呢?

相关系列博文:

  • 180628-Quick-Task 动态任务执行框架想法篇 (opens new window)
  • 180702-Quick-Task 动态脚本支持框架整体介绍篇 (opens new window)
  • 180723-Quick-Task 动态脚本支持框架之结构设计篇 (opens new window)
  • 180729-Quick-Task 动态脚本支持框架之任务动态加载 (opens new window)

# I. Groovy文件动态加载

要想动态加载类,可以怎么办?如果对JVM有一定了解的朋友可能知道,自定义一个ClassLoader,可以实现从文件/网络/DB/Jar包中读取class文件,而Groovy,动态语言,简单来说就是.groovy文件可以直接运行,那么我们编码中要怎么玩?

# 1. 依赖

让我自己来实现Groovy文件的编译执行,目前基本上是看不到啥希望的,所以果断的借助第三方工具类加载Groovy文件

pom文件添加依赖

<dependency>
    <groupId>org.codehaus.groovy</groupId>
    <artifactId>groovy-all</artifactId>
    <version>2.4.3</version>
</dependency>
1
2
3
4
5

# 2. 加载Groovy

直接利用上面jar包中提供的GroovyCalssLoader来加载Groovy文件即可,使用也比较简单

@Slf4j
public class GroovyCompile {

    @SuppressWarnings("unchecked")
    public static <T> T compile(File codeSource, Class<T> interfaceType, ClassLoader classLoader)
            throws CompileTaskScriptException {
        try {
            GroovyClassLoader loader = new GroovyClassLoader(classLoader);
            Class clz = loader.parseClass(codeSource);

            // 接口校验
            if (!interfaceType.isAssignableFrom(clz)) {
                throw new CompileTaskScriptException("illegal script type!");
            }

            return (T) clz.newInstance();
        } catch (IOException e) {
            log.error("load code from {} error! e: {}", codeSource, e);
            throw new CompileTaskScriptException("load code from " + codeSource + " error!");
        } catch (CompileTaskScriptException e) {
            throw e;
        } catch (Exception e) {
            log.error("initial script error! codePath: {}, e: {}", codeSource, e);
            throw new CompileTaskScriptException(
                    "initial script error! clz: " + codeSource + " msg: " + e.getMessage());
        }
    }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28

上面看着挺多,关键地方就三行,编译为class对象之后,借助反射来创建对象

GroovyClassLoader loader = new GroovyClassLoader(classLoader);
Class clz = loader.parseClass(codeSource);
return (T) clz.newInstance();
1
2
3

另外还有一行,也可以顺带凑一眼,判断一个class是否为另一个class的子类,用的是

interfaceType.isAssignableFrom(clz)
1

而判断某个对象是否为某类的子类用的则是 instance of

# 3. 调用包装

上面既然提供了一个工具类,那么接上篇的获取变动文件之后,获取File对象,借此拿到任务对象,就比较清晰了

@Slf4j
public class ScriptLoadUtil {

    public static ITask loadScript(File file) {
        try {
            return GroovyCompile.compile(file, ITask.class, ScriptLoadUtil.class.getClassLoader());
        } catch (CompileTaskScriptException e) {
            log.error("un-expect error! e: {}", e);
            return null;
        }
    }
}
1
2
3
4
5
6
7
8
9
10
11
12

# 4. 小结

本篇内容比较简单,知识点也没多少,一个是利用GroovyClassLoader来编译Groovy文件并获取实例;另一个就是如何判断一个class是否为另一个class的子类

还有一个隐藏的点上面没有说,那就是上面的GroovyCompile文件中,每次加载Groovy文件时,都是新创建了一个GroovyClassLoader,并由它来加载并实例Groovy任务,那么问题来了

  • 能否用一个GoorvyClassLoader来管理所有的Groovy任务呢?
  • 上面的代码实现中,不同的Groovy任务之间,可以相互通信么?

针对上面的问题,暂不给出答案,后面再说

# II. 其他

# 0. 相关

博文:

  • 180628-Quick-Task 动态任务执行框架想法篇 (opens new window)
  • 180702-Quick-Task 动态脚本支持框架整体介绍篇 (opens new window)
  • 180723-Quick-Task 动态脚本支持框架之结构设计篇 (opens new window)
  • 180729-Quick-Task 动态脚本支持框架之任务动态加载 (opens new window)

项目:

  • https://github.com/liuyueyi/quick-task (opens new window)

# 1. 一灰灰Blog (opens new window): https://liuyueyi.github.io/hexblog

一灰灰的个人博客,记录所有学习和工作中的博文,欢迎大家前去逛逛

# 2. 声明

尽信书则不如,已上内容,纯属一家之言,因个人能力有限,难免有疏漏和错误之处,如发现bug或者有更好的建议,欢迎批评指正,不吝感激

  • 微博地址: 小灰灰Blog (opens new window)
  • QQ: 一灰灰/3302797840

# 3. 扫描关注

一灰灰blog

QrCode

知识星球

goals

编辑 (opens new window)
#技术方案
上次更新: 2021/10/15, 19:56:22
Quick-Task 动态脚本支持框架之任务动态加载

← Quick-Task 动态脚本支持框架之任务动态加载

最近更新
01
【WEB系列】从0到1实现自定义web参数映射器
01-23
02
【WEB系列】如何支持下划线驼峰互转的传参与返回
01-17
03
【DB系列】Mybatis之批量插入的几种姿势
01-11
更多文章>
Theme by Vdoing | Copyright © 2017-2022 一灰灰Blog
MIT License | 鄂ICP备18017282号 |
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式
×