`

javassist【动态改字节码】学习三

    博客分类:
  • j2se
 
阅读更多
这里举个简单的例子,来记录下如何用CtClass创建一个类,并且往这个类里面新加方法,

下面是代码,可以直接运行:
Java代码

package seeeyou.app.test;

import java.io.IOException;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

import javassist.CannotCompileException;
import javassist.ClassPool;
import javassist.CtClass;
import javassist.CtMethod;
import javassist.CtNewMethod;
import javassist.NotFoundException;

/**
*
* @author seeeyou
*
*/
public class TestHelloWorld2 {
public static void main(String[] args) throws NotFoundException,
IOException, CannotCompileException, InstantiationException,
IllegalAccessException, SecurityException, NoSuchMethodException,
IllegalArgumentException, InvocationTargetException {
// 用于取得字节码类,必须在当前的classpath中,使用全称
ClassPool pool = ClassPool.getDefault();
/**
* makeClass() cannot create a new interface; makeInterface() in
* ClassPool can do. Member methods in an interface can be created with
* abstractMethod() in CtNewMethod. Note that an interface method is an
* abstract method.
*/
CtClass ccClass = pool.makeClass("Point");
String bodyString = "{System.out.println(\"Call to method \");}";
//为新创建的类新加一个方法execute,无任何参数
CtMethod n1 = CtNewMethod.make(CtClass.voidType, "execute", null, null,
bodyString, ccClass);
ccClass.addMethod(n1);
/**
* 这里无法用new的形式来创建一个对象,因为已经classloader中不能有两个相同的对象,否则会报异常如下:
*Caused by: java.lang.LinkageError: loader (instance of  sun/misc/Launcher$AppClassLoader):
*attempted  duplicate class definition for name: "Point"
**/
Object oo = ccClass.toClass().newInstance();
Method mms = oo.getClass().getMethod("execute", null);
System.out.println("new class name is : " + oo.getClass().getName());
System.out.println("new class's method is : " + mms.invoke(oo, null));
System.out.println("---------------------------------------------");
//这一行代码将class冻结了,下面无法再对类多编辑或者修改,下面的setName会报异常如:
//Exception in thread "main" java.lang.RuntimeException: Point class is frozen
ccClass.freeze();
try{
ccClass.setName("Point2");
}catch (Exception e) {
System.out.println(e);
}
//对已经冻结的class解冻之后还可以继续编辑修改
ccClass.defrost();
System.out.println("------------- 上面的代码是对的,下面的代码将会无法执行出结果,会报错------------------------");
//第二个方法
bodyString = "public int getNumber(Integer num){System.out.println(\"Point2 Call to method \");return 10+num;}";
CtMethod n2 = CtNewMethod.make(bodyString, ccClass);//直接创建一个方法,带有一个int的参数和返回值
ccClass.addMethod(n2);
Class[] params = new Class[1];
Integer num = new Integer(15);
params[0] = num.getClass();
mms = oo.getClass().getMethod("getNumber", params);
System.out.println("new class name is : " + oo.getClass().getName());
System.out.println("new class's method is : " + mms.invoke(oo, 100));
System.out.println("---------------------------------------------");
}
}

Java代码

上面的结果是:new class name is : Point
Call to method
new class's method is : null
---------------------------------------------
java.lang.RuntimeException: Point class is frozen
---------------------------------------------
Exception in thread "main" java.lang.NoSuchMethodException: Point.getNumber(java.lang.Integer)
at java.lang.Class.getMethod(Class.java:1605)
at seeeyou.app.test.TestHelloWorld2.main(TestHelloWorld2.java:66)



错误的原因其实和简单,因为我第二次新加一个方法后,没有再次实例化一个对象,所以oo还是原来的对象,他的成员函数肯定没有新加的方法。

那我可以再次实例化下试试,代码和结果如下:
Java代码

package seeeyou.app.test;

import java.io.IOException;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

import javassist.CannotCompileException;
import javassist.ClassPool;
import javassist.CtClass;
import javassist.CtMethod;
import javassist.CtNewMethod;
import javassist.NotFoundException;

/**
*
* @author seeeyou
*
*/
public class TestHelloWorld2 {
public static void main(String[] args) throws NotFoundException,
IOException, CannotCompileException, InstantiationException,
IllegalAccessException, SecurityException, NoSuchMethodException,
IllegalArgumentException, InvocationTargetException {
// 用于取得字节码类,必须在当前的classpath中,使用全称
ClassPool pool = ClassPool.getDefault();
/**
* makeClass() cannot create a new interface; makeInterface() in
* ClassPool can do. Member methods in an interface can be created with
* abstractMethod() in CtNewMethod. Note that an interface method is an
* abstract method.
*/
CtClass ccClass = pool.makeClass("Point");
String bodyString = "{System.out.println(\"Call to method \");}";
//为新创建的类新加一个方法execute,无任何参数
CtMethod n1 = CtNewMethod.make(CtClass.voidType, "execute", null, null,
bodyString, ccClass);
ccClass.addMethod(n1);
/**
* 这里无法用new的形式来创建一个对象,因为已经classloader中不能有两个相同的对象,否则会报异常如下:
*Caused by: java.lang.LinkageError: loader (instance of  sun/misc/Launcher$AppClassLoader):
*attempted  duplicate class definition for name: "Point"
**/
Object oo = ccClass.toClass().newInstance();
Method mms = oo.getClass().getMethod("execute", null);
System.out.println("new class name is : " + oo.getClass().getName());
System.out.println("new class's method is : " + mms.invoke(oo, null));
System.out.println("---------------------------------------------");
//这一行代码将class冻结了,下面无法再对类多编辑或者修改,下面的setName会报异常如:
//Exception in thread "main" java.lang.RuntimeException: Point class is frozen
ccClass.freeze();
try{
ccClass.setName("Point2");
}catch (Exception e) {
System.out.println(e);
}
//对已经冻结的class解冻之后还可以继续编辑修改
ccClass.defrost();
System.out.println("------------- 上面的代码是对的,下面的代码将会无法执行出结果,会报错------------------------");
//第二个方法
bodyString = "public int getNumber(Integer num){System.out.println(\"Point2 Call to method \");return 10+num;}";
CtMethod n2 = CtNewMethod.make(bodyString, ccClass);//直接创建一个方法,带有一个int的参数和返回值
ccClass.addMethod(n2);
Class[] params = new Class[1];
Integer num = new Integer(15);
params[0] = num.getClass();//就多了下面这个实例化,但是这样会导致一个错误
oo = ccClass.toClass().newInstance();
mms = oo.getClass().getMethod("getNumber", params);
System.out.println("new class name is : " + oo.getClass().getName());
System.out.println("new class's method is : " + mms.invoke(oo, 100));
System.out.println("---------------------------------------------");
}
}

Java代码

这也会导致一个错误:new class name is : Point
Call to method
new class's method is : null
---------------------------------------------
java.lang.RuntimeException: Point class is frozen
------------- 上面的代码是对的,下面的代码将会无法执行出结果,会报错------------------------
Exception in thread "main" javassist.CannotCompileException: by java.lang.LinkageError: loader (instance of  sun/misc/Launcher$AppClassLoader): attempted  duplicate class definition for name: "Point"
at javassist.ClassPool.toClass(ClassPool.java:1051)
at javassist.ClassPool.toClass(ClassPool.java:994)
at javassist.ClassPool.toClass(ClassPool.java:952)
at javassist.CtClass.toClass(CtClass.java:1079)
at seeeyou.app.test.TestHelloWorld2.main(TestHelloWorld2.java:66)
Caused by: java.lang.LinkageError: loader (instance of  sun/misc/Launcher$AppClassLoader): attempted  duplicate class definition for name: "Point"
at java.lang.ClassLoader.defineClass1(Native Method)
at java.lang.ClassLoader.defineClass(ClassLoader.java:621)
at java.lang.ClassLoader.defineClass(ClassLoader.java:466)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
at java.lang.reflect.Method.invoke(Method.java:597)
at javassist.ClassPool.toClass2(ClassPool.java:1063)
at javassist.ClassPool.toClass(ClassPool.java:1045)
... 4 more

Java代码

    原因也很简单,一个classloader里面怎么有两个重复的对象呢,除非是两个不同的classloader。。所以爆了个重复加载类的错误

原因也很简单,一个classloader里面怎么有两个重复的对象呢,除非是两个不同的classloader。。所以爆了个重复加载类的错误

Java代码



对的方式是只实例化一次:如下:
Java代码

package seeeyou.app.test;

import java.io.IOException;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

import javassist.CannotCompileException;
import javassist.ClassPool;
import javassist.CtClass;
import javassist.CtMethod;
import javassist.CtNewMethod;
import javassist.NotFoundException;

/**
*
* @author seeeyou
*
*/
public class TestHelloWorld3 {

public static void main(String[] args) throws NotFoundException,
IOException, CannotCompileException, InstantiationException,
IllegalAccessException, SecurityException, NoSuchMethodException,
IllegalArgumentException, InvocationTargetException {
// 用于取得字节码类,必须在当前的classpath中,使用全称
ClassPool pool = ClassPool.getDefault();
/**
* makeClass() cannot create a new interface; makeInterface() in
* ClassPool can do. Member methods in an interface can be created with
* abstractMethod() in CtNewMethod. Note that an interface method is an
* abstract method.
*/
CtClass ccClass = pool.makeClass("Point");
String bodyString = "{System.out.println(\"Call to method \");}";
//为新创建的类新加一个方法execute,无任何参数
CtMethod n1 = CtNewMethod.make(CtClass.voidType, "execute", null, null,
bodyString, ccClass);
ccClass.addMethod(n1);
//新加第二个方法
bodyString = "public Integer getNumber(Integer num);";
CtMethod n2 = CtNewMethod.make(bodyString, ccClass);//直接创建一个方法,带有一个int的参数和返回值
n2.setBody("{System.out.println(\"Point Call to method \");return $1;}");
ccClass.addMethod(n2);

/**
* 这里无法用new的形式来创建一个对象,因为已经classloader中不能有两个相同的对象,否则会报异常如下:
*Caused by: java.lang.LinkageError: loader (instance of  sun/misc/Launcher$AppClassLoader):
*attempted  duplicate class definition for name: "Point"
**/
Object oo = ccClass.toClass().newInstance();
Method mms = oo.getClass().getMethod("execute", null);
System.out.println("new class name is : " + oo.getClass().getName());
System.out.println("new class's method is : " + mms.invoke(oo, null));
System.out.println("---------------------------------------------");
//这一行代码将class冻结了,下面无法再对类多编辑或者修改,下面的setName会报异常如:
//Exception in thread "main" java.lang.RuntimeException: Point class is frozen
ccClass.freeze();
try{
ccClass.setName("Point2");
}catch (Exception e) {
System.out.println(e);
}
//对已经冻结的class解冻之后还可以继续编辑修改
ccClass.defrost();
System.out.println("------------- 上面的代码是对的,下面的代码将会无法执行出结果,会报错------------------------");
Class[] params = new Class[1];
Integer num = new Integer(0);
params[0] = num.getClass();
mms = oo.getClass().getMethod("getNumber",params);
System.out.println("new class name is : " + oo.getClass().getName());
System.out.println("new class's method is : " + mms.invoke(oo, 100));
System.out.println("---------------------------------------------");
}
}

Java代码

结果如下:new class name is : Point
Call to method
new class's method is : null
---------------------------------------------
java.lang.RuntimeException: Point class is frozen
------------- 上面的代码是对的,下面的代码将会无法执行出结果,会报错------------------------
new class name is : Point
Point2 Call to method
new class's method is : 100
---------------------------------------------

http://alicsd.com/a/226
分享到:
评论

相关推荐

    Javassist 修改class字节码

    Javassist 提供了java类库,用于方便操控Java字节码。功能包括:运行时创建java class,修改class。与其他同类工具(asm等)不同的是,Javassist提供了两个层面的API: 1.java代码层 2.字节码层 通过java代码层,...

    借助jclasslib与javassist修改java class字节码

    `javassist`是一个强大的字节码编辑器和类文件操作库,它能够帮助开发者动态地创建或修改类定义。通过`javassist`,可以方便地添加方法、字段或修改现有方法的行为,而无需接触原始源代码。 ##### 2. jclasslib `...

    javassist,Java字节码工程工具包.zip

    3. 字节码生成:Javassist提供了一系列API,可以方便地生成新的类和方法的字节码。这些API类似于Java源代码的构造,使得生成字节码的过程更加直观和简洁。 4. AOP支持:面向切面编程是一种编程范式,用于解耦关注点...

    javassist, Java字节码工程工具包.zip

    javassist, Java字节码工程工具包 Java字节码工程工具包 版本 3版权所有( C ) 1999 -2017按 Shigeru Chiba,保留所有权利。Javassist ( Java编程助手) 使Java字节码操作简单。 它是一个类库,用于在Java中编辑字节码

    修改字节码 jclasslib

    尽管jclasslib主要用于查看和分析字节码,但通过与其他工具(如ASM、BCEL或Javassist)结合,可以实现字节码的修改。例如,可以先使用jclasslib来理解一个类的字节码,然后利用这些工具进行动态或静态的字节码注入,...

    javassist试图简化Java字节码的编辑

    Java字节码编辑是Java开发中的一个高级主题,它允许开发者在运行时修改或增强类的行为。`javassist`库正是这样一个工具,它为Java...学习并熟练掌握`javassist`,将极大地提升你在Java动态编程和字节码操作方面的技能。

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

    3. **检查和修改字节码**:现在,我们可以通过`CtMethod`对象访问和修改方法的字节码。例如,我们可以在方法调用之前或之后插入额外的代码。 ```java ctMethod.insertBefore("{ System.out.println(\"Before method...

    Javassist如何操作Java 字节码

    Javassist是一个强大的Java字节码操作库,它允许开发者在运行时动态修改或创建类。Javassist的出现使得开发者无需深入理解底层字节码指令就能方便地操作类、方法和字段,降低了字节码操作的难度。下面将详细介绍...

    javassist官方手册(中文)

    Javassist 官方手册中文版 Javassist 是一个动态生成字节码的框架,生成的字节码可以输出或实时加载到 JVM 中。...通过学习和掌握 Javassist,可以使开发者更好地控制和操作 Java 字节码,提高开发效率和质量。

    JavaAgent:Javassist 与 Asm JavaAgent 字节码动态编程项目

    在这个“JavaAgent: Javassist与Asm JavaAgent字节码动态编程项目”中,我们将深入探讨如何利用Javassist和ASM这两个库来实现JavaAgent。 首先,Javassist是一个开源的Java字节码操作框架,它使得开发者可以在运行...

    javassist:Java字节码工程工具包

    Javassist(JAVA编程ASSISTant)使Java字节码操作变得简单。 它是一个用Java编辑字节码的类库。 它使Java程序可以在运行时定义新类,并在JVM加载它时修改类文件。 与其他类似的字节码编辑器不同,Javassist提供了两...

    JAVA字节码操作库 BCEL

    2. **字节码生成**:BCEL提供了一套接口和类,使得开发者可以动态地生成字节码,这对于创建动态代理、AOP(面向切面编程)等场景非常有用。 3. **代码优化**:通过修改字节码,BCEL可以实现代码的优化,例如去除...

    jboss-javassist和JByteMode整合包

    3. **字节码重新生成**:使用Javassist将修改后的源代码重新编译为字节码,并替换原有的.class文件。 4. **加载到JVM**:最后,将修改后的字节码加载到Java虚拟机,使修改生效。 **注意事项** 虽然这种动态字节码...

    java 反编译工具 jboss-javassist

    2. **学习API**:熟悉Javassist提供的各种类和方法,如`ClassPool`、` CtClass`、`CtMethod`等,这些都是操作字节码的核心类。 3. **创建或修改类**:通过`ClassPool`获取或创建类对象,然后利用`CtClass`的方法来...

    字节码编程和操作系统等知识文档

    通过小傅哥的文章和教程,读者可以学习到如何使用这三种不同的字节码操作框架,以及如何将这些知识应用于全链路监控的实际案例中。通过案例学习,可以更好地将枯燥的字节码编程知识融入实际场景,深化理解并应用于...

    基于 javaagent 对 java 原生类的 方法进行字节码动态修改, 以此引发的一些关于 绕过 Java 软件.zip

    在实际操作中,我们可能会使用如 ASM、ByteBuddy 或 Javassist 这样的字节码库来帮助创建和应用 ClassFileTransformer。这些库提供了友好的 API,让我们能够方便地读取、修改和生成字节码。 然而,这种方法并不总是...

    Android字节码插件,编译期间动态修改代码,改造添加全埋点日志采集功能模块

    字节码插件,如ASM、Javassist或ByteBuddy,是在Java字节码层面操作,可以修改或增强类的功能。例如,我们可以在编译时自动插入代码,以实现埋点逻辑。 全埋点日志采集是通过跟踪用户在应用中的所有操作,记录下每...

    javaagent+javassist

    Javaagent和javassist是Java开发中的两个重要工具,它们在软件开发中有着广泛的应用,特别是在动态代理、代码增强和字节码操作等领域。本篇将详细介绍这两个技术,并结合实际示例进行解析。 首先,让我们来了解`...

    javassist+jd-gui

    3. **字节码修改**:可以插入、删除或替换已有的字节码指令,实现代码的动态增强。 4. **类转换**:`javassist` 提供了转换器,可以将类从一种形式转换为另一种形式,例如从 Java 源码转换为字节码。 另一方面,`jd...

Global site tag (gtag.js) - Google Analytics