1、静态代理(装饰者模式)
//1、首先创建一个接口,定义一个方法
public interface UserService {
/**
* 模拟装修
*/
void save();
}
//2、编写接口的实现类,这里的save() 方法可以理解为原始方法
public class UserServiceImpl implements UserService {
@Override
public void save() {
System.out.println("开始给墙刷大白");
}
}
//3、编写代理类,实现此接口,定义成员变量和带参构造方法,然后重写save方法,调用原save方法后编写增强代码
public class Decorator01 implements UserService {
private UserService userService;
public Decorator01(UserService userService) {
this.userService = userService;
}
@Override
public void save() {
userService.save();
System.out.println("方法加强: 开始贴瓷砖了");
}
}
//4、编写测试类,测试方法是否被增强,需要注意的是,代理类需要传入实现类的对象参数
@Test
public void test01() {
UserServiceImpl userService = new UserServiceImpl();
userService.save();
System.out.println("-----------------");
Decorator01 decorator01 = new Decorator01(userService);
decorator01.save();
}
静态代理
优点:不改变原有代码的情况下对方法进行了增强,提高了代码的复用性
缺点:代理对象仅支持当前接口,多个接口需要编写多个实现类
且接口中添加方法时,所有实现类和代理类都要实现新添加的方法
2、动态代理
2.1 基于接口(jdk默认)
//基于接口的动态代理,格式比较固定,只需要在重写的invoke方法里添上增强代码即可
public class JDKProxy {
public UserService getUserServiceImpl(UserService userService){
UserService proxy = (UserService) Proxy.newProxyInstance(
userService.getClass().getClassLoader(),
userService.getClass().getInterfaces(),
new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
Object rtValue = method.invoke(userService, args);
System.out.println("贴壁纸了");
//匹配方法名进行增强
String methodName = method.getName();
if ("save".equals(methodName)){
System.out.println("匹配到了save方法,再一次给墙装修,挂上字画");
}
return rtValue;
}
}
);
return proxy;
}
}
//对基于jdk动态代理类进行测试
@Test
public void test001(){
UserServiceImpl userService = new UserServiceImpl();
UserService proxy = new JDKProxy().getUserServiceImpl(userService);
proxy.save();
}
基于jdk的动态代理
优点:不修改原有代码对方法进行增强,提高代码的复用性
缺点:只能通过接口方式进行代理,若没有接口就不能实现此种动态代理
2.2 基于子类
//基于子类的动态代理,格式依旧比较固定,在重写的方法里添上增强代码
public class TestCglibProxy {
public UserService getCglibProxy(Class clazz){
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(clazz);
enhancer.setCallback(new MethodInterceptor() {
@Override
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
Object rtValue = null;
if("save".equals(method.getName())){
rtValue = methodProxy.invokeSuper(o,objects);
System.out.println("我使用cglib代理对save方法进行了增强");
}
return rtValue;
}
});
return (UserService) enhancer.create();
}
}
//测试基于cglib的动态代理
@Test
public void tes002(){
UserService cglibProxy = new TestCglibProxy().getCglibProxy(UserServiceImpl.class);
cglibProxy.save();
}
优点:不改变原有代码的情况下对方法进行了增强,提高了代码的复用性
缺点:只能通过继承关系来实现动态代理,若被代理类final修饰则不适用
(spring 里可以通过配置来选择动态代理的方式,默认是基于接口方式)
静态/动态代理区别:
静态代理在原方法执行前就已经编译
动态代理是在执行到原始方法时才开始增强原方法