java中的代理
静态代理
1.要求被装饰者和装饰者实现同一个接口或者继承同一个类
2.装饰者中要有被装饰者的引用
3.对需要加强的方法进行增强
4.对不需要加强的方法调用原来的方法
动态代理
在程序运行的时候,动态的创建一个对象,用这个对象去操作方法方法
jdk的中Proxy ,前提:必须实现一个接口
Object Proxy.newProxyInstance(ClassLoader 被代理对象的类加载器,Class[] 被代理对象实现的所有接口,InvocationHandler 处理方法);
InvocationHandler:接口 只需要重写一个方法
Object invoke(Object 代理对象,Method 当前执行的方法,Object[] 当前方法执行的时候需要的参数)
返回值就是 当前方法执行之后的返回值
这次修改在之前的 解耦合的基础上操作的
添加动态代理之后的代码
package utils;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import org.dom4j.Document;
import org.dom4j.Element;
import org.dom4j.io.SAXReader;
public class BeanFactory {
public static Object getBean(String id) {
//通过id从bean.xml中查找对应的class
try {
SAXReader saxReader = new SAXReader();
//获取document对象
Document document = saxReader.read(BeanFactory.class.getClassLoader().getResourceAsStream("bean.xml"));
//获取指定的bean对象 xpath
Element ele = (Element) document.selectSingleNode("//bean[@id='"+id+"']");
//获取bean对象的class属性的值
String className = ele.attributeValue("class");
//反射 以前的逻辑直接返回的是实例
//return Class.forName(value).newInstance();
//本次代理的代码演示
Object obj = Class.forName(className).newInstance();
//判断是否是service的实现类
if(id.endsWith("Service")){
//若是实现类就添加操作
Object proxyObj = Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj.getClass().getInterfaces(), new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//判断方法是否是需要的制定方法
if("findByPage".equals(method.getName())) {
System.out.println("已经执行了findByPage操作。");
//返回函数的调用
return method.invoke(obj, args);
}
//不是指定方法没有附加操作直接执行
return method.invoke(obj, args);
}
});
//若是service方法返回的是代理对象
return proxyObj;
}
//若不是service实现类直接返回obj
return obj;
} catch (Exception e) {
System.out.println("输入的bean不存在。");
e.printStackTrace();
}
return null;
}
}