-
静态代理
由程序员创建或特定工具自动生成源代码,再对其编译,在程序运行前,代理类的.class文件就已经存在了。
原理:
对普通一个接口与一直实现类在加入一个代理类,用于包装实现类中实现的方法,而业务使用方只需要实例化代理类,并传递需要代理的普通类即可。
优点:
编译时生成代码,性能高
缺点:
-
一个代理类只能为一个接口服务,开发中必然会产生过多的代理
-
所有的代理操作除了调用的方法不一样之外,其他的操作都一样,则此时肯定是重复代码
代码示例
/**
* 通过代理类,实现代理接口,通过构造进行实现的代理,在每个方法里面进行日志捕获
* <pre>
* 外部实现权限验证的时候,只需要之所想该方法即可,不需要再去实现UserImpl方法了
* <pre>
*/
public class UserProxy implements IUser {
private UserImpl userImpl;
// 构造的时候直接传入代理实现类
public UserProxy(UserImpl userImpl) {
super();
this.setUserImpl(userImpl);
}
@Override
public boolean isAuthUser(int uid) {
System.out.println("proxy insert msg:准备权限验证,有必要这里可以发送消息到MQ,做实时登录验证次数预警处理");
boolean b = userImpl.isAuthUser(uid);
System.out.println("proxy insert msg:验证完成,做一些清理等工作.....");
return b;
}
// ***********get,set************
public UserImpl getUserImpl() {
return userImpl;
}
public void setUserImpl(UserImpl userImpl) {
this.userImpl = userImpl;
}
}
public static void main(String[] args) {
UserProxy userProxy = new UserProxy(new UserImpl());
userProxy.isAuthUser(5);
}
-
动态代理
原理:
动态代理类的字节码在程序运行时由Java反射机制动态生成,无需程序员手工编写它的源代码。
优点
可以通过一个代理类,完成所有代理工作,不需要向静态代理需要一个一个实现接口来代理
缺点
通过反射动态代理方法将消耗系统性能,如果非常多的话,性能比较低
JDK动态代理
原理:JDK的动态代理机制只能代理实现了接口的类,而不能实现接口的类就不能实现JDK的动态代理
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/**
* JDK动态代理代理类,可以将InvocationHandler接口的子类想象成一个代理的最终操作类
* JDK的动态代理依靠接口实现,如果有些类并没有实现接口,则不能使用JDK代理,这个也是缺陷
* @author Jason
*
*/
public class UserJDKProxy implements InvocationHandler {
// 需要代理的类
private Object target;
public UserJDKProxy() {
super();
}
/**
* 绑定委托对象并返回一个代理类 ClassLoader loader:类加载器 Class<?>[] interfaces:得到全部的接口 InvocationHandler
* h:得到InvocationHandler接口的子类实例
*
* @param target
* @return
*/
public Object initUserJDKProxy(Object target) {
this.target = target;
// 可以看出这里的第二个参数是获取接口,那么也就是说我们实现代理,需要类去实现接口,在有的时候,类是没有接口的,所以这里是一个缺陷
return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this);
}
/**
* 调用具体的方法
*
* @param proxy
* 指被代理的对象
* @param method
* 要调用的方法
* @param args
* 方法调用时所需要的参数
* @return
* @throws Throwable
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("proxy:这里拦截处理一下事情....,如监控参数、插入日志....");
//传入处理对象和参数
Object object = method.invoke(target, args);
System.out.println("proxy:这里做一些收尾工作....");
return object;
}
}
UserJDKProxy userJDKProxy=new UserJDKProxy();
IUser iUser=(IUser) userJDKProxy.initUserJDKProxy(new UserImpl());
iUser.isAuthUser(19);
CGLib动态代理
原理:cglib是针对类来实现代理的,他的原理是对指定的目标类生成一个子类,并覆盖其中方法实现增强,但因为采用的是继承,所以不能对final修饰的类进行代理。
import java.lang.reflect.Method;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
/**
* 采用CGLib方式动态代理类
* @author Jason
*
*/
public class UserCglibProxy implements MethodInterceptor {
private Object target;
public Object getInstance(Object target) {
this.target = target;
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(target.getClass());
//回调方法
enhancer.setCallback(this);
//创建代理对象
return enhancer.create();
}
// 回调方法
@Override
public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
System.out.println("可以做一些监控、预警等工作...");
Object object = proxy.invokeSuper(obj, args);
System.out.println("收尾工作....");
return object;
}
}
UserCglibProxy userCglibProxy=new UserCglibProxy();
UserImpl userImpl=(UserImpl) userCglibProxy.getInstance(new UserImpl());
userImpl.isAuthUser(50);