《一切皆是映射:代码的本质》Java 动态读取源代码,并编译 & 加载执行

动态的执行一段简单代码,采用生成java文件,调用javac编译,反射执行的方式。

使用输入输出流(或者你说的可能是要用反射得到程序结果来解析)解析做出*.Java文件。
然后可以使用runtime调用Dos下的java编译命令编译取得class文件。
然后使用classloader,反射等组合执行生成的class文件。

package loadjarclass;
 
import java.io.File;
import java.lang.reflect.Method;
import java.net.URL;
import java.net.URLClassLoader;
 
import org.junit.Test;
 
 
public class LoadJarClassTest {
    
    @Test
    public void testLoadClass() throws Exception{
        /*动态加载指定类*/
        File file=new File("D:/test");//类路径(包文件上一层)
        URL url=file.toURI().toURL();
        ClassLoader loader=new URLClassLoader(new URL[]{url});//创建类加载器
        //import com.sun.org.apache.bcel.internal.util.ClassLoader;
        //ClassLoader classLoader = new ClassLoader(new String[]{""});//类路径
        Class<?> cls=loader.loadClass("loadjarclass.TestTest");//加载指定类,注意一定要带上类的包名
        Object obj=cls.newInstance();//初始化一个实例
        Method method=cls.getMethod("printString",String.class,String.class);//方法名和对应的参数类型
        Object o=method.invoke(obj,"chen","leixing");//调用得到的上边的方法method
        System.out.println(String.valueOf(o));//输出"chenleixing"
        
        /*动态加载指定jar包调用其中某个类的方法*/
        file=new File("D:/test/commons-lang3.jar");//jar包的路径
        url=file.toURI().toURL();
        loader=new URLClassLoader(new URL[]{url});//创建类加载器
        cls=loader.loadClass("org.apache.commons.lang3.StringUtils");//加载指定类,注意一定要带上类的包名
        method=cls.getMethod("center",String.class,int.class,String.class);//方法名和对应的各个参数的类型
        o=method.invoke(null,"chen",Integer.valueOf(10),"0");//调用得到的上边的方法method(静态方法,第一个参数可以为null)
        System.out.println(String.valueOf(o));//输出"000chen000","chen"字符串两边各加3个"0"字符串
    }
 
}

使用com.sun.tools.javac.Main编译Java源代码的,脚本如下。就研究了一番,写了个demo,记录一下,也方便后来人学习。

$ bin/hadoop com.sun.tools.javac.Main WordCount.java
$ jar cf wc.jar WordCount*.class
com.sun.tools.javac.Main 

这个类位于${JAVA_HOME}/lib/tools.jar中,需要添加到classpath中或者直接在IDE中把它引入。这个方式跟直接调用javac命令效果是一样。下面是demo,使用Main类中的compile方法编译一个Person.java源文件后,再加载字节码进行执行。

1、准备待编译的java源代码。

下面代码是一个简单的PersonAction,实现了一个行动接口Action。实现接口不是必须的,只是后面方便实例化一个有具体类型对象才用的。

import inf.Action;
 
public class PersonAction implements Action{
 
    @Override
    public void say(String msg){
        System.out.println("Person say a message: "+msg);
    }
}

package inf;
 
public interface Action {
    public void say(String msg);
}

2、编写执行的代码,该代码用来编译PersonAction.java,编译成功后并加载字节码到JRE中进行执行


package demo;  
  
import inf.Action;  
  
import java.io.*;  
import java.lang.reflect.Method;  
  
/** 
 * Created by rns on 17-1-7. 
 */  
public class DynamicCompiler {  
    public static void main(String[] args) throws IOException {  
        //待编译的源代码放置的文件夹路径  
        String basedir = "/home/rns/Desktop/test/";  
        //待编译的类名称,不包含.java  
        String classname = "PersonAction";  
        //执行代码的路径,下面的路径是本人的idea编译后输出路径  
        String executedir = "/home/rns/IdeaProjects_community/"  
                +"DynamicCompileAndRun/out/production/DynamicCompileAndRun/";  
        //创建编译器  
        com.sun.tools.javac.Main javac = new com.sun.tools.javac.Main();  
        //设置编译命令参数,与使用javac命令后面的参数一样  
        String[] params = new String[] {  
                "-d",  
                basedir,basedir+classname+".java",  
                "-verbose"  
        };  
        int status = javac.compile(params);  
        //当编译返回值为0时成功  
        if(status == 0)  
            System.out.println("compiled successfully!");  
        else  
            System.out.println("errors occurs");  
        //部署编译好的class到执行目录  
        copyTo(basedir+classname+".class",executedir+classname+".class");  
        //加载class字节码并实例化,再调用相应方法  
        invoke(classname,"say",new Class[]{String.class},new String[]{"Hello"});  
    }  
  
    /** 
     * 实例化并调用相应方法 
     * @param classname 类名 
     * @param methodname 方法名 
     * @param paramType 方法参数类型 
     * @param paramValues 方法参数值 
     */  
    public static void invoke(String classname, String methodname,  
                              Class[] paramType, Object[] paramValues){  
        try {  
            Class cls = Class.forName(classname);  
            // 方式一、不转化为具体类型,  
            // 利用反射创建一个Method实例,继而实现方法调用  
            Method method = cls.getMethod(methodname, paramType);  
            method.invoke(cls.newInstance(),paramValues);  
  
            // 方式二、转化为具体类型(需要设计相应接口),  
            // 反射实例化后强制转换为接口类型,再进行方法调用  
            Action person = (Action) cls.newInstance();  
            person.say(paramValues[0].toString());  
        } catch (Exception e) {  
            e.printStackTrace();  
        }  
    }  
  
    /** 
     * 复制文件到指定目录 
     * @param from 源文件 
     * @param to 目的文件 
     * @throws IOException 
     */  
    public static void copyTo(String from,String to) throws IOException {  
        FileInputStream fi = new FileInputStream(from);  
        FileOutputStream fo = new FileOutputStream(to);  
        File df = new File(to);  
        if(!df.exists())  
            df.createNewFile();  
        for(int read = fi.read(); read !=-1; read=fi.read()){  
            fo.write(read);  
        }  
        fo.close();  
        fi.close();  
    }  
}  

3、执行结果

/usr/jdk1.8.0_111/bin/java 
...
compiled successfully!
Person say a message: Hello
Person say a message: Hello
 
Process finished with exit code 0
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,110评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,443评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,474评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,881评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,902评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,698评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,418评论 3 419
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,332评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,796评论 1 316
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,968评论 3 337
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,110评论 1 351
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,792评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,455评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,003评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,130评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,348评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,047评论 2 355

推荐阅读更多精彩内容

  • 十六、模板注入 作者:Peter Yaworski 译者:飞龙 协议:CC BY-NC-SA 4.0 模板引擎是允...
    布客飞龙阅读 471评论 0 4
  • 作者:宇翔 一个人的声音能体现出他的修养,当然一个人的修行功力也决定了声音品质;有修为的人,他的声音是...
    宇翔空间阅读 912评论 0 2
  • 每两天,群里的组长和管理员会下发素材,让我们根据素材写作,定时交作业。 原来从不写作的我,在经过三十天的训练后,也...
    阿伊兰香阅读 109评论 0 0
  • 失去你 肯定如同地狱之苦 但大家都知道痛苦是可以承受的 会过去的 ——《...
    Amoonlight阅读 297评论 0 0
  • 在看这篇文章之前大家先看做出来的产品是什么:www.WebStack.cc --这是一个设计师的网址导航网。 写这...
    viggoz阅读 2,067评论 1 13