java实现动态编译并动态加载

 更新时间:2021年04月14日 23:45:00   作者:扬帆舟  
这篇文章主要介绍了java实现动态编译并动态加载,需要的朋友可以参考下

在D盘test目录下有个java文件:AlTest.java

public class AlTest { 
	public String sayHello(){
		System.out.println("AlTest类 sayHello()方法正在执行....");
		return "hello word";
	}
}

现需要实现在工程已经运行过程中,进行java文件到class文件的编译操作,并运行AlTest类的方法

package com.piao.job;
 
import java.lang.reflect.Method;
import javax.tools.JavaCompiler;
import javax.tools.ToolProvider;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Configurable;
import org.springframework.scheduling.annotation.EnableScheduling;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;
 
@Component
@Configurable
@EnableScheduling
public class CompilerJob {
 
  private static final Logger logger = LoggerFactory.getLogger(CompilerJob.class);
 
  private static boolean isExecute = false;
 
  /**
   * 任务:job test
   */
  @Scheduled(cron = "*/10 * * * * * ")
  public void test2() {
    try {
       if (isExecute) {
         return;
       }
       isExecute = true;		//只是测试,所以只执行一次
			
	   complierAndRun();
	} catch (Exception e) {
	   logger.error("test", e);
	}
  }
	
 public void complierAndRun(){
   try {
			
	 System.out.println(System.getProperty("user.dir"));
	 //动态编译
	 JavaCompiler javac = ToolProvider.getSystemJavaCompiler();
	 int status = javac.run(null, null, null, "-d", System.getProperty("user.dir")+"\\target\\classes","D:/test/AlTest.java");
	 if(status!=0){
		 System.out.println("没有编译成功!");
	 }
			
	 //动态执行
	 Class clz = Class.forName("AlTest");//返回与带有给定字符串名的类 或接口相关联的 Class 对象。
	 Object o = clz.newInstance();
	 Method method = clz.getDeclaredMethod("sayHello");//返回一个 Method 对象,该对象反映此 Class 对象所表示的类或接口的指定已声明方法
	 String result= (String)method.invoke(o);//静态方法第一个参数可为null,第二个参数为实际传参
	 System.out.println(result);
	 } catch (Exception e) {
		 logger.error("test", e);
	 }
  }
}

运行结果:

E:\zhoufy\small\piao-admin

AlTest类 sayHello()方法正在执行....

hello word

其中代码:

 int status = javac.run(null, null, null, "-d", System.getProperty("user.dir")+"\\target\\classes","D:/test/AlTest.java");

把class文件生成到了当前工程目录下的classes目录(E:\zhoufy\small\piao-admin\target\classess)所以classloader是可以加载到的,如果想知道是哪个类加载器:

Class clz = Class.forName("AlTest");//返回与带有给定字符串名的类 或接口相关联的 Class 对象。
Object o = clz.newInstance();
System.out.println(clz.getClassLoader().getSystemClassLoader());

打印的是: sun.misc.Launcher$AppClassLoader@4e0e2f2a 说明使用的是AppClassLoader

当然也可以生成到Bootstrap ClassLoader可加载的目录下

//生成到工程classes下
//int status = javac.run(null, null, null, "-d", System.getProperty("user.dir")+"\\target\\classes","D:/test/AlTest.java");
			
//生成到BootStrap ClassLoader可加载目录下
int status = javac.run(null, null, null, "-d", "C:\\Program Files\\Java\\jdk1.8.0_65\\jre\\classes","D:/test/AlTest.java");

当然也可以自定义类加载器,把文件生成在指定的外部目录 :

public void complierAndRun(){
		try {
			
			System.out.println(System.getProperty("user.dir"));
			 //动态编译
			JavaCompiler javac = ToolProvider.getSystemJavaCompiler();
			int status = javac.run(null, null, null, "-d", "D:\\","D:/test/AlTest.java");
			if(status!=0){
				System.out.println("没有编译成功!");
			}
			
			//动态执行
			//Class clz = Class.forName("AlTest");//返回与带有给定字符串名的类 或接口相关联的 Class 对象。
			//自定义类加载器的加载路径
			MyClassLoader myClassLoader = new MyClassLoader("D:\\");
			//包名+类名
			Class clz = myClassLoader.loadClass("AlTest");
			Object o = clz.newInstance();
			Method method = clz.getDeclaredMethod("sayHello");//返回一个 Method 对象,该对象反映此 Class 对象所表示的类或接口的指定已声明方法
			String result= (String)method.invoke(o);//静态方法第一个参数可为null,第二个参数为实际传参
			System.out.println(result);
		} catch (Exception e) {
			logger.error("test", e);
		}
	}

java动态执行代码的代码, java eval

public class ScriptUtils {
     
    private static final Logger logger = LoggerFactory.getLogger(ScriptUtils.class);
     
    /**
     * 
     * <p>执行字符串计算</p>
     * @param express
     * @param params
     * @return
     * @throws ScriptException 
     */
    @SuppressWarnings("unchecked")
    public static <T, E> E eval(String express, Map<String, T> params) throws ScriptException{
        ScriptEngineManager manager = new ScriptEngineManager();  
        ScriptEngine engine = manager.getEngineByName("js");
        if(params == null){
            params = new HashMap<String,T>();
        }
        Iterator<Map.Entry<String, T>> iter = params.entrySet().iterator();
        Map.Entry<String, T> entry = null;
        while(iter.hasNext()){
            entry = iter.next();
            engine.put(entry.getKey(), entry.getValue());
        }
        E result = null;
        try {
            result = (E)engine.eval(express);
        } catch (ScriptException e) {
            logger.warn("表达式执行异常: " + e.getMessage());
        } 
        return result;
    }
     
    /**
     * 解析字符串, 并将其当作表达式执行
     * @param express
     * @param params
     * @return
     * @throws ScriptException
     */
    public static <T> Boolean evalBoolean(String express, Map<String, T> params) {
        ScriptEngineManager manager = new ScriptEngineManager();  
        ScriptEngine engine = manager.getEngineByName("js");
        if(params == null){
            params = new HashMap<String,T>();
        }
        Iterator<Map.Entry<String, T>> iter = params.entrySet().iterator();
        Map.Entry<String, T> entry = null;
        while(iter.hasNext()){
            entry = iter.next();
            engine.put(entry.getKey(), entry.getValue());
        }
        Boolean result = null;
        try {
            result = (Boolean)engine.eval(express);
        } catch (ScriptException e) {
            result = false;
            logger.warn("表达式执行异常: " + e.getMessage());
        } 
        return result;
    }

到此这篇关于java实现动态编译并动态加载的文章就介绍到这了,更多相关java动态编译内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • java中ArrayList的两种排序方法实例

    java中ArrayList的两种排序方法实例

    ArrayList是一个数组队列,相当于 动态数组,与Java中的数组相比,它的容量能动态增长,这篇文章主要给大家介绍了关于java中ArrayList的两种排序方法,需要的朋友可以参考下
    2021-07-07
  • Java 实战交易平台项目之宠物在线商城系统

    Java 实战交易平台项目之宠物在线商城系统

    读万卷书不如行万里路,只学书上的理论是远远不够的,只有在实战中才能获得能力的提升,本篇文章手把手带你用Java实现一个宠物在线商城系统,大家可以在过程中查缺补漏,提升水平
    2021-11-11
  • ShardingSphere数据库读写分离算法及测试示例详解

    ShardingSphere数据库读写分离算法及测试示例详解

    这篇文章主要为大家介绍了ShardingSphere数据库读写分离算法及测试示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-03-03
  • eclipse老是自动跳到console解决办法

    eclipse老是自动跳到console解决办法

    eclipse启动服务后,想看一些properties信息或者别的,但老是自动跳转到console页面,本文给大家介绍了解决办法,对大家的学习或工作有一定的帮助,需要的朋友可以参考下
    2024-03-03
  • JAVA环境搭建之MyEclipse10+jdk1.8+tomcat8环境搭建详解

    JAVA环境搭建之MyEclipse10+jdk1.8+tomcat8环境搭建详解

    本文详细讲解了MyEclipse10+jdk1.8+tomcat8的JAVA环境搭建方法,希望能帮助到大家
    2018-10-10
  • springboot中设置定时任务的三种方法小结

    springboot中设置定时任务的三种方法小结

    在我们开发项目过程中,经常需要定时任务来帮助我们来做一些内容,本文介绍了springboot中设置定时任务的三种方法,主要包括@Scheduled注解,Quartz框架和xxl-job框架的实现,感兴趣的可以了解一下
    2023-12-12
  • Java之经典排序算法

    Java之经典排序算法

    这篇文章主要介绍了Java的一些经典排序算法,对Java算法感兴趣的小伙伴可以详细参考阅读本文,对同学们有一定的参考价值
    2023-03-03
  • java实现ReadWriteLock读写锁的示例

    java实现ReadWriteLock读写锁的示例

    ReadWriteLock是Java并发包中的接口,定义了读锁和写锁,读锁允许多线程同时访问共享资源,而写锁则要求独占,这种机制适用于读多写少的场景,可以提高并发效率同时保证数据一致性,本文就来详细的介绍一下如何实现,感兴趣的可以了解一下
    2024-09-09
  • swagger中如何给请求添加header

    swagger中如何给请求添加header

    这篇文章主要介绍了swagger中如何给请求添加header,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-06-06
  • SpringBoot Admin 如何实现Actuator端点可视化监控

    SpringBoot Admin 如何实现Actuator端点可视化监控

    这篇文章主要介绍了SpringBoot Admin 如何实现Actuator端点可视化监控,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-08-08

最新评论