动态代理(底层:反射)
优点:
- 可以使真实角色的操作更加纯粹,不需要关注一些公共的业务
- 公共部分交给代理角色,实现了业务分工
- 公共业务发生扩展,方便集中管理
- 一个动态代理类代理的是一个接口,一般对应的一类业务
- 一个动态代理类可以代理多个类,只要是实现了同一个接口即可
package com.ss.demo04;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
//自动生成代理类
public class ProxyInvocationHandler implements InvocationHandler {
//被代理的接口
private Object target;
public void setTarget(Object target) {
this.target = target;
}
//生成得到代理类
public Object getProxy(){
return Proxy.newProxyInstance(this.getClass().getClassLoader(), target.getClass().getInterfaces(),this);
}
//处理代理实例,并返回结果
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
Object result = method.invoke(target, args);
return result;
}
}
package com.ss.demo02;
public class UserServiceImpl implements UserService{
@Override
public void add() {
System.out.println("增加了一个用户");
}
@Override
public void delete() {
System.out.println("删除了一个用户");
}
@Override
public void update() {
System.out.println("更新了一个用户");
}
@Override
public void query() {
System.out.println("查找了一个用户");
}
}
package com.ss.demo04;
import com.ss.demo02.UserService;
import com.ss.demo02.UserServiceImpl;
public class Client {
public static void main(String[] args) {
//实例化真实对象
UserServiceImpl userService = new UserServiceImpl();
//此时代理对象不存在,调用 处理程序
ProxyInvocationHandler pih = new ProxyInvocationHandler();
pih.setTarget(userService);//设置要代理的对象
//动态生成代理类
UserService proxy = (UserService) pih.getProxy();
proxy.add();
}
}
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable
//proxy - 调用该方法的代理实例
//method -所述方法对应于调用代理实例上的接口方法的实例。方法对象的声明类将是该方法声明的接口,它可以是代理类继承该方法的代理接口的超级接口。
//args -包含的方法调用传递代理实例的参数值的对象的阵列,或null如果接口方法没有参数。原始类型的参数包含在适当的原始包装器类的实例中,例如java.lang.Integer或java.lang.Boolean 。