ASM在运行时通过修改字节码动态给接口添加Annotation

1. 对接口预先添加一个自定义注解:@ServiceType

 

2. 系统启动时,通过Spring扫描有包含注解:@ServiceType的接口,然后通过ASM类库修改接口字节码

 

3. 用到一些辅助工具:java命令,asm-util.jar,asm.jar

============================================================

具体操作:

IIndividualTax接口代码如下(import相关代码省略):

@ServiceType

public interface IIndividualTax {

BusinessResponse request(BusinessRequest request) throws GatewayException;

}

 

目的是要在IIndividualTax 接口上添加注解,并且删掉ServiceType注解,以及在方法request上添加注解,最终想要运行时的代码如下(import相关代码省略):

@Path("it")

@Consumes({"application/json", "text/xml"})

@Produces({"application/json; charset=UTF-8", "text/xml; charset=UTF-8"})

public abstract interface IIndividualTax

{

  @POST

  @Path("request")

  BusinessResponse request(BusinessRequest paramBusinessRequest) throws GatewayException;

}

 

首先:java -classpath D:\;D:\asm-util-5.1.jar;D:\asm-5.1.jar;. org.objectweb.asm.util.ASMifier cn.com.*.*.gateway.service.IIndividualTax

运行这个命令,需要注意的是classpath参数以及,在当前cmd位置下要有IIndividualTax这个类的全包路径

借助ASMifier类,可以看到这个接口IIndividualTax修改前完整的字节码,如下:

package asm.*****************************.service;

import java.util.*;

import org.objectweb.asm.*;

public class IIndividualTaxDump implements Opcodes {

 

public static byte[] dump () throws Exception {

 

ClassWriter cw = new ClassWriter(0);

FieldVisitor fv;

MethodVisitor mv;

AnnotationVisitor av0;

 

cw.visit(V1_7, ACC_PUBLIC + ACC_ABSTRACT + ACC_INTERFACE, "cn/com/*/*/gateway/service/IIndividualTax", null, "java/lang/Object", null);

 

{

av0 = cw.visitAnnotation("Lcn/com/*/*/gateway/service/ServiceType;", false);

av0.visitEnd();

}

{

mv = cw.visitMethod(ACC_PUBLIC + ACC_ABSTRACT, "request", "(Lcn/com/*/*/gateway/dto/BusinessRequest;)Lcn/com/*/*/gateway/dto/BusinessResponse;", null, new String[] { "cn/com/*/*/gateway/exception/GatewayException" });

mv.visitEnd();

}

cw.visitEnd();

 

return cw.toByteArray();

}

}

 =================================================================================================================

 

在次运行:java -classpath D:\;D:\asm-util-5.1.jar;D:\asm-5.1.jar;. org.objectweb.asm.util.ASMifier cn.com.*.*.gateway.service.IIndividualTax,查看想要修改后接口完整的字节码,如下:

package asm.*************************************service;

import java.util.*;

import org.objectweb.asm.*;

public class IIndividualTaxDump implements Opcodes {

 

public static byte[] dump () throws Exception {

 

ClassWriter cw = new ClassWriter(0);

FieldVisitor fv;

MethodVisitor mv;

AnnotationVisitor av0;

 

cw.visit(V1_7, ACC_PUBLIC + ACC_ABSTRACT + ACC_INTERFACE, "cn/com/*/*/gateway/service/IIndividualTax", null, "java/lang/Object", null);

 

{

av0 = cw.visitAnnotation("Ljavax/ws/rs/Path;", true);

av0.visit("value", "it");

av0.visitEnd();

}

{

av0 = cw.visitAnnotation("Ljavax/ws/rs/Consumes;", true);

{

AnnotationVisitor av1 = av0.visitArray("value");

av1.visit(null, "application/json");

av1.visit(null, "text/xml");

av1.visitEnd();

}

av0.visitEnd();

}

{

av0 = cw.visitAnnotation("Ljavax/ws/rs/Produces;", true);

{

AnnotationVisitor av1 = av0.visitArray("value");

av1.visit(null, "application/json; charset=UTF-8");

av1.visit(null, "text/xml; charset=UTF-8");

av1.visitEnd();

}

av0.visitEnd();

}

{

av0 = cw.visitAnnotation("Lcn/com/*/*/gateway/service/ServiceType;", false);

av0.visitEnd();

}

{

mv = cw.visitMethod(ACC_PUBLIC + ACC_ABSTRACT, "request", "(Lcn/com/*/*/gateway/dto/BusinessRequest;)Lcn/com/*/*/gateway/dto/BusinessResponse;", null, new String[] { "cn/com/*/*/gateway/exception/GatewayException" });

{

av0 = mv.visitAnnotation("Ljavax/ws/rs/POST;", true);

av0.visitEnd();

}

{

av0 = mv.visitAnnotation("Ljavax/ws/rs/Path;", true);

av0.visit("value", "request");

av0.visitEnd();

}

mv.visitEnd();

}

cw.visitEnd();

 

return cw.toByteArray();

}

}

 

 

===================================================================================================================

 

对于新手玩ASM不熟悉的情况,通过对比上面的二段字节码存在的差异,用ASM提供的API进行码代码,生成运行时的关键代码如下:

 

public class TaxVisitor extends ClassVisitor implements Opcodes{

 

public TaxVisitor(int api, ClassVisitor cv) {

super(api, cv);

}

 

@Override

public MethodVisitor visitMethod(int access, String name, String desc, String signature, String[] exceptions) {

MethodVisitor mv = super.visitMethod(access, name, desc, signature, exceptions);

if (name.equals("request")) {

AnnotationVisitor  av1 = mv.visitAnnotation("Ljavax/ws/rs/POST;", true);//在request上添加@POST注解,对比看上面红色粗体类源码

AnnotationVisitor  av2 = mv.visitAnnotation("Ljavax/ws/rs/Path;", true);//同上

av2.visit("value", "request");

av2.visitEnd();//这些api用法,查看asm相关说明

av1.visitEnd();

}

 

return mv;

}

 

@Override

public AnnotationVisitor visitAnnotation(String name, boolean arg1) {

if (name.equals("Lcn/com/*/*/gateway/service/ServiceType;")) {

return null;//返回null说明,要删掉接口上@ServiceType注解

}

 

return super.visitAnnotation(name, arg1);

}

}

 

=====================================================================================================

public class AsmTest extends ClassLoader{

 

public static void main(String[] args) throws Exception {

 

ClassReader cr=new ClassReader(IIndividualTax.class.getName());

ClassWriter cw=new ClassWriter(ClassWriter.COMPUTE_MAXS);

 

AnnotationVisitor av0 = cw.visitAnnotation("Ljavax/ws/rs/Path;", true);//在接口上添加注解@Path

av0.visit("value", "it");

av0.visitEnd();

 

AnnotationVisitor av1 = cw.visitAnnotation("Ljavax/ws/rs/Consumes;", true);//在接口上添加注解@Consumes

AnnotationVisitor av2 = av1.visitArray("value");

av2.visit(null, "application/json");

av2.visit(null, "text/xml");

av2.visitEnd();

av1.visitEnd();

 

AnnotationVisitor av3 = cw.visitAnnotation("Ljavax/ws/rs/Produces;", true);//在接口上添加注解@Produces

AnnotationVisitor av4 = av3.visitArray("value");

av4.visit(null, "application/json; charset=UTF-8");

av4.visit(null, "text/xml; charset=UTF-8");

av4.visitEnd();

av3.visitEnd();

 

TaxVisitor myv=new TaxVisitor(Opcodes.ASM4,cw);

cr.accept(myv, 0);

 

byte[] code=cw.toByteArray();//最终想要类的字节码

 

AsmTest loader=new AsmTest();

Class<?> appClass=loader.defineClass(null, code, 0,code.length);//字节码加载到jvm

System.out.println(AnnotationUtils.findAnnotation(appClass, Path.class));

FileOutputStream fos = new FileOutputStream(appClass.getResource("").getPath()+"/IIndividualTax.class");//覆盖当前class文件

fos.write(code);

fos.close();

 

}

}

 

注意:对类或接口的字节码的修改必须用ClassWriter的实例进行api相关操作,对类的方法,属性字节码操作需要通过对应的Visitor操作,最终调用visitEnd()

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Android运行时修改字节码是指在Android应用程序的运行过程中,通过修改字节码文件来动态改变程序的行为。这种技术一般用于实现一些高级的功能或者实现对已有程序的增强,有助于优化性能、扩展功能等。 在Android系统中,Dalvik虚拟机是负责解释和执行字节码的,而字节码是由Java源代码编译而来的。通过Dalvik的类加载器机制,我们可以在运行时字节码文件进行修改和替换。 Android运行时修改字节码的具体实现方法有很多,常见的一种是使用字节码操控库,例如ASM(Objectweb ASM),它是一个强大的Java字节码操控库,可以用来读取、修改、生成Java字节码。开发者可以使用ASM库对Android应用程序的字节码进行修改,例如插入新的方法修改方法体、修改字段等。 通过在应用程序的关键位置插入修改字节码的代码,可以实现一些有趣的功能。例如,我们可以通过修改字节码来实现方法动态插桩,即在方法的开头和结尾处插入额外的代码,用于统计方法的执行时间、记录方法的调用顺序等。这对于性能分析和调试非常有帮助。 另外,Android运行时修改字节码也可以用于实现一些热修复的功能。我们可以在应用程序的运行过程中下载新的补丁包,然后通过修改字节码,将补丁代码动态地插入到已有的方法中,从而修复程序中的bug或者实现新功能。 总之,Android运行时修改字节码是一种高级的技术手段,可以帮助开发者实现一些灵活而强大的功能,但也需要慎重使用,因为错误的字节码修改可能导致应用崩溃或者安全风险。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值