【Java】反射、动态代理

反射允许对封装类的字段,方法和构造函数的信息进行编程访问

获取Class对象

  • Class.forName(“全类名”)===源代码阶段

  • 类名.class === 加载阶段

  • 对象.getClass ===运行阶段

Class c1 = Class.forName("Test.Student");

Class c2 = Student.class;

Student s = new Student();
Class c3 = s.getClass();

获取构造方法

暴力反射

临时取消权限的校验,可以创建私有方法的对象

con.setAccessible(true);
Student stu = (Student) con.newInstance("张三", 23);

获取成员变量

//获取单个成员变量
Field name = c.getDeclareField("name");

//获取权限修饰符
int m = name.getModifiers();

//获取成员变量名字
String n = name.getName();

//获取成员变量数据类型
Class type = name.getType();

//获取成员变量记录的值
Student s = new Student("zhangsan", 23);
name.setAccessible(true);
String vlaue = (String) name.get(s);

//修改对象里面记录的值
name.set(s, "lisi");

获取成员方法

作用

  • 获取一个类里面的所有信息。获取之后再执行其他的业务逻辑

  • 结合配置文件,动态的创建对象并调用方法

把对象里面所有的成员变量名和值保存到本地文件中

public static void saveObject(Object o) throws IOException, IllegalAccessException {
        Class c = o.getClass();

        //创建IO流
        BufferedWriter bw = new BufferedWriter(new FileWriter("a.txt"));
        
        //获取所有的成员变量
        Field[] field = c.getDeclaredFields();
        for (Field f : field) {
            f.setAccessible(true);
            //获取成员变量的名字
            String name = f.getName();
            //获取成员变量的值
            Object value = f.get(o);
            //写出数据
            bw.write(name + "=" + value);
            bw.newLine();
        }
        
        bw.close();
    }

反射与动态文件相结合,动态创建对象并调用方法

public static void main(String[] args) throws IOException, ClassNotFoundException, NoSuchMethodException, InvocationTargetException, InstantiationException, IllegalAccessException {
        //1.读取配置文件中的信息
        Properties p = new Properties();
        FileInputStream fis = new FileInputStream("a.txt");
        p.load(fis);
        fis.close();
        
        //2.获取全类名和方法名
        String className = (String) p.get("classname");
        String methodName = (String) p.get("method");
        
        //3.利用反射创建对象并运行方法
        Class c = Class.forName(className);
        
        //获取构造方法
        Constructor con = c.getDeclaredConstructor();
        Object o = con.newInstance();
        
        //获取成员方法并运行
        Method m = c.getDeclaredMethod(methodName);
        m.setAccessible(true);
        m.invoke(o);
    }

动态代理

无侵入式的给代码增加额外的功能

代理里面就是要对象被代理的方法

通过接口保证,后面的对象和代理需要实现同一个接口,接口中就是被代理的所有方法

Star star = (Star) Proxy.newProxyInstance(
    ProxyUtil.class.getClassLoader(),//指定用哪个类的加载器,去加载生成的代理类
    new Class[]{Star.class},//指定接口,这些接口用于指定生成的代理长什么样,有哪些方法
    new InvocationHandler() {//用来指定生成的对象要干什么事情
        public Object invoke(Object proxy, Method method, Object[] args) throw Throwable {
            if ("sing".equals(method.getName())){
              System.out.println("开始唱歌");}
            return method.invoke(star, args);
        }
    };
    return star;
    }
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

啊呜冷

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值