精通Java设计模式从初见到相爱之代理设计模式(2)

本文深入浅出地介绍了代理设计模式的概念及其应用场景,并详细探讨了三种代理类型:静态代理、JDK动态代理和CGLIB动态代理。通过具体的代码示例,帮助读者理解不同代理模式的特点及其实现方式。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

序言:UML图我就不画了,毕竟大部分都不是软件出身的,每一篇博客争取都用大白话!

1、代理设计模式的概念:

    它是个什么玩意?和上一篇讲述的单例设计模式一样,也是一种解决程序的高级解决方案。

    举例:我要灭了传说中的山口组,我通知青龙帮,然后青龙帮 帮我灭了,最终达到了我要灭山口组的目的,我亲自灭和别人帮我灭结果都一样的,所以,青龙帮就相当于代替了我,这种方式叫代理模式。

2、涉及到的技术点(反射)

    了解下反射,可能只讲代理设计模式的话可能你不需要掌握,但是代理设计模式的应用场景会讲到,还是希望你能够了解下反射知识点,我就不讲了,之后我会发表一篇关于反射的博客

3、代理设计模式的种类

    3.1 静态代理

        3.1.1 概念:目标对象、和代理对象统一实现一样的接口 缺点,代理类太多的情况,一旦增加接口方法,那么目标对象与代理对象都要重新修改和维护。

        3.1.2 代码讲解 类:接口IUserDao、实现接口的两个类,目标类和代理类 GirlUser和ProxyGirl,最后main函数执行(java9版本以上)        

package proxy.jingtaiProxy;

public interface IUserDao {

    public static final String girlName ="七七";

    void miss(String boyName);

    static void love(){
        System.out.println("恋爱");
    };

    default void noLove(){
        System.out.println("高冷,不谈恋爱");
    }

    private static void securt(){
        System.out.println("每个女孩子内心都有一个小秘密");
    }
}
package proxy.jingtaiProxy;

public class GirlUser implements IUserDao{

    @Override
    public void miss(String boyName) {
        System.out.println("我在思念岳飞将军");
    }

    @Override
    public void noLove() {
        System.out.println("我跟谁也不谈恋爱,你们男的一个比一个丑");
    }
}
package proxy.jingtaiProxy;

public class ProxyGirl implements IUserDao{

    private IUserDao iUserDao;

    public ProxyGirl(IUserDao iUserDao){
        this.iUserDao = iUserDao;
    }

    @Override
    public void miss(String boyName) {
        System.out.println("我告诉你们男生他在思念几百年前的岳飞大将军");
    }

    @Override
    public void noLove() {
        System.out.println("她不会和你们恋爱的,你们死心吧");
    }
}
package proxy.jingtaiProxy;

public class Main {

    public static void main(String[] args){
        //可以做到在不修改目标对象的功能前提下,对目标功能扩展.
        //因为代理对象需要与目标对象实现一样的接口,所以会有很多代理类,类太多.同时,一旦接口增加方法,目标对象与代理对象都要维护.
        GirlUser girlUser   = new GirlUser();
        ProxyGirl proxyGirl = new ProxyGirl(girlUser);
        proxyGirl.noLove();

    }
}

运行结果:

144522_VGNv_3209213.png

怕你们看不清,手打一下,运行结果:她不会和你们恋爱的,你们死心吧

    3.2 jdk动态代理

    3.2.1 概念:jdk动态代理是代理对象类不需要实现接口!但是目标对象一定要实现接口!

    3.2.2 代码实现 用到的类  IUserDao GirlUser DongtaiProxy,由于3.1.2已经写了两个类了,我就最后在把DongtaiProxy 和重写的main函数代码贴一下

package proxy.dongtaiProxy;

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

public class DongtaiProxy {

    //传入的是目标对象
    private Object object;

    public DongtaiProxy(Object object){
        this.object = object;
    }

    public Object getProxyInstance(){
        return Proxy.newProxyInstance(object.getClass().getClassLoader(),
                object.getClass().getInterfaces(),
                new InvocationHandler() {
                    @Override
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                        return method.invoke(object,args);
                    }
                });
    }
}
package proxy.dongtaiProxy;

import proxy.jingtaiProxy.GirlUser;
import proxy.jingtaiProxy.IUserDao;

public class Main {

    public static void main(String[] args){
        //代理对象不需要实现接口,但是目标对象一定要实现接口,否则不能用动态代理
        IUserDao iUserDao = new GirlUser();
        IUserDao proxy    = (IUserDao) new DongtaiProxy(iUserDao).getProxyInstance();
        proxy.noLove();
    }
}

运行结果如图:

145657_aVUj_3209213.png

    3.3 cglib动态代理

    3.3.1 概念:它不需要目标对象类实现接口了!就用目标对象自己的方法,需要用到jar包,maven依赖如下:

<dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-core</artifactId>
    <version>5.0.4.RELEASE</version>
</dependency>

    3.3.2 代码实现  用到的类:Girl、CglibProxy、main函数,重新贴一下这个cglib代理的代码    

package proxy.CglibProxy;

public class Girl {
    public void love(){
        System.out.println("爱家人爱朋友爱和平!");
    }
}
package proxy.CglibProxy;

import org.springframework.cglib.proxy.Enhancer;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;

import java.lang.reflect.Method;

public class CglibProxy {

    private Object target;

    public CglibProxy(Object target){
        this.target = target;
    }

    public Object getProxyInstance(){

        //org.springframework.cglib.proxy.Enhancer cglib中的一个工具类
        Enhancer enhancer = new Enhancer();

        enhancer.setSuperclass(target.getClass());

        enhancer.setCallback(new MethodInterceptor() {
            public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {

                Object result = method.invoke(target,objects);
                return result;
            }
        });

        //创建子类对象
        return enhancer.create();

    }
}
package proxy.CglibProxy;

public class main {


    public static void main(String[]  args){

        Girl girl        = new Girl();

        Girl proxy = (Girl) new CglibProxy(girl).getProxyInstance();

        proxy.love();

    }

}

运行结果:

145506_cH8l_3209213.png

    
4、代理设计模式在项目,工作中的应用场景

    最典型的就是Spring AOP,它就是根据代理模式实现的,当然还有我上面讲的反射。

    4.1 先说反射,只是大概说下,反射在我们工作中应用的还是比较多的,比如我们的eclipse、idea,当我们输入一个对象的时候,只要一点就会出现它相应的方法和属性,还有它能够开发各种通用的框架,struts2举例

150447_0x05_3209213.png

我们前端请求的是ModifyLoginPasswordAction,那么我们就会解析这个struts.xml文件,然后找到方法是sms4LoginPwd的action,然后创建这个action的实例,去调用execute()执行后台代码,就讲到了这里,主要说设计模式。

    4.2 spring aop 用到的代理设计模式

    它主要用到jdk动态代理设计以及cglib动态代理设计,spring aop怎么用到的呢,它会判断你的目标对象类有木有实现接口,然后做一个判断,如图:

    151043_DxCO_3209213.png

    在说下aop应用在那些场景呢?事物管理,日志,缓存等等.......

    说下事物配置?我们都是在xml文件配置相关属性,如图:

    151224_DQfP_3209213.png

然后看到我红色标出来的那个类了吗,进去看一下!

151354_g5Z5_3209213.png

这个类里面封装了很多控制事物的方法,commit(),begin()...等等,因为文件太大我只截图一部分,所以我们在项目开发中就在事物方面一直用这些方法,但是绝对不会创建这个类的实例,因为这是spring核心内部的类,怎么可能是我们随随便便拿出来new的,所以我们只能通过代理模式用这个类的事物控制方法,同理好多类我们也如此!

    希望大家记住代理设计模式在我看来就是,框架里面的部分核心类,官方的部分类,咱就不直接new,用另外一种方式用他们的方法,就是代理!为什么不直接 new,看下面

      在实际的项目中,在你通往架构师的道路上,你要培养出一种感觉:要new一个实体对象是件很谨慎的事情(不是指值对象),不要随便new。最好不要自己new,让别人去new,传给你去调用。这样new错了也是别人的事,换而言之你的模块是好质量的,禁得起推敲的。那么都不愿意去new,谁去new?你可以通过代理模式获取该类实例,获取对应的方法,当然还有另外一种代替new的,那就是用抽象工厂模式,下一节讲解!

 

 


    

    

转载于:https://my.oschina.net/mdxlcj/blog/1786848

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值