现在的位置: 首页 > 综合 > 正文

webx—javassist动态创建class文件

2014年09月05日 ⁄ 综合 ⁄ 共 1909字 ⁄ 字号 评论关闭

在开发过程中,经常会遇到第三方jar包的类提供的服务不能满足我们的需要。这时我们通常的做法是定义一个子类,复写父类方法,用子类实例化对象,其引用类型不变。 这种方式的特点是:

1. 修改源文件,新定义java文件

2. 编译.java文件为.class文件

3. 由classLoader加载字节码文件到内存中,由解析器来执行


现存在这么一种场景,如taobao开放平台,界面上有很多的选项框,我们只要选择相应的接口、方法,输入分配给我们的帐号,然后在文本框中编写一段java调用代码,就可以在页面上看到返回结果。

 

此时需要在运行的情况下,动态编译字符串代码,在内存中生成新的class文件,后续步骤同上

目前有很多成熟的开源项目支持分析、编辑和创建Java字节码,如cglib、asm、javassist

本文以javassist为例子,通过一个简单例子来描述上面是如何实现。

Person.java

package com.alibaba.model;

/**
 * 类Person.java的实现描述:TODO 类实现描述
 * 
 * @author onlyone 2012-6-17 下午04:19:49
 */
public class Person {

    private String name;
    private String age;
    private String address;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getAge() {
        return age;
    }

    public void setAge(String age) {
        this.age = age;
    }

    public String getAddress() {
        return address;
    }

    public void setAddress(String address) {
        this.address = address;
    }

}

AssistFactory.java

package com.alibaba.factory;

import java.util.concurrent.atomic.AtomicInteger;

import javassist.ClassPool;
import javassist.CtClass;
import javassist.CtMethod;

import com.alibaba.model.Person;

/**
 * 类AssistFactory.java的实现描述:TODO 类实现描述
 * 
 * @author onlyone 2012-6-17 下午04:20:46
 */
public class AssistFactory {

    // Class载入器
    private static ClassPool     pool;
    // 原子计数器
    private static AtomicInteger number = new AtomicInteger(1);
    static {
        pool = ClassPool.getDefault();
    }

    public void compileAndExe(String body) throws Exception {
        String name = Person.class.getName();
        // 新定义的子类,可以修改
        CtClass cc = pool.makeClass(name + "$" + number.incrementAndGet());
        // 父类
        cc.setSuperclass(pool.get(name));
        // 复写父类方法
        String method = "public String getName(){ System.out.println(super.getName()+\" is %s!\"); return super.getName(); }";
        
        method=String.format(method, body);
        // 将新方法添加到类中
        cc.addMethod(CtMethod.make(method, cc));

        // 类模板
        Class<?> c = cc.toClass();
        cc.detach();

        // 实例化对象
        Person p = (Person) c.newInstance();
        p.setName("onlyone");

        p.getName();
    }

    public static void main(String[] args) throws Exception {
        new AssistFactory().compileAndExe("sign");
    }
}

结果:

onlyone is sign!

代码下载地址:https://javassist.googlecode.com/svn/trunk/

抱歉!评论已关闭.