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

网友投稿 291 2023-01-22

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 statuhttp://s = 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);

/**

*

*

执行字符串计算

* @param express

* @param params

* @return

* @throws ScriptException

*/

@SuppressWarnings("unchecked")

public static E eval(String express, Map params) throws ScriptException{

ScriptEngineManager manager = new ScriptEngineManager();

ScriptEngine engine = manager.getEngineByName("js");

if(params == null){

params = new HashMap();

}

Iterator> iter = params.entrySet().iterator();

Map.Entry 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 Boolean evalBoolean(String express, Map params) {

ScriptEngineManager manager = new ScriptEngineManager();

ScriptEngine engine = manager.getEngineByName("js");

if(params == null){

params = nhttp://ew HashMap();

}

Iterator> iter = params.entrySet().iterator();

Map.Entry 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;

}

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:java与C 代码运行效率的对比(整理)
下一篇:Java使用ScriptEngine动态执行代码(附Java几种动态执行代码比较)
相关文章

 发表评论

暂时没有评论,来抢沙发吧~