goovy简介

Goovy是一种基于Java虚拟机(JVM)的动态编程语言,它结合了Python、Ruby和Smalltalk的许多强大特性,并能够与Java完美结合,使用Java所有的库。以下是关于Goovy的详细解释:

语法

Goovy的语法非常简洁明了,类似于Python和Ruby。它支持动态类型和静态类型,可以在运行时进行类型检查和推断。Goovy还支持闭包、高阶函数和匿名函数等特性,这些特性使得代码更加简洁、易于阅读和维护。

类型系统

Goovy的类型系统非常灵活,支持动态类型和静态类型。在动态类型的情况下,变量的类型会在运行时自动推断。在静态类型的情况下,变量的类型需要在编译时声明。Goovy还支持类型转换和类型检查,使得代码更加健壮和可读。

闭包

Goovy支持闭包,闭包可以捕获其创建时所在的词法作用域中的变量。闭包可以作为参数传递给函数,也可以作为函数的返回值。闭包使得代码更加简洁、灵活和可重用。

高阶函数

Goovy支持高阶函数,高阶函数是指接受其他函数作为参数或返回函数的函数。高阶函数使得代码更加简洁、易于阅读和维护。

反射

Goovy支持反射,反射是指在运行时检查程序的行为和结构的能力。通过反射,可以在运行时获取类的信息、创建对象、调用方法等。反射使得代码更加灵活和可扩展。

并发编程

Goovy支持并发编程,它提供了类似于Java的线程和锁等机制,同时也提供了类似于Python的协程和异步编程等机制。这些机制使得开发人员可以更加轻松地编写并发程序,提高程序的性能和可靠性。

集成Java库

Goovy可以无缝集成Java库,开发人员可以使用Java所有的库来编写Goovy代码。这使得开发人员可以更加方便地利用现有的Java资源,同时也使得Goovy成为了一种非常灵活的动态脚本语言。

应用场景

Goovy适用于各种应用场景,例如Web开发、桌面应用、移动应用、游戏开发等。它既可以作为主要开发语言使用,也可以作为辅助语言使用,与Java完美结合,提高开发效率和代码质量。

总之,Goovy是一种非常强大的动态编程语言,它结合了Python、Ruby和Smalltalk的许多强大特性,并能够与Java完美结合,使用Java所有的库。Goovy的语法简洁明了、类型灵活、支持闭包和高阶函数等特性,使得代码更加简洁、易于阅读和维护。同时,Goovy还支持反射和并发编程等机制,使得程序更加灵活和可扩展。Goovy可以无缝集成Java库,使得开发人员可以更加方便地利用现有的Java资源。总之,Goovy是一种非常优秀的编程语言,适用于各种应用场景,值得广大开发人员学习和使用。

引入pom依赖

在maven项目中pom.xml添加如下依赖

<dependency>
            <groupId>org.codehaus.groovy</groupId>
            <artifactId>groovy-all</artifactId>
            <version>2.4.11</version>
        </dependency>

代码实现

创建一个名为ScriptProvider的groovy脚本辅助类,实现java调用groovy脚本中的方法。完整代码如下:

import groovy.lang.GroovyClassLoader;
import groovy.lang.GroovyObject;
import lombok.extern.slf4j.Slf4j;
import org.springblade.coalface.modules.opcua.dto.OpcData;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;

import java.io.File;
import java.io.IOException;
import java.util.List;

/**
 * @author tarzan
 */
@Component
@Slf4j
public class ScriptProvider {

    private GroovyObject groovyObject;

    @Value("${app.work-face}")
    private  void loadScript(String name){
        try (GroovyClassLoader classLoader = new GroovyClassLoader()) {
            Class<?> groovyClass = classLoader.parseClass(new File("etc/"+name+".groovy"));
            groovyObject= (GroovyObject) groovyClass.newInstance();
        } catch (InstantiationException | IOException | IllegalAccessException e) {
            log.error(e.getMessage());
        }
    }

    public  void overloadScript(String filePath){
        try (GroovyClassLoader classLoader = new GroovyClassLoader()) {
            Class<?> groovyClass = classLoader.parseClass(new File(filePath));
            groovyObject= (GroovyObject) groovyClass.newInstance();
        } catch (InstantiationException | IOException | IllegalAccessException e) {
            log.error(e.getMessage());
        }
    }

    public float calStentHeight(int i, List<OpcData> dataValues){
        Object[] objects = new Object[]{i,dataValues};
        Object result=groovyObject.invokeMethod("calStentHeight",objects);
        return Float.parseFloat(result.toString());
    }

    public float revisedStentHeight(int i, float value){
        Object[] objects = new Object[]{i,value};
        Object result=groovyObject.invokeMethod("revisedStentHeight",objects);
        return Float.parseFloat(result.toString());
    }

    public float revisedStentStroke(float value) {
        Object[] objects = new Object[]{value};
        Object result=groovyObject.invokeMethod("revisedStentStroke",objects);
        return Float.parseFloat(result.toString());
    }

}
  • app.work-face 是在spring配置文件中的goovy脚本名称
  • etc/“+name+”.groovy 是指在项目根目录下 etc文件夹下 的某个 goovy脚本

groovy脚本代码示例

import org.springblade.coalface.modules.opcua.dto.OpcData

def calStentHeight(int i,List<OpcData> dataValues){
     float d= dataValues.get(i).getValue()
     def f1=0.175
     def f2=0.136
     def e= new BigDecimal(Float.toString(d)).add(f1).add(f2)
    return e.doubleValue()
}

def revisedStentHeight(int i,float height){
    int stentNo=i+1
    if(stentNo<=2||stentNo>=92){
        height=height>3.3||height<1.15?3.3:height
    }else{
        if(height>1.9){
            height=1.9
        }
        if(height<1.15){
            height=1.2
        }
    }
    return height
}

def revisedStentStroke(float stroke){
    if(stroke>900F){
        stroke=900F
    }
    if(stroke<=0F){
        stroke=0F
    }
    return stroke
}
  • groovy 脚本中可以直接引入java的类,使用java对象的方法和属性

调用代码示例

@Resource
  private ScriptProvider scriptProvider;

 
  private void test() {
      float revised = scriptProvider.revisedStentStroke(modification);    
  }
  • 注入ScriptProvider工具类
  • 通过ScriptProvider工具类调用对应grovvy脚本中的方法

动态监听脚本

当脚本被修改时候,java代码重新加载groovy脚本中的最新代码,需要一个监听工具类FileSystemWatcher,完整代码如下:

import lombok.AllArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springframework.scheduling.annotation.Async;
import org.springframework.stereotype.Component;

import java.io.IOException;
import java.nio.file.*;

/**
 * @author tarzan
 */
@Component
@AllArgsConstructor
@Slf4j
public class FileSystemWatcher {

    private final ScriptProvider scriptProvider;

    @Async
    public void modifyWatch(){
        try {
            watch(StandardWatchEventKinds.ENTRY_MODIFY);
        } catch (IOException | InterruptedException e) {
            log.error(e.getMessage());
        }
    }

    @SuppressWarnings({"unchecked", "BusyWait"})
    private  void watch(WatchEvent.Kind<Path> eventKind) throws IOException, InterruptedException {
        // 定义你想要监听的路径
        Path path = Paths.get("etc");
        // 创建 WatchService
        WatchService watchService = FileSystems.getDefault().newWatchService();
        // 将路径注册到 WatchService,并指定你想要监听的事件类型
        path.register(watchService, eventKind);

        while (true) {
            // 获取下一个文件系统事件
            WatchKey key = watchService.take();
            for (WatchEvent<?> event : key.pollEvents()) {
                // 获取事件类型
                WatchEvent.Kind<?> kind = event.kind();
                if (kind == StandardWatchEventKinds.OVERFLOW) {
                    continue;
                }
                // 获取发生事件的文件
                WatchEvent<Path> ev = (WatchEvent<Path>) event;
                Path fileName = ev.context();
                // 打印出发生事件的文件名和事件类型
                boolean eventFlag=!fileName.toFile().getName().endsWith("~");
                if(eventFlag){
                    log.info("etc/"+fileName +" be modified");
                    scriptProvider.overloadScript("etc/"+fileName);
                }
            }
            // 重置 WatchKey,以便接收下一个事件
            boolean valid = key.reset();
            if (!valid) {
                break;
            }
            //睡眠1秒
            Thread.sleep(1000);
        }
    }

}
  • @SuppressWarnings 用于抑制编译器产生某些警告,@SuppressWarnings({“unchecked”, “BusyWait”})告诉编译器不要对未检查的类型转换和忙等待产生警告。
  • 这个监听工具类,需要在项目启动的时候调用它的监听方法。下面是代码示例:

创建一个启动监听类 AppStartedListener ,当项目启动完成后,就会调用,groovy脚本文件修改事件,当groovy脚本内的代码被修改后,保存groovy脚本时,会触发更新ScriptProvider类中加载的groovy脚本代码,实现动态更新。

import lombok.SneakyThrows;
import lombok.extern.slf4j.Slf4j;
import org.springblade.coalface.script.FileSystemWatcher;
import org.springframework.boot.context.event.ApplicationStartedEvent;
import org.springframework.context.ApplicationListener;
import org.springframework.lang.NonNull;
import org.springframework.stereotype.Component;

import javax.annotation.Resource;


/**
 *
 * @author tarzan
 * @date 2023-10-05
 */
@Slf4j
@Component
public class AppStartedListener implements ApplicationListener<ApplicationStartedEvent> {

    @Resource
    private FileSystemWatcher fileSystemWatcher;

    @SneakyThrows
    @Override
    public void onApplicationEvent(@NonNull ApplicationStartedEvent event) {
        fileSystemWatcher.modifyWatch();
        log.info("服务初始化。。。");
    }

}