spring——代理(静/动态代理)

一.静态代理

1.为什么要学习代理模式?因为这就是springAOP的底层!【springAOP 和 springMVC】

代理模式的分类:

  • 静态代理
  • 动态代理

代理的分析:代理实际上就相当于中介,它可以代替主人做事,而且可以做一些主人做不来的事,比如我们的房屋主人会只提供防止给用户,但是如果把房屋主人把房子的售卖权交给中介,中介不但可以帮卖房,而且还会提供看房服务。

所以所代理其实就是对真实对象的拓展,完成拓展的同时也不会改变原始的代码和业务,避免的底层发生改变而导致整个项目都崩掉了。

2.静态代理角色分析:

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

代理模式的好处:

  • 可以使真实角色的操作更加纯粹!不用去关注一些公共的业务
  • 公共的也可以交给代理角色!实践了业务的分工
  • 公共的业务发生拓展的时候,方便集中

缺点:

每产生一个新的真实角色,都会产生一个代理角色,代码量会翻倍,相应的开发效率也会降低~

3.静态代理的代码测试实现步骤:

接口:

package top.lostyou.demo1;
//租房
public interface Rent {
    public void rent();
}

 

真实对象:

package top.lostyou.demo1;
//房东
public class Host implements Rent{
    public void rent() {
        System.out.println("Host出租房子!");
    }
}

 

代理角色:

package top.lostyou.demo1;

public class Proxy implements Rent {
    private Host host;

    public Proxy(Host host) {
        this.host = host;
    }

    public Proxy() {
    }

    public void rent() {
        seeHose();
        host.rent();
        payMoney();
    }
    // 代理的拓展部分,此时原方法都没有发生改动
    public void seeHose(){
        System.out.println("中介带看房!");
    }
    public void payMoney(){
        System.out.println("交付中介费!");
    }
}

 

客户段:

package top.lostyou.demo1;

public class Client {

    public static void main(String[] args) {
        Host host = new Host();
        //代理,虽然我们的输出对象是一样的,但是代理的作用是对真实对象进行拓展,
        //即为拓展,目的是不改变源代码,不改变底层
        Proxy proxy = new Proxy(host);
        proxy.rent();

    }
}

 

二.静态代理的加深印象

接口:

package top.lostyou.demo2;

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

 

真实对象:

package top.lostyou.demo2;
// 真实对象
public class UserServiceImp implements UserService {
    public void add() {
        System.out.println("增加了一个用户");
    }

    public void delete() {
        System.out.println("删除了一个用户");
    }

    public void update() {
        System.out.println("更新了一个用户");
    }

    public void query() {
        System.out.println("查询了一个用户");
    }
    //  为什么要在代理中新增日志,而不是直接在业务中新增
    //  改动业务本来的代码在公司中是大忌
}

 

代理角色:

package top.lostyou.demo2;

public class UserServiceProxy implements UserService  {
    private UserServiceImp imp;

    public UserServiceProxy(UserServiceImp imp) {
        this.imp = imp;
    }

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

    public void delete() {
        log("del");
        imp.delete();
    }

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

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

    // 代理的拓展:日志方法,每使用一次方法都会打印一次日志
    public void  log(String msg){
        System.out.println("使用了"+msg+"方法");
    }
}

 

客户端:

package top.lostyou.demo2;

public class Client {
    public static void main(String[] args) {
        UserServiceImp imp = new UserServiceImp();
        UserServiceProxy proxy = new UserServiceProxy(imp);
        proxy.query();
    }
}

dao

service   <——  log 方法

controller

前端

 

AOP的开发,就像log方法(日志方法)的插入一样是:横向开发的(面向切面)

三.动态代理

  • 动态代理和静态代理的角色一样
  • 动态代理的代理类是动态生成的,不是我们直接写好的!
  • 动态代理分为两大类:基于接口的动态代理,基于类的动态代理
  1. 基于接口----JDK动态代理【我们在这里使用】
  2. 基于类:cglib
  3. Java字节码实现:Javasisit

需要了解两个类:Proxy :代理

InvocationHandler:调用处理程序

接口:

package top.lostyou.demo3;

public interface Rent {
    public void rent();
}

 

真实对象:

package top.lostyou.demo3;

public class Host implements Rent {
    public void rent() {
        System.out.println("Host出租房子!");
    }
}

 

动态代理角色(通过反射机制拿到代理类):

package top.lostyou.demo3;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

// 这个类是实现了InvocationHandler类,运行时会自动生成代理类
public class ProxyInvocationHandler implements InvocationHandler {
    // 被代理的类
    private Rent rent;

    public void setRent(Rent rent) {
        this.rent = rent;
    }
    // 生成得到代理类
    public Object getProxy(){
       return Proxy.newProxyInstance(this.getClass().getClassLoader(),rent.getClass().getInterfaces(),this);
    }
    // 处理代理的实例,通过反射机制,并返回结果
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        // 动态代理的本质,就是通过反射机制实现
     Object result = method.invoke(rent, args);
        return result;
    }
}

 

测试类:

package top.lostyou.demo3;

public class Client {
    public static void main(String[] args) {
        // 真实角色必须要存在
        Host host = new Host();
        // 代理角色现在没有,通过运行时反射去拿到
        ProxyInvocationHandler handler = new ProxyInvocationHandler();
        // 通过调用程序处理角色来处理我们要用的接口对象!
        handler.setRent(host);

        Rent proxy = (Rent)handler.getProxy();

        proxy.rent();
    }
}

 

动态代理的好处:

  • 可以使真实角色的操作更加纯粹!不用去关注一些公共的业务
  • 公共也就交给代理角色!实现了业务的分工!
  • 公共业务发生变化时,方便集中管理!
  • 一个动态代理类的是一个接口,一般就是应对一类业务
  • 一个动态代理可以代理多个类,只要实现了同一个接口

 

posted @ 2023-02-20 22:21  回忆也交给时间  阅读(18)  评论(0编辑  收藏  举报