反射的作用?
可以在运行时的到一个类的全部成分然后操作
可以破坏封装性
可以破坏泛型的约束性
更重要的用途是:做java高级框架
注解(annotation)
对java中类、方法、成员变量做标记,然后进行特殊处理,至于到底做什么处理,由业务需求决定
元注解
就是注解注解的注解
有两个
@Target :约束自定义注解只能在哪些地方使用 ElementType枚举类
TYPE 类
FIELD成员变量
METHOD成员方法
PARAMETER方法参数
CONSTRUCTOR 构造器
LOCAL_VARIABLE 局部变量
@Retention(n.保留物、保存):申明注解的生命周期RententionPolicy
SOURCE注解只作用在源码阶段 生成的字节码文件中不存在
CLASS 注解作用在源码阶段 字节码文件阶段 运行阶段不存在 默认值
RUNTIME注解作用在源码阶段 、字节码文件阶段、运行阶段(开发阶段)长生不si
注解解析
就是判断是否存在注解,存在直接就解析出内容
Annotation:注解的顶级接口,注解都是Annotation类型的对象
AnnotatedElement:该接口定义可与注解解析相关的解析方法
动态代理
代理就是被代理者没有能力或者不愿意去完成某件事情,需要找个人代替自己去完成这件事,动态代理就是用来对业务功能(方法)进行代理的
关键步骤:
1.必须有接口,实现要实现接口(代理通常是基于接口实现的)
2.创建一个实现类对象,该对象为业务对象,紧接着为业务对象做一个代理对象
Proxy类下
研究代码流程先看下各参数含义 就大致可以了解流程了
package com.itheima.d9_proxy;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/**
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)
参数一:类加载器,负责加载代理类到内存中使用。
参数二:获取被代理对象实现的全部接口。代理要为全部接口的全部方法进行代理
参数三:代理的核心处理逻辑
*/
public class ProxyUtil {
/**
生成业务对象的代理对象。
* @param obj
* @return
*/
public static <T> T getProxy(T obj) {
// 返回了一个代理对象了
return (T)Proxy.newProxyInstance(obj.getClass().getClassLoader(),
obj.getClass().getInterfaces(),
new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
// 参数一:代理对象本身。一般不管
// 参数二:正在被代理的方法
// 参数三:被代理方法,应该传入的参数
long startTimer = System .currentTimeMillis();
// 马上触发方法的真正执行。(触发真正的业务功能)
Object result = method.invoke(obj, args);
long endTimer = System.currentTimeMillis();
System.out.println(method.getName() + "方法耗时:" + (endTimer - startTimer) / 1000.0 + "s");
// 把业务功能方法执行的结果返回给调用者
return result;
}
});
}
}
动态代理的优点:
非常灵活,支持任意接口类型的实现类对象做代理,也可以直接为接口本身做代理
可以为代理对象的所有方法做代理
可以在不改变方法源码的情况下,实现对方法功能的增强。
不仅简化了编程工作、提高了软件系统的可扩展性,同时也提高了开发效率
切面编程