使用javassist修改类字节码时关于范型一个小坑

javassist作为一款低门槛的java字节码修改工具,允许用户以java代码的方式修改java类文件,无需对类文件的结构有深入的了解。

最近,在公司项目中使用了javassist,使用过程中遇到了一个和范型有关的小坑,特此记录。

代码如下:

        StringBuffer stringBuffer = new StringBuffer();
        stringBuffer.append("{\n");
        stringBuffer.append("Map<String, String> tags = new HashMap<>();");
        stringBuffer.append("\n}");

        newCtMethod.setBody(stringBuffer.toString());
        ctClass.addMethod(newCtMethod);

此时执行报错,错误栈为

Exception in thread "main" javassist.CannotCompileException: [source error] ; is missing
	at javassist.CtBehavior.setBody(CtBehavior.java:446)
	at javassist.CtBehavior.setBody(CtBehavior.java:412)
	at com.jstudy.javassist.EnhanceJedis.genMethod(EnhanceJedis.java:19)
	at com.jstudy.javassist.EnhanceJedis.main(EnhanceJedis.java:30)
Caused by: compile error: ; is missing
	at javassist.compiler.Parser.parseDeclarationOrExpression(Parser.java:594)
	at javassist.compiler.Parser.parseStatement(Parser.java:277)
	at javassist.compiler.Parser.parseBlock(Parser.java:289)
	at javassist.compiler.Parser.parseStatement(Parser.java:243)
	at javassist.compiler.Javac.compileBody(Javac.java:214)
	at javassist.CtBehavior.setBody(CtBehavior.java:438)
	... 3 more

可以看到,javassist编译器认为内容出错。

经过google,得知错误在于:javassist中写入函数体中含有范型时时:

1.对于范型符号需要特殊处理

2.对饮用的外部类显式声明包路径

将代码改为如下:


        StringBuffer stringBuffer = new StringBuffer();
        stringBuffer.append("{\n");
        stringBuffer.append("java.util.Map/*<String, String>*/ tags = new java.util.HashMap/*<>*/();");
        stringBuffer.append("return null;\n}");

        newCtMethod.setBody(stringBuffer.toString());
        ctClass.addMethod(newCtMethod);

正常运行。

发布了15 篇原创文章 · 获赞 3 · 访问量 1万+
展开阅读全文

关于javassist重复加载的问题

11-05

最近需要使用javassist动态改变类里的方法,问题是javassist使用过一次以后,第二次使用就会冻结,可是在第二次就算用了解冻,调用toclass以后还是会报异常,主要是为了得到frmAttributesI的实体,怎样已经冻结的class解冻之后还可以继续编辑修改并调用呢? String body = null; CtClass cc = null; IFrmAttributes frmAttributesI = null; //从数据查找判断是否是二次加载 String isCondUpdate = "Y"; if(isClassInit == false){ //第一次调用 // 获取所有条件信息(执行脚本) ClassPool pool = ClassPool.getDefault(); //导入相关jar包 pool.importPackage("java.util.Hashtable"); pool.importPackage("java.util.List"); pool.importPackage("java.util.Date"); pool.importPackage("java.text.SimpleDateFormat"); // 在外部变量执行类中,动态加载外部变量执行算法 cc = pool.get("com.process.FrmAttributes"); CtMethod cm = cc.getDeclaredMethod("process"); //cc.defrost(); System.out.println("测试1=================="+body.toString()); cm.setBody("{" + body.toString() + "}"); Class c = cc.toClass(); frmAttributesI = (FrmAttributes) c.newInstance(); this.isClassInit = true; }else{ if(isCondUpdate == "Y"){ frmAttributesI = null; ClassPool pool2 = ClassPool.getDefault(); cc = pool2.get("com.process.FrmAttributes"); if(cc.isFrozen()){ cc.defrost(); } //删除方法,增加方法,试试二次加载 //第二次调用 CtMethod cm2 = cc.getDeclaredMethod("process"); System.out.println("测试2============================"+body.toString()); cm2.setBody("{" + body.toString() + "}"); cc.writeFile(); Class c = cc.toClass(); frmAttributesI = (FrmAttributes) c.newInstance(); } } 问答

javassist一个很奇怪的问题

08-20

采用javassist来将一个类中的field改变类型。待改变的类代码如下: [code="java"]public class JassistTest { @Autowired private StpService stpService; public void doit(Long userid){ PeakSeasonMainResponse res = stpService.getPeakOverView(userid); System.out.println(stpService.getClass().getName()); System.out.println("hello" + res); } }[/code] 进行改变操作的类如下: [code="java"]public class TestMain { public TestMain(){ } public static void main(String[] args) throws Exception{ ClassPool pool = ClassPool.getDefault(); CtClass cc = pool.get("JassistTest"); CtField f = cc.getDeclaredField("stpService"); cc.removeField(f); cc.addField(CtField.make("private Proxy stpService;", cc)); cc.writeFile("D:\\DevProgram\\eclipse-jee-kepler-R-win32\\workspace\\stable\\fc-deimos\\target\\test-classes"); JassistTest test = new JassistTest(); test.doit(7060L); } } [/code] class文件生成后,我用反编译工具查看,Javassist已经变为如下: [code="java"]public class JassistTest { private Proxy stpService; public void doit(Long userid) { PeakSeasonMainResponse res = this.stpService.getPeakOverView(userid); System.out.println(this.stpService.getClass().getName()); System.out.println("hello" + res); } }[/code] 但当我执行TestMain.java文件中的test.doit()方法时,还是报异常了。 [code="java"]Exception in thread "main" java.lang.NoSuchFieldError: stpService at JassistTest.doit(JassistTest.java:13) at TestMain.main(TestMain.java:33) [/code] 实在搞不懂,请教大家 问答

没有更多推荐了,返回首页

©️2019 CSDN 皮肤主题: 编程工作室 设计师: CSDN官方博客

分享到微信朋友圈

×

扫一扫,手机浏览