Spring学习五:代理

代理

为什么么要学习代理模式,因为aop的底层机制就是动态代理;

代理模式:

  • 静态代理
  • 动态代理

静态代理

代理模式的角色分析:

抽象角色:一般会使用抽象类或者接口实现

真实角色:被代理的角色

代理角色:代理真实角色,代理真实角色后,一般会做一些附属的操作;

客户:使用代理角色进行一些操作

在这里插入图片描述

代码实现例1—出租房子

创建一个租房的接口

package com.westos.staticproxy;

//出租的接口:抽象
public interface Rent {

    //出租
    void rent();
}

真实对象–真正的房东

package com.westos.staticproxy;

//这个房子要出租
//真实对象
public class Host implements Rent{
    //出租
    public void rent(){
        System.out.println("房东要出租房子");
    }
}

代理对象–代理

package com.westos.staticproxy;
//代理对象
//房屋中介--代理
public class Proxy  implements  Rent{
      //房东
    private Host host;

    public void setHost(Host host) {
        this.host = host;
    }

    public void rent() {
        lookHouse();
        host.rent();
        fare();
    }

    private void lookHouse(){
        System.out.println("中介带你去看房");
    }
    private void fare(){
        System.out.println("中介收取中介费用");
    }
}

测试

package com.westos.staticproxy;

public class You {
    public static void main(String[] args) {
        Host host = new Host();
        Proxy proxy = new Proxy();
        proxy.setHost(host);
        proxy.rent();
    }
}

测试结果:
在这里插入图片描述

代码实现例2

创建UserService接口,这个接口声明了与用户相关的增删改查。

package com.westos.user;

public interface UserService {

    void add();
    void delete();
    void update();
    void query();
}

创建真实对象UserServiceImpl实现UserService 接口中的方法。专注于处理核心业务。

package com.westos.user;

public class UserServiceImpl 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 com.westos.user;

public class UserServiceImplProxy implements  UserService{
    private UserServiceImpl userService;

    public void setUserService(UserServiceImpl 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 methodName){
        System.out.println("执行"+methodName+"方法");

    }

}

测试

package com.westos.user;

public class test {
    public static void main(String[] args) {
        UserServiceImpl userService = new UserServiceImpl();
        UserServiceImplProxy proxy = new UserServiceImplProxy();
        proxy.setUserService(userService);
        proxy.add();

    }
}

测试结果
在这里插入图片描述

静态代理模式的好处

  • 可以使真实角色更加纯粹,不用去关注一些公共的事情;
  • 公共的业务由代理来完成,实现业务的分工;
  • 公共业务的要扩展的话,可以更加集中和方便;

缺点:

  • 假如我们的真实角色变得非常多,代理类也会随之增多,工作量变大,开发效率变低!

然后我们想需要一种能够有静态代理的全部好处,但是又不存在这种缺点的东西。

Aop就是横向编程,可以理解一下
在这里插入图片描述

动态代理

  • 动态代理和静态代理的角色都是一样;

  • 静态代理模式的代理类是我们提前写好的,动态代理的类是动态生成的;

  • 动态代理大概分两类:

基于接口实现:JDK

基于类实现:cglib

当今用的比较多的是 JAVAssist来生成动态代理

了解动态代理之前,需要掌握两个类:

  • InvocationHandler
  • Proxy

InvocationHandler是由代理实例的调用处理程序实现的接口

invoke(Object proxy, 方法 method, Object[] args) 处理代理实例上的方法调用并返回结果。

Proxy提供了创建动态代理类和实例的静态方法,它也是由这些方法创建的所有动态代理类的超类。**

newProxyInstance(ClassLoader loader, 类<?>[] interfaces, InvocationHandler h) 返回指定接口的代理类的实例,该接口将方法调用分派给指定的调用处理程序。

代码实现

抽象角色

package com.westos.staticproxy;
//出租的接口:抽象
public interface Rent {

    //出租
    void rent();
}

真实角色

package com.westos.staticproxy;


//这个房子要出租
//真实对象
public class Host implements Rent{
    //出租
    public void rent(){
        System.out.println("房东要出租房子");
    }
}

动态代理类生成的接口对象;

package com.westos.dynamicproxy;

import com.westos.staticproxy.Rent;

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

public class InvocationHandlerProxy 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);
    }


    //proxy:代理类
    //method :代理类的调用处理程序的方法 的对象
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        lookHouse();
        Object result = method.invoke(rent, args);
        zhongJieFei();
        return null;
    }

    private void lookHouse(){
        System.out.println("中介带看房子");
    }

    private void zhongJieFei(){
        System.out.println("收中介费");
    }
}

测试

package com.westos.dynamicproxy;

import com.westos.staticproxy.Host;
import com.westos.staticproxy.Rent;

public class Test {
    public static void main(String[] args) {
        Host host = new Host();
        InvocationHandlerProxy ihp = new InvocationHandlerProxy();
        ihp.setRent(host);

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

        proxy.rent();
    }
}

测试结果:
在这里插入图片描述

动态代理的好处:

  • 可以使真实角色更加纯粹,不用去关注一些公共的事情;
  • 公共的业务由代理来完成,实现业务的分工;
  • 公共业务的要扩展的话,可以更加集中和方便;
  • 一个动态代理,一般代理一类的业务,一个动态代理可以代理多个类,代理接口;
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值