代理模式

代理模式

代理模式中,代理类与主体类实现同样的接口,代理类持有实体类的引用,并接受客户端对代理类中实体引用的外部注入,并代理实体类的功能。
注:描述中的这种外部注入形式有个专有技术名词:依赖注入

在这里插入图片描述

静态代理模式

角色

  • 抽象角色:一般会使用接口或者抽象类来解决
  • 真实角色:被代理的角色
  • 代理角色:代理真实角色,一般会有一些附属操作
  • 客户:访问代理对象的角色

优点

  • 可以使真实角色更加集中本身的业务,不用去关注一些公共的业务
  • 公共业务交给代理角色,实现业务的分工
  • 公共业务发生扩展时,方便集中管理

缺点

每多一个真实角色就会产生一个代理角色:代码量会翻倍,开发效率低

代码步骤

1.接口

public interface UsersService {
    public void add();
    public void delete();
    public void update();
    public void query();
}

2.真实角色

public class UserServiceImp implements UsersService{
    public void add() {
        System.out.println("添加");
    }

    public void delete() {
        System.out.println("删除");
    }

    public void update() {
        System.out.println("修改");
    }

    public void query() {
        System.out.println("查询");
    }
}

3.代理角色

public class UserServiceProxy implements UsersService{
    private UserServiceImp userService;

    public void setUserService(UserServiceImp userService) {
        this.userService = userService;
    }

    public void add() {
        log("add");
        userService.add();
    }

    public void delete() {
        log("delete");
        userService.delete();
    }

    public void update() {
        log("update");
       userService.update();
    }

    public void query() {
        log("query");
        userService.query();
    }

    public void log(String msg){
        System.out.println("使用了"+msg+"方法");
    }
}

4.客户访问代理角色

public class User {
    public static void main(String[] args) {
        UserServiceImp userService = new UserServiceImp();
        UserServiceProxy proxy = new UserServiceProxy();
        proxy.setUserService(userService);

        proxy.add();
    }
}

运行结果:

使用了add方法
添加

Process finished with exit code 0

动态代理模式

  • 和静态代理模式中的角色一样
  • 动态代理的具体代理类的生成是在运行期动态产生的,而非编译期就已经静态存在;
  • 动态代理分为:
    • 基于接口:JDK动态代理
    • 基于类:CGLIB动态代理
    • Java字节码实现:javassist

​ JDK中的动态代理是通过反射类Proxy以及InvocationHandler回调接口实现的;但是,JDK中所要进行动态代理的类必须要实现一个接口,也就是说只能对该类所实现接口中定义的方法进行代理,这在实际编程中具有一定的局限性,而且使用反射的效率也并不是很高。

JDK动态代理核心代码:

1.实现这个接口InvocationHandler,创建代理对象

2.Proxy提供创建动态代理类和实例的静态方法:Proxy.newProxyInstance()

//自动生成代理类
public class DynamicProxy implements InvocationHandler {

    //被代理的接口
    private Object target;

    //set方法
    public void setTarget(Object target) {
        this.target = target;
    }

    //生成得到代理类
    public Object getProxy(){
        return Proxy.newProxyInstance(this.getClass().getClassLoader(),
                target.getClass().getInterfaces(),this);
    }

    //处理代理实例并返回结果
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        log(method.getName());
        Object result = method.invoke(target,args);
        return result;
    }
	//代理角色的附属业务的例子
    public void log(String msg){
        System.out.println("使用了"+msg+"方法");
    }
}

target可表示任务场景中的真实角色,这是一个模板!

客户实现业务:

public class User {
    public static void main(String[] args) {
        //真实角色
        UserServiceImp userService = new UserServiceImp();
        //代理角色,因为是动态代理所以不存在
        DynamicProxy dp = new DynamicProxy();
        //设置代理对象
        dp.setTarget(userService);

        //动态生成代理类
        UsersService proxy =(UsersService) dp.getProxy();
		
        //调用真实角色中的方法
        proxy.delete();
    }
}