java 代理模式简述

一、什么是代理模,代理模式有哪些特征?

       1.1 、什么是代理:代理在我们身边很常见,比如帮我们 买火车票的黄牛,租房的中介,AOP 等等都是代理,我们不想做或者没时间做而不得不做的事去委托别人去做,它就是代理。

      1.2、 代理模式有哪些特征呢?


                 特征一:它有一个被代理对象(代理的目标target)、一个代理对象

                 特征二:虽然他是让别人替自己干事,但是最终的决定权在自己上手,这个理解起来有点抽象

                 特征三:代理对象持有被代理对象的引用

 

      1.3、代理模式的类别

               代理模式有两种类型:

                                            1、静态代理 (我们采用Hard Code 的方式)

                                            2、动态代理 (在运行时动态生成字节码文件去替我们干事,在Spring 中有两种:1、jdk 2、CGLib)

    二、静态代理的实现(以我们在项目中的向数据库中添加记录添加日志为例)

         

 

//用户实体类

public class UserEntity implements Serializable {
    private int id;
    private String name;
    private int age;


}

//定义用户接口
public interface UserService {

    //向数据库添加一条记录
    public void add(UserEntity userEntity);


}
//
/*
接口实现类:正常情况下,我们在这个类添加日志,

但现在我们并没有改这个类的东西,在某种程度上是符合设计原则的
*/
public class UserImpl implements  UserService {

    private UserDao userDao=new UserDao();

    public void add(UserEntity userEntity) {
     
        userDao.add(userEntity);
     
    }
}
//接下来我们的日志代理出现了

/**
 * 观察代理类的特征:
 *             1、拥有被代理对象UserImpl的引用UserService
 *             2、 this.userService.add(userEntity); 这里做日志的最终决定权在被代理对象决定
 * */
public class UserProxy implements  UserService {

    private UserService userService;

    public UserProxy(UserService userService) {
        this.userService = userService;
    }

    public void add(UserEntity userEntity) {
        System.out.println("增加前置日志");
        this.userService.add(userEntity);
        System.out.println("增加后置日志");
    }
}
//我们的测试类
public class Test {
    public static void main(String[] args) {
        UserProxy userProxy=new UserProxy(new UserImpl());
        userProxy.add(new UserEntity());
    }
}
//输出结果如下

增加前置日志
向数据库增加了一条记录
增加后置日志

 

                       

 

     三、动态地理(jdk)的实现

           


  

package com.whs.proxy.log;

import com.whs.proxy.staticed.Person;

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

public class DyProxy implements InvocationHandler {
    private UserService target;

    public Object getInstance(UserService target) throws Exception{
        this.target = target;

        Class<?> clazz = target.getClass();

        //下半截,老师深入底层来给大家讲解字节码是如何重组的
        //用来生成一个新的对象(字节码重组来实现)
        return Proxy.newProxyInstance
                (
                        clazz.getClassLoader(),
                        clazz.getInterfaces(),
                    this
                );
    }


    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        Class<?> clazz=target.getClass();
        System.out.println("增加前置日志");

        method.invoke(this.target,args);
        System.out.println("增加后置日志");
        return null;
    }
}
//测试文件
 public static void main(String[] args) {

 try
       {
           UserService obj = (UserService)new DyProxy().getInstance(new UserImpl());
           //System.out.println(obj.getClass());
           obj.add(new UserEntity());
       }catch(Exception  e)
       {
           e.printStackTrace();
       }
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值