JAVA通过ASM字节码修改成员属性的权限修饰符

1. 程序实例

package asm.demo2.modify.access;

import org.apache.commons.lang3.StringUtils;
import org.objectweb.asm.ClassReader;
import org.objectweb.asm.ClassVisitor;
import org.objectweb.asm.ClassWriter;
import org.objectweb.asm.FieldVisitor;
import static org.objectweb.asm.Opcodes.ACC_PUBLIC;
import static org.objectweb.asm.Opcodes.ASM6;
import java.io.IOException;
import java.lang.reflect.Field;

/**
 * 修改成员属性的权限修饰符
 * @date 2018/12/28 22:47
 */
public class ModifyFieldAccess {

    public static class ModifyAccessVisitor extends ClassVisitor {
    
        public ModifyAccessVisitor(int i, ClassVisitor classVisitor) {
            super(i, classVisitor);
        }

        /**
         * 读取到成员属性时,触发该方法
         * @param access 字段的权限修饰符
         * @param name 字段名
         * @param descriptor 字段的描述符
         * @param signature 签名
         * @param value 数值
         * @return
         */
        @Override
        public FieldVisitor visitField(int access, String name, String descriptor, String signature, Object value) {
            //只修改成员属性name的权限,该修改只在内存中修改,如果不写入到class文件中,是不会对原class有任何影响
            if(StringUtils.equals(name,"name")) {
                return super.visitField(ACC_PUBLIC,name,descriptor,signature,value);
            }
            return super.visitField(access, name, descriptor, signature, value);
        }
    }

    public static void main(String[] args) throws IOException, IllegalAccessException, InstantiationException, NoSuchFieldException {

        String fullName = Demo.class.getName();
        String fullNameType = fullName.replace(".","/");
        ClassReader cr = new ClassReader(fullNameType);
        ClassWriter cw = new ClassWriter(0);
        ModifyAccessVisitor mv = new ModifyAccessVisitor(ASM6,cw);
        cr.accept(mv,ClassReader.SKIP_DEBUG);
        byte[] bytes = cw.toByteArray();

        MyClassLoader classLoader = new MyClassLoader();
        Class<?> cls = classLoader.defineClassPublic(fullName, bytes, 0, bytes.length);

        //创建实例对象
        Object o = cls.newInstance();
        Field name = cls.getField("name");
        name.set(o,"kobin");
        Object nameValue = name.get(o);
        System.out.println(nameValue);
        
    	/**未将成员属性age修改为public,所以调用会报错*/
        Field age = cls.getField("age");
        age.set(o,21);
        Object ageValue = age.get(o);
        System.out.println("age:"+ageValue);
    }
}

package asm.demo2.modify.access;

/** 测试实体类
 * @date 2018/12/28 22:48
 */
public class Demo {

    private String name;

    private Integer age;

    public String getName() {
        return name;
    }

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

    public Integer getAge() {
        return age;
    }

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

package asm.demo2.modify.access;

/**
 * @date 2018/12/28 23:04
 */
public class MyClassLoader extends ClassLoader {

    public Class<?> defineClassPublic(String name, byte[] b, int off, int len) throws ClassFormatError {
        Class<?> clazz = defineClass(name, b, off, len);
        return clazz;
    }
}

2. 运行结果

在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值