服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - Java教程 - 代理模式:JAVA静态代理和动态代理的实例和实现详解

代理模式:JAVA静态代理和动态代理的实例和实现详解

2021-12-01 13:38长臂人猿 Java教程

这篇文章主要给大家介绍了关于Java静态代理和动态代理的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

前言

代理模式,我们这里结合JAVA的静态代理和动态代理来说明,类比Spring AOP面向切面编程:增强消息,也是代理模式。

而我们的静态代理和动态代理,与(service)接口和(serviceImpl)接口实现类有什么区别呢?静态代理的概念与其在理解上类似,可以说静态代理是实现类的增强消息。而且,静态代理针对的是所以实现接口的实现类(通过向上转型实现)。

静态代理

实现简述

本质上是一接口一代理,对该接口的所有实现类进行"增强"(额外)操作:

如下例:human接口有两个实现类:man和woman,我们需要在每个human接口被实现的时候说他是god创造的(增强操作)。那么我们针对human接口手动实现一个GodProxy静态代理类。那么我们通过这个代理执行human接口的实现对象的方法时就可以引入增强操作。

创建human接口

?
1
2
3
public interface Human {
    public void sex();
}

创建接口实现类

?
1
2
3
4
5
6
7
8
9
10
11
12
public class Man implements Human{
    @Override
    public void sex() {
        System.out.println( "this is Man" );
    }
}
public class Women implements Human{
    @Override
    public void sex() {
        System.out.println( "this is Women" );
    }
}

创建针对接口实现增强操作的代理

?
1
2
3
4
5
6
7
8
9
10
11
12
public class GodProxy implements Human{
    Human huamnGenarator;
    public GodProxy(Human huamnGenarator){
        this.huamnGenarator = huamnGenarator;
    }
    @Override
    public void sex() {
        System.out.println( "God begin to make human" );
        huamnGenarator.sex();
        System.out.println(" End of work ");
    }
}

代理实现效果

?
1
2
3
4
5
6
7
8
9
public class 静态代理 {
    public static void main(String[] args) {
        GodProxy proxy_1 = new GodProxy(new Man());
        GodProxy proxy_2 = new GodProxy(new Women());
        proxy_1.sex();
        System.out.println("\n\n");
        proxy_2.sex();
    }
}

代理模式:JAVA静态代理和动态代理的实例和实现详解

动态代理

实现简述

相对与静态代理,一个代理类proxy只能针对一个接口实现操作,众所周知类可以实现多个接口,那能不能对所有的实现类的接口实现增强操作呢?可以的,我们需要通过反射的方式,通过向上转型使得传入参数为Object对象,但是Object里面的getClass方法获取Class对象,实际上是实现类的Class(由Java的向上转型特性可知)。 获取实现类对象的class对象且获得其实现的接口数组。我们可以达到这个目的:这就是动态代理:对多个接口实现静态代理。

动态代理类主要实现InvocationHandler这个接口,接口实现invoke方法来实现增强操作。并通过一个自定义的方法来创建和绑定动态代理类和入参(向上转型为obj的实现类对象),从而实现实现类对象方法的增强操作。

即我们通过动态代理传入实例类对象,在自己写的newProxyInstance方法中通过Proxy类的newProxyInstance方法代理生成一个新的实例类对象,这个新的对象不仅包含所有的入参实力类对象信息,且在通过代理类生成新的实例类对象过程中注入了invoke方法(我们实现InvocationHandler接口的核心方法)的逻辑。

 

这里我们有两个疑问:

1、如何实现增强的invoke方法;——通过java.lang.reflect.Proxy.newProxyInstance方法中复原向上转型的obj对象为原对象(具体实现类对象),并绑定增强的invoke方法。

2、代理对象如何复现对象方法; ——利用向上转型的复原不变性。

要点:向上转型

向上转型:子类实例赋值给父类引用。 无法调用子类拓展方法,但是她的实现确确实实存在:

?
1
2
3
4
5
Object obj = new Girl();
Girl girl = (Girl) obj;
System.err.println(obj.getClass().getName());
System.err.println(obj.getClass().getInterfaces().length);
girl.nickName();

输出:

src.代理模式.Girl
2

少女

创建YoungMan接口

这里还将用到Human接口,两个接口。

?
1
2
3
public interface YoungMan {
    public void nickName();
}

创建两个接口实现类

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
public class Boy implements Human,YoungMan{
    @Override
    public void sex() {
        System.out.println( "this is Man" );
    }
    @Override
    public void nickName() {
        System.out.println( "少年" );
    }
}
 
public class Girl implements Human,YoungMan{
    @Override
    public void sex() {
        System.out.println( "this is Women" );
    }
    @Override
    public void nickName() {
        System.out.println( "少女" );
    }
}

创建动态代理实例对象

这里我们主要通过Proxy.newProxyInstance方法创建一个代理类,传参:

?
1
2
3
public static Object newProxyInstance(ClassLoader loader,
                                         Class<?>[] interfaces,
                                         InvocationHandler h)

类加载器:指定代理类由哪个classloader加载;代理类需要代理实现的接口方法;InvocationHandler对象:表示的是当动态代理对象调用方法的时候实际执行的会是该InvocationHandler对象上的invoke方法(即增强方法)。在invoke方法中通过反射方法名称去执行实际要执行的方法和增强操作。(实例说明可以看文末补充)

注意返回的这里自定义的newProxyInstance是Object。

invoke方法中的args为方法传入的参数们;

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public class GodForYoungProxy implements InvocationHandler {
    private Object godForYoungProxy;
    //参数为Object设计一个向上转型
    public Object newProxyInstance(Object godForYoungProxy) {
        this.godForYoungProxy = godForYoungProxy;
        //this指的是GodForYoungProxy这个InvocationHandler实现类
        System.err.println( godForYoungProxy.getClass().getName() );
        return Proxy.newProxyInstance(godForYoungProxy.getClass().getClassLoader(), godForYoungProxy.getClass().getInterfaces(), this);
    }
 
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println(" God for Young begin to work ... ");
        proxy = method.invoke(godForYoungProxy, args);
        System.out.println(" End of work ");
        return proxy;
    }
}

代理实现效果

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
public class 动态代理 {
    public static void main(String[] args) {
        GodForYoungProxy godForYoungProxy = new GodForYoungProxy();
        Human human = (Human) godForYoungProxy.newProxyInstance(new Boy());
        YoungMan youngMan = (YoungMan) godForYoungProxy.newProxyInstance(new Boy());
        human.sex();
        youngMan.nickName();
        //向上转型测试
//        Object obj = new Girl();
//        Girl girl = (Girl) obj;
//        System.err.println(obj.getClass().getName());
//        System.err.println(obj.getClass().getInterfaces().length);
//        girl.nickName();
    }
}

代理模式:JAVA静态代理和动态代理的实例和实现详解

代理生成对象中的obj向上转型对象的.getClass().getName()打印:

代理模式:JAVA静态代理和动态代理的实例和实现详解

要点:InvocationHandler补充

结合创建动态代理实例对象目录内容补充说明如下:官网的Proxy.newProxyInstance中的入参说明

loader – the class loader to define the proxy class
interfaces – the list of interfaces for the proxy class to implement
h – the invocation handler to dispatch method invocations to =>将方法调用分派到的调用处理程序

同样的上述代码实现效果,我们将实现的GodForYoungProxy implements InvocationHandler中的invoke方法注释一行:

?
1
2
3
4
5
6
7
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    System.out.println(" God for Young begin to work ... ");
   // proxy = method.invoke(godForYoungProxy, args);
    System.out.println(" End of work ");
    return proxy;
}

明显的,我们发现少了实际的sex和nickName方法的输出内容,因为没有实现对应调用方法:

代理模式:JAVA静态代理和动态代理的实例和实现详解

通过这样的特性,其实我们在invoke方法里面拥有很强的操作性,比如说让指定方法执行,对不同方法执行不同策略等。

代理模式和修饰模式的区别

代理模式和装饰者模式很相似,但是他们的区别在于:

1、代理模式是在类编译的时候,增强方法就已经确定的,有些动态代理不支持多层嵌套;装饰者则可以不断递归被构造装饰;

2、代理模式:强调对对象的访问控制,方法的使用都是通过反射实现;代理模式则是强调功能的增加,相当于在原基础的被修饰者上不断套娃。

总结

本篇文章就到这里了,希望能给你带来帮助,也希望您能够多多关注服务器之家的更多内容!

原文链接:https://blog.csdn.net/qq_37334150/article/details/119851562

延伸 · 阅读

精彩推荐
  • Java教程Java BufferWriter写文件写不进去或缺失数据的解决

    Java BufferWriter写文件写不进去或缺失数据的解决

    这篇文章主要介绍了Java BufferWriter写文件写不进去或缺失数据的解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望...

    spcoder14552021-10-18
  • Java教程Java使用SAX解析xml的示例

    Java使用SAX解析xml的示例

    这篇文章主要介绍了Java使用SAX解析xml的示例,帮助大家更好的理解和学习使用Java,感兴趣的朋友可以了解下...

    大行者10067412021-08-30
  • Java教程20个非常实用的Java程序代码片段

    20个非常实用的Java程序代码片段

    这篇文章主要为大家分享了20个非常实用的Java程序片段,对java开发项目有所帮助,感兴趣的小伙伴们可以参考一下 ...

    lijiao5352020-04-06
  • Java教程xml与Java对象的转换详解

    xml与Java对象的转换详解

    这篇文章主要介绍了xml与Java对象的转换详解的相关资料,需要的朋友可以参考下...

    Java教程网2942020-09-17
  • Java教程小米推送Java代码

    小米推送Java代码

    今天小编就为大家分享一篇关于小米推送Java代码,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧...

    富贵稳中求8032021-07-12
  • Java教程升级IDEA后Lombok不能使用的解决方法

    升级IDEA后Lombok不能使用的解决方法

    最近看到提示IDEA提示升级,寻思已经有好久没有升过级了。升级完毕重启之后,突然发现好多错误,本文就来介绍一下如何解决,感兴趣的可以了解一下...

    程序猿DD9332021-10-08
  • Java教程Java8中Stream使用的一个注意事项

    Java8中Stream使用的一个注意事项

    最近在工作中发现了对于集合操作转换的神器,java8新特性 stream,但在使用中遇到了一个非常重要的注意点,所以这篇文章主要给大家介绍了关于Java8中S...

    阿杜7482021-02-04
  • Java教程Java实现抢红包功能

    Java实现抢红包功能

    这篇文章主要为大家详细介绍了Java实现抢红包功能,采用多线程模拟多人同时抢红包,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙...

    littleschemer13532021-05-16