`
kavy
  • 浏览: 891008 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

Javassist 编程的动态性

 
阅读更多

讲过了 Java 类格式和利用反射进行的运行时访问后,本系列到了进入更高级主题的时候了。本月我将开始本系列的第二部分,在这里 Java 类信息只不过是由应用程序操纵的另一种形式的数据结构而已。我将这个主题的整个内容称为 classworking。

我将以 Javassist 字节码操作库作为对 classworking 的讨论的开始。Javassist 不仅是一个处理字节码的库,而且更因为它的另一项功能使得它成为试验 classworking 的很好的起点。这一项功能就是:可以用 Javassist 改变 Java 类的字节码,而无需真正了解关于字节码或者 Java 虚拟机(Java virtual machine JVM)结构的任何内容。从某方面将这一功能有好处也有坏处 -- 我一般不提倡随便使用不了解的技术 -- 但是比起在单条指令水平上工作的框架,它确实使字节码操作更可具有可行性了。

Javassist 基础

Javassist 使您可以检查、编辑以及创建 Java 二进制类。检查方面基本上与通过 Reflection API 直接在 Java 中进行的一样,但是当想要修改类而不只是执行它们时,则另一种访问这些信息的方法就很有用了。这是因为 JVM 设计上并没有提供在类装载到 JVM 中后访问原始类数据的任何方法,这项工作需要在 JVM 之外完成。

不要错过本系列的其余部分

第一部分:“ 类和类装入”(2003年4月)

第二部分:“ 引入反射” (2003年6月)

第三部分:“ 应用返射” (2003年7月)

Javassist 使用 javassist.ClassPool 类跟踪和控制所操作的类。这个类的工作方式与 JVM 类装载器非常相似,但是有一个重要的区别是它不是将装载的、要执行的类作为应用程序的一部分链接,类池使所装载的类可以通过 Javassist API 作为数据使用。可以使用默认的类池,它是从 JVM 搜索路径中装载的,也可以定义一个搜索您自己的路径列表的类池。甚至可以直接从字节数组或者流中装载二进制类,以及从头开始创建新类。

装载到类池中的类由 javassist.CtClass 实例表示。与标准的 Java java.lang.Class 类一样, CtClass 提供了检查类数据(如字段和方法)的方法。不过,这只是 CtClass 的部分内容,它还定义了在类中添加新字段、方法和构造函数、以及改变类、父类和接口的方法。奇怪的是,Javassist 没有提供删除一个类中字段、方法或者构造函数的任何方法。

字段、方法和构造函数分别由 javassist.CtField、javassist.CtMethod 和 javassist.CtConstructor 的实例表示。这些类定义了修改由它们所表示的对象的所有方法的方法,包括方法或者构造函数中的实际字节码内容。

所有字节码的源代码 

Javassist 让您可以完全替换一个方法或者构造函数的字节码正文,或者在现有正文的开始或者结束位置选择性地添加字节码(以及在构造函数中添加其他一些变量)。不管是哪种情况,新的字节码都作为类 Java 的源代码声明或者 String 中的块传递。Javassist 方法将您提供的源代码高效地编译为 Java 字节码,然后将它们插入到目标方法或者构造函数的正文中。

Javassist 接受的源代码与 Java 语言的并不完全一致,不过主要的区别只是增加了一些特殊的标识符,用于表示方法或者构造函数参数、方法返回值和其他在插入的代码中可能用到的内容。这些特殊标识符以符号 $ 开头,所以它们不会干扰代码中的其他内容。

对于在传递给 Javassist 的源代码中可以做的事情有一些限制。第一项限制是使用的格式,它必须是单条语句或者块。在大多数情况下这算不上是限制,因为可以将所需要的任何语句序列放到块中。下面是一个使用特殊 Javassist 标识符表示方法中前两个参数的例子,这个例子用来展示其使用方法:

{

  System.out.println("Argument 1: " + $1);

  System.out.println("Argument 2: " + $2);

}

对于源代码的一项更实质性的限制是不能引用在所添加的声明或者块外声明的局部变量。这意味着如果在方法开始和结尾处都添加了代码,那么一般不能将在开始处添加的代码中的信息传递给在结尾处添加的代码。有可能绕过这项限制,但是绕过是很复杂的 -- 通常需要设法将分别插入的代码合并为一个块。

回页首

用 Javassist 进行 Classworking

作为使用 Javassist 的一个例子,我将使用一个通常直接在源代码中处理的任务:测量执行一个方法所花费的时间。这在源代码中可以容易地完成,只要在方法开始时记录当前时间、之后在方法结束时再次检查当前时间并计算两个值的差。如果没有源代码,那么得到这种计时信息就要困难得多。这就是 classworking 方便的地方 -- 它让您对任何方法都可以作这种改变,并且不需要有源代码。

清单 1 显示了一个(不好的)示例方法,我用它作为我的计时试验的实验品: StringBuilder 类的 buildString 方法。这个方法使用一种所有 Java 性能优化的高手都会叫您 不要使用的方法构造一个具有任意长度的 String -- 它通过反复向字符串的结尾附加单个字符来产生更长的字符串。因为字符串是不可变的,所以这种方法意味着每次新的字符串都要通过一个循环来构造:使用从老的字符串中拷贝的数据并在结尾添加新的字符。最终的效果是用这个方法产生更长的字符串时,它的开销越来越大。

清单 1. 需要计时的方法

public class StringBuilder

{

    private String buildString(int length) {

        String result = "";

        for (int i = 0; i < length; i++) {

            result += (char)(i%26 + 'a');

        }

        return result;

    }

    

    public static void main(String[] argv) {

        StringBuilder inst = new StringBuilder();

        for (int i = 0; i < argv.length; i++) {

            String result = inst.buildString(Integer.parseInt(argv[i]));

            System.out.println("Constructed string of length " +

                result.length());

        }

    }

}

添加方法计时

因为有这个方法的源代码,所以我将为您展示如何直接添加计时信息。它也作为使用 Javassist 时的一个模型。清单 2 只展示了 buildString() 方法,其中添加了计时功能。这里没有多少变化。添加的代码只是将开始时间保存为局部变量,然后在方法结束时计算持续时间并打印到控制台。

清单 2. 带有计时的方法

    private String buildString(int length) {

        long start = System.currentTimeMillis();

        String result = "";

        for (int i = 0; i < length; i++) {

            result += (char)(i%26 + 'a');

        }

        System.out.println("Call to buildString took " +

            (System.currentTimeMillis()-start) + " ms.");

        return result;

    }

用 Javassist 来做

来做 使用 Javassist 操作类字节码以得到同样的效果看起来应该不难。Javassist 提供了在方法的开始和结束位置添加代码的方法,别忘了,我在为该方法中加入计时信息就是这么做的。

不过,还是有障碍。在描述 Javassist 是如何让您添加代码时,我提到添加的代码不能引用在方法中其他地方定义的局部变量。这种限制使我不能在 Javassist 中使用在源代码中使用的同样方法实现计时代码,在这种情况下,我在开始时添加的代码中定义了一个新的局部变量,并在结束处添加的代码中引用这个变量。

那么还有其他方法可以得到同样的效果吗?是的,我 可以在类中添加一个新的成员字段,并使用这个字段而不是局部变量。不过,这是一种糟糕的解决方案,在一般性的使用中有一些限制。例如,考虑在一个递归方法中会发生的事情。每次方法调用自身时,上次保存的开始时间值就会被覆盖并且丢失。

幸运的是有一种更简洁的解决方案。我可以保持原来方法的代码不变,只改变方法名,然后用原来的方法名增加一个新方法。这个 拦截器(interceptor)方法可以使用与原来方法同样的签名,包括返回同样的值。清单 3 展示了通过这种方法改编后源代码看上去的样子:

清单 3. 在源代码中添加一个拦截器方法

    private String buildString$impl(int length) {

        String result = "";

        for (int i = 0; i < length; i++) {

            result += (char)(i%26 + 'a');

        }

        return result;

    }

    private String buildString(int length) {

        long start = System.currentTimeMillis();

        String result = buildString$impl(length);

        System.out.println("Call to buildString took " +

            (System.currentTimeMillis()-start) + " ms.");

        return result;

    }

通过 Javassist 可以很好地利用这种使用拦截器方法的方法。因为整个方法是一个块,所以我可以毫无问题地在正文中定义并且使用局部变量。为拦截器方法生成源代码也很容易 -- 对于任何可能的方法,只需要几个替换。

运行拦截

实现添加方法计时的代码要用到在 Javassist 基础中描述的一些 Javassist API。清单 4 展示了该代码,它是一个带有两个命令行参数的应用程序,这两个参数分别给出类名和要计时的方法名。 main() 方法的正文只给出类信息,然后将它传递给 addTiming() 方法以处理实际的修改。 addTiming() 方法首先通过在名字后面附加“ $impl” 重命名现有的方法,接着用原来的方法名创建该方法的一个拷贝。然后它用含有对经过重命名的原方法的调用的计时代码替换拷贝方法的正文。

清单4. 用 Javassist 添加拦截器方法

public class JassistTiming 

{

    public static void main(String[] argv) {

        if (argv.length == 2) {

            try {

                

                // start by getting the class file and method

                CtClass clas = ClassPool.getDefault().get(argv[0]);

                if (clas == null) {

                    System.err.println("Class " + argv[0] + " not found");

                } else {

                    

                    // add timing interceptor to the class

                    addTiming(clas, argv[1]);

                    clas.writeFile();

                    System.out.println("Added timing to method " +

                        argv[0] + "." + argv[1]);

                    

                }

                

            } catch (CannotCompileException ex) {

                ex.printStackTrace();

            } catch (NotFoundException ex) {

                ex.printStackTrace();

            } catch (IOException ex) {

                ex.printStackTrace();

            }

            

        } else {

            System.out.println("Usage: JassistTiming class method-name");

        }

    }

    

    private static void addTiming(CtClass clas, String mname)

        throws NotFoundException, CannotCompileException {

        

        //  get the method information (throws exception if method with

        //  given name is not declared directly by this class, returns

        //  arbitrary choice if more than one with the given name)

        CtMethod mold = clas.getDeclaredMethod(mname);

        

        //  rename old method to synthetic name, then duplicate the

        //  method with original name for use as interceptor

        String nname = mname+"$impl";

        mold.setName(nname);

        CtMethod mnew = CtNewMethod.copy(mold, mname, clas, null);

        

        //  start the body text generation by saving the start time

        //  to a local variable, then call the timed method; the

        //  actual code generated needs to depend on whether the

        //  timed method returns a value

        String type = mold.getReturnType().getName();

        StringBuffer body = new StringBuffer();

        body.append("{\nlong start = System.currentTimeMillis();\n");

        if (!"void".equals(type)) {

            body.append(type + " result = ");

        }

        body.append(nname + "($$);\n");

        

        //  finish body text generation with call to print the timing

        //  information, and return saved value (if not void)

        body.append("System.out.println(\"Call to method " + mname +

            " took \" +\n (System.currentTimeMillis()-start) + " +

            "\" ms.\");\n");

        if (!"void".equals(type)) {

            body.append("return result;\n");

        }

        body.append("}");

        

        //  replace the body of the interceptor method with generated

        //  code block and add it to class

        mnew.setBody(body.toString());

        clas.addMethod(mnew);

        

        //  print the generated code block just to show what was done

        System.out.println("Interceptor method body:");

        System.out.println(body.toString());

    }

}

构造拦截器方法的正文时使用一个 java.lang.StringBuffer 来累积正文文本(这显示了处理 String 的构造的正确方法,与在 StringBuilder 的构造中使用的方法是相对的)。这种变化取决于原来的方法是否有返回值。如果它 有返回值,那么构造的代码就将这个值保存在局部变量中,这样在拦截器方法结束时就可以返回它。如果原来的方法类型为 void ,那么就什么也不需要保存,也不用在拦截器方法中返回任何内容。

除了对(重命名的)原来方法的调用,实际的正文内容看起来就像标准的 Java 代码。它是代码中的 body.append(nname + "($$);\n") 这一行,其中 nname 是原来方法修改后的名字。在调用中使用的 $$ 标识符是 Javassist 表示正在构造的方法的一系列参数的方式。通过在对原来方法的调用中使用这个标识符,在调用拦截器方法时提供的参数就可以传递给原来的方法。

清单 5 展示了首先运行未修改过的 StringBuilder 程序、然后运行 JassistTiming 程序以添加计时信息、最后运行修改后的 StringBuilder 程序的结果。可以看到修改后的 StringBuilder 运行时会报告执行的时间,还可以看到因为字符串构造代码效率低下而导致的时间增加远远快于因为构造的字符串长度的增加而导致的时间增加。

清单 5. 运行这个程序

[dennis]$ java StringBuilder 1000 2000 4000 8000 16000

Constructed string of length 1000

Constructed string of length 2000

Constructed string of length 4000

Constructed string of length 8000

Constructed string of length 16000

[dennis]$ java -cp javassist.jar:. JassistTiming StringBuilder buildString

Interceptor method body:

{

long start = System.currentTimeMillis();

java.lang.String result = buildString$impl($$);

System.out.println("Call to method buildString took " +

 (System.currentTimeMillis()-start) + " ms.");

return result;

}

Added timing to method StringBuilder.buildString

[dennis]$ java StringBuilder 1000 2000 4000 8000 16000

Call to method buildString took 37 ms.

Constructed string of length 1000

Call to method buildString took 59 ms.

Constructed string of length 2000

Call to method buildString took 181 ms.

Constructed string of length 4000

Call to method buildString took 863 ms.

Constructed string of length 8000

Call to method buildString took 4154 ms.

Constructed string of length 16000

回页首

可以信任源代码吗?

Javassist 通过让您处理源代码而不是实际的字节码指令清单而使 classworking 变得容易。但是这种方便性也有一个缺点。正如我在 所有字节码的源代码中提到的,Javassist 所使用的源代码与 Java 语言并不完全一样。除了在代码中识别特殊的标识符外,Javassist 还实现了比 Java 语言规范所要求的更宽松的编译时代码检查。因此,如果不小心,就会从源代码中生成可能会产生令人感到意外的结果的字节码。

作为一个例子,清单 6 展示了在将方法开始时的拦截器代码所使用的局部变量的类型从 long 变为 int 时的情况。Javassist 会接受这个源代码并将它转换为有效的字节码,但是得到的时间是毫无意义的。如果试着直接在 Java 程序中编译这个赋值,您就会得到一个编译错误,因为它违反了 Java 语言的一个规则:一个窄化的赋值需要一个类型覆盖。

清单 6. 将一个 long 储存到一个 int 中

[dennis]$ java -cp javassist.jar:. JassistTiming StringBuilder buildString

Interceptor method body:

{

int start = System.currentTimeMillis();

java.lang.String result = buildString$impl($$);

System.out.println("Call to method buildString took " +

 (System.currentTimeMillis()-start) + " ms.");

return result;

}

Added timing to method StringBuilder.buildString

[dennis]$ java StringBuilder 1000 2000 4000 8000 16000

Call to method buildString took 1060856922184 ms.

Constructed string of length 1000

Call to method buildString took 1060856922172 ms.

Constructed string of length 2000

Call to method buildString took 1060856922382 ms.

Constructed string of length 4000

Call to method buildString took 1060856922809 ms.

Constructed string of length 8000

Call to method buildString took 1060856926253 ms.

Constructed string of length 16000

取决于源代码中的内容,甚至可以让 Javassist 生成无效的字节码。清单7展示了这样的一个例子,其中我将 JassistTiming 代码修改为总是认为计时的方法返回一个 int 值。Javassist 同样会毫无问题地接受这个源代码,但是在我试图执行所生成的字节码时,它不能通过验证。

清单7. 将一个 String 储存到一个 int 中

[dennis]$ java -cp javassist.jar:. JassistTiming StringBuilder buildString

Interceptor method body:

{

long start = System.currentTimeMillis();

int result = buildString$impl($$);

System.out.println("Call to method buildString took " +

 (System.currentTimeMillis()-start) + " ms.");

return result;

}

Added timing to method StringBuilder.buildString

[dennis]$ java StringBuilder 1000 2000 4000 8000 16000

Exception in thread "main" java.lang.VerifyError:

 (class: StringBuilder, method: buildString signature:

 (I)Ljava/lang/String;) Expecting to find integer on stack

只要对提供给 Javassist 的源代码加以小心,这就不算是个问题。不过,重要的是要认识到 Javassist 没有捕获代码中的所有错误,所以有可能会出现没有预见到的错误结果。

回页首

后续内容

Javassist 比我们在本文中所讨论的内容要丰富得多。下一个月,我们将进行更进一步的分析,看一看 Javassist 为批量修改类以及为在运行时装载类时对类进行动态修改而提供的一些特殊的功能。这些功能使 Javassist 成为应用程序中实现方面的一个很棒的工具,所以一定要继续跟随我们了解这个强大工具的全部内容。

参考资料

您可以参阅本文在 developerWorks 全球站点上的 英文原文.

Javassist 是由东京技术学院的数学和计算机科学系的 Shigeru Chiba 所创建的。它最近加入了开放源代码 JBoss 应用服务器项目,并成为其中新增加的面向方面的编程功能的基础。Javassist 以 Mozilla Public License (MPL) 和 GNU Lesser General Public License (LGPL) 开放源代码许可证的形式发布。

从“ Java bytecode: Understanding bytecode makes you a better programmer” ( developerWorks, 2001年7月)中了解有关 Java 字节码设计的更多内容。

要了解更多面向方面编程的内容吗?可以从“ Improve modularity with aspect-oriented programming” ( developerWorks, 2002年1月)中找到关于使用 AspectJ 语言的概述。

开放源代码 Jikes 项目提供了一个非常快速和高度兼容 Java 编程语言的编译器。用它以老的方式生成您的字节码 -- 从 Java 源代码生成。

在 developerWorks Java 技术专区 可以找到关于 Java 编程各方面的数百篇文章。

 

 

    关于java字节码的处理,目前有很多工具,如bcel,asm。不过这些都需要直接跟虚拟机指令打交道。如果你不想了解虚拟机指令,可以采用javassist。javassist是jboss的一个子项目,其主要的优点,在于简单,而且快速。直接使用java编码的形式,而不需要了解虚拟机指令,就能动态改变类的结构,或者动态生成类。
    下面通过一个简单的例子,通过javassist来实现如何动态注入代码。
    假设,存在类A,如下:
public class A {
    public void method() {
        for (int i = 0; i < 1000000; i++) {
        }
        System.out.println("method1");
    }
}
测试类B如下:
public class B {
    public static void main(String[] args) {
        A a = new A();
        a.method();    
    }
}
现在想统计一下method的执行时间,
默认的实现是修改method:
 public void method() {
        long start = System.currentTimeMillis();
        for (int i = 0; i < 1000000; i++) {
        }
        System.out.println("method1");
        long end = System.currentTimeMillis();
        System.out.println(end - start);
    }
如果A的方法很多,统计方法的执行时间的代码就会相应的增加。为了减少工作量,通过动态注入代码的形式来实现。
修改B的main方法:
    public static void main(String[] args) throws Exception {
      //用于取得字节码类,必须在当前的classpath中,使用全称
        CtClass ctClass = ClassPool.getDefault().get("org.esoft.A");
         //需要修改的方法名称
        String mname = "method";        
        CtMethod mold = ctClass.getDeclaredMethod(mname);
         //修改原有的方法名称
        String nname = mname + "$impl";
        mold.setName(nname);
         //创建新的方法,复制原来的方法
        CtMethod mnew = CtNewMethod.copy(mold, mname, ctClass, null);
         //主要的注入代码
        StringBuffer body = new StringBuffer();
        body.append("{\nlong start = System.currentTimeMillis();\n");
        //调用原有代码,类似于method();($$)表示所有的参数
        body.append(nname + "($$);\n");
        body.append("System.out.println(\"Call to method "
                    + mname
                    + " took \" +\n (System.currentTimeMillis()-start) + "
                    + "\" ms.\");\n");
       
        body.append("}");
         //替换新方法
        mnew.setBody(body.toString());
         //增加新方法
        ctClass.addMethod(mnew);
        //类已经更改,注意不能使用A a=new A();,因为在同一个classloader中,不允许装载同一个类两次
        A a=(A)ctClass.toClass().newInstance();
        a.method();
    }
这只是简单的一个应用。javassist还提供了很多的功能,用于更改类结构。有兴趣的可以参考相关文档

分享到:
评论

相关推荐

    动态代理-jdk、cglib、javassist.zip

    动态代理在Java编程中是一种非常重要的技术,它允许我们在运行时创建对象的代理,从而可以在不修改原有代码的情况下,为对象添加额外的...理解并掌握这三种动态代理技术,对于提升Java编程的灵活性和效率具有重要意义。

    javaagent+javassist

    这使得javassist成为单元测试框架、AOP(面向切面编程)和动态代理实现的理想工具。 将`javaagent`与`javassist`结合使用,可以实现强大的功能。例如,我们可以通过javaagent在程序启动时注入javassist生成的代码,...

    javassistDemo

    Javaassist是一个开源库,它允许在运行时修改Java类和创建新的类。这个库在Java世界里被广泛用于动态代理、AOP(面向切面编程)以及类的...理解这些知识点对于深入学习Java的动态性、网络调试和AOP编程具有重要意义。

    javassist+jd-gui

    Java编程语言以其跨平台性和强大的库支持而广受欢迎,但有时候开发者可能需要在程序已经编译后对字节码进行修改。在这种情况下,`javassist` 和 `jd-gui` 这两个工具就显得尤为重要。 `javassist` 是一个开源库,...

    javassistDemo.zip

    这个"javassistDemo.zip"压缩包提供了一个示例,展示了如何利用Javaassist库动态修改Java类的方法。在Java应用程序中,这种能力非常有用,特别是在进行AOP(面向切面编程)或者在无法重新编译源代码的情况下需要修改...

    Java动态代理机制详解(JDK 和CGLIB,Javassist,ASM)

    总的来说,Java动态代理机制为我们提供了灵活的代码扩展性,可以根据需要选择JDK、CGLIB、Javassist或ASM来实现不同的代理需求。理解并掌握这些技术,能帮助我们在开发过程中更加高效地实现功能扩展和代码解耦。在...

    运用javassist和annotation修改class的特定method的class byte code

    Javaassist是一个开源库,它允许我们在运行时动态地修改或创建Java类。这个强大的工具广泛应用于框架、代理生成以及AOP(面向...尽管这需要对字节码和反射有深入的理解,但一旦掌握,就能极大地增强我们的编程灵活性。

    jboss-javassist和JByteMode整合包

    通过Javassist,我们可以方便地生成和修改类的字节码,从而实现AOP(面向切面编程)等高级功能。它的主要特性包括: 1. 提供了类似于C/C++的API,使得字节码操作更加直观。 2. 支持处理类、接口、方法、字段等元数据...

    Javassist18版20版22版的jar包

    - AOP(面向切面编程)框架:如Spring AOP,Javassist可以用来生成代理类以实现切面逻辑。 - 框架开发:许多自定义框架会使用Javassist来实现动态代码生成,比如ORM框架Hibernate。 - 性能监控:可以通过修改字节...

    agent+javassist例子

    `Java Agent`允许我们对Java应用程序进行预处理,比如字节码注入,而`javassist`库则提供了一个方便的方式来动态地操作和修改Java类的字节码。 `Java Agent`是Java平台提供的一种机制,允许开发者在程序运行前或...

    javassist-3.18.1-GA.jar

    在Java世界里,这种技术通常被称为字节码工程,对于实现如AOP(面向切面编程)、动态代理和运行时代码优化等高级功能非常有用。 `javassist-3.18.1-GA.jar`是Javaassist的一个特定版本,GA代表“General ...

    javassist-3.20.0-GA.zip

    这对于AOP(面向切面编程)、代理、代码生成以及运行时优化等场景特别有用。 在Javassist中,`ClassPool`对象是核心组件,用于管理和操作类的信息。你可以通过`ClassPool.getDefault()`获取默认的类池,或者通过...

    javassist-3.15.0-GA

    总结来说,"javassist-3.15.0-GA"是Java开发中的一个重要工具,它使得在运行时动态修改和增强Java代码成为可能,广泛应用于AOP、动态代理、代码生成等领域。通过深入理解和熟练运用Javaassist,开发者可以提高代码的...

    javassist-3.7.ga.jar

    `javassist-3.7.ga.jar` 文件是 Javaassist 库的一个版本,版本号为 3.7 GA(General Availability),意味着这是一个公开可用的稳定版本。 Javaassist 提供了一系列的 API,使得开发者可以方便地读取、修改和创建 ...

    javassist-3.9.0.GA.jar.zip

    Javaassist是一个开源库,主要用在Java编程中,它提供了对字节码操作的强大功能,使得程序员可以在运行时动态修改类或创建新的类。这个"javassist-3.9.0.GA.jar.zip"文件包含了Javaassist库的版本3.9.0 GA,这是一个...

    javassist-3.7.ga.jar下载

    6. 与JDK动态代理的兼容性:Javaassist可以与JDK的动态代理结合使用,为已经存在的接口提供动态实现。 `javassist-3.7.ga.jar`的下载意味着用户可能需要这个特定版本来兼容他们的项目或者满足特定的环境需求。在...

    javassist.jar源码

    2. **字节码生成**:Javaassist可以生成新的类或方法的字节码,这对于实现动态代理、AOP(面向切面编程)或者在运行时创建新的类型非常有用。`ClassPool`类用于管理类的字节码,而` CtNewMethod `和` ...

    javassist-3.18.0-ga

    6. **兼容性**:`javassist-3.18.0-ga`版本支持Java 5及以上版本,这意味着它可以广泛应用于各种Java项目,包括那些基于旧版JDK的项目。 7. **应用领域**:Javaassist常用于动态代理框架(如Spring AOP)、代码生成...

    基于javassist调用方法.zip_动态编码_快速反射

    Java编程语言以其强大的灵活性和可扩展性闻名,其中一个关键特性就是反射(Reflection)。然而,反射在性能上并不总是最优的,特别是在需要频繁调用方法的情况下。这就是`javassist`库的作用,它提供了一种更高效的...

    javassist-3.7.GA.zip

    3. **错误处理和日志记录**:Javaassist可能用于在发生异常时动态插入错误处理代码,或者添加日志记录语句,以增强系统的可维护性和可追溯性。 在解压"javassist-3.7.GA.zip"后,你会得到"javassist-3.7"目录,其中...

Global site tag (gtag.js) - Google Analytics