Java 8为什么使用动态代理?

在Java开发中,常常用到动态动态代理,这是Java 1.3版本就已经有的特性,所以应用非常广泛,特别是J2EE的远程调用。今天小编就和大家分享,为什么要使用动态代理及其使用技巧,希望对Java初学者有所帮助。

在J2EE的远程调用中,给定一个抽象接口以及这个接口的具体实现,就可以通过创建两个额外的类来实现这个接口的远程调用了(如,跨JVM)。首先,在源JVM上实现相应的接口,并将调用细节序列化后通过网络传输。然后,在目标JVM上,获取到序列化后的调用的细节,并分配给具体的的类去调用。

如果没有动态代理和反射的话,Java开发者就必须为每个远程接口都提供两个类。而动态代理是运行时产生的类,实现一个或多个接口,接口中每个方法的调用都会自动转换为 java.runtime.InvocationHandler 提供的方法调用:

public interface InvocationHandler {

Object invoke(Object proxy, Method method, Object[] args) throws Throwable;

}

InvocationHandler决定如何处理调用,如何在运行时使用方法的有效信息,包括注解、参数类型及方法的返回类型。这样就可以实现一个通用逻辑来定义方法调用的分发。一旦写好了一个InvocationHandler,就可以调用代理类的 handler 来完成所有接口中的方法,而不是为每一个接口写一个单独的实现。

虽然远程调用最近几年里已经没那么受欢迎了,因为开发者需要明白方法调用分发与网络请求发送在语义和失败模式上的本质区别,但是动态代理仍保留在Java语言中。下面我们就来看看动态代理在其他方面的应用。

魔法匹配器

为了能写出简洁的流式测试,可以使用一个“Magic”对象,定义一个“magic”接口,然后通过一个动态代理来实现目标行为。比较特别的是,在测试时候用”magic builders”来生成测试值,然后用“magic matchers”来表述断言属性测试的结果。我们这里只关注匹配器。

我们有一个Person支撑类,这是一个典型的bean——成员变量是私有的,通过getter和setter方法暴露。

public class Person {

private String name;

private int age;

// insert getters and setters here

}

使用一个简单Hamcrest类,可以使用两种方式来断言该类的实例。一种方法是单独抽取每个值,分开断言。

assertThat(person.getName(), containsString("Smith"));

assertThat(person.getAge(), greaterThan(30));

另一种方式是使用allOf和hasProperty方法,将对象作为一个整体,通过一组期望值来匹配。

assertThat(person, allOf(

hasProperty("name", containsString("Smith")),

hasProperty("age", greaterThan(30)));

这样能很好的工作,但是这种方式对 Hamcrest 描述整体匹配和错误匹配并没有什么帮助。

Expected: (hasProperty("name", a string containing "Putey") and hasProperty("age", a value greater than <43>))

but: hasProperty("age", a value greater than <43>) property ‘age‘ <42> was less than <43>

hasProperty的匹配在类型一致性的检测也是非常弱的:我们可以写成 hasProperty(“age”, containsString(“Smith”)),这样类型检测也不会拒绝。

我们真正想要的是一个流式API,能够像下面一样使用:

assertThat(person, aPerson()

.withName("Arthur Putey")

.withAge(greaterThan(43)));

并且能够很好且易于理解地报告错误的匹配:

Expected:

name: a string containing "Putey"

age: a value greater than <43>

but:

age: <42> was less than <43>

很容易写一个上述功能的自定义匹配器,但是不得不很乏味地写很多次。幸运的是,可以通过动态代理来帮我们解决。首先,我们定义一个流式接口,该接口包含如下方法:

interface PersonMatcher extends Matcher<Person> {

PersonMatcher withName(String expected);

PersonMatcher withName(Matcher<? super String> matching);

PersonMatcher withAge(int expected);

PersonMatcher withAge(Matcher<Integer> matching);

}

然后,使用在一个名为 MagicMatcher 的类上的静态方法来获取动态代理,该代理实现了这个接口,然后通过方法调用来获取调节表达式:

static PersonMatcher aPerson() {

return MagicMatcher.proxying(PersonMatcher.class);

}

每个方法的调用都通过代理类的“interpreted”方法来实现,该代理从方法(“withAge”)中获取属性(“age”),并指定调用匹配对象上的(“getAge”)方法来获取属性值。属性的名称以及匹配中对应的值将会被存储,直到代理类的 match 或 describeMismatch 方法被调用(这就是为什么接口需要继承 Matcher)。在调用的时候需要抽取并测试对象的属性,如果有必要,会创建错误匹配报告。

这种方式是轻量级的,我们可以引入任何新的自定义的接口,并在测试中重用,这样,是非常有利于编写自定义Hamcrest匹配器的,因为不再需要编写接口的实现。所有需要生成的在接口中定义的匹配器行为,都只需要实现一次,我们通过一个合适的 InvocationHandler 来完成逻辑功能的实现。

以上就是Java8中动态代理相关应用,可能介绍得还不够全面,但文章篇幅有限,在后面的文章中,小编将继续为大家分享。

相关阅读:《Java常量使用中如何避免反模式?

时间: 2024-10-07 06:17:01

Java 8为什么使用动态代理?的相关文章

Java语言中反射动态代理接口的解释与演示

Java语言中反射动态代理接口的解释与演示 Java在JDK1.3的时候引入了动态代理机制.可以运用在框架编程与平台编程时候捕获事件.审核数据.日志等功能实现,首先看一下设计模式的UML图解: 当你调用一个接口API时候,实际实现类继承该接口,调用时候经过proxy实现. 在Java中动态代理实现的两个关键接口类与class类分别如下: java.lang.reflect.Proxy java.lang.reflect.InvocationHandler 我们下面就通过InvocationHan

java反射机制与动态代理

在学习HadoopRPC时,用到了函数调用,函数调用都是采用的java的反射机制和动态代理来实现的,所以现在回顾下java的反射和动态代理的相关知识. 一.反射 JAVA反射机制定义: JAVA反射机制是java程序在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法:这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制. 反射就是把Java类中的各种成分映射成相应的Java类. Java反射机制主要提供了以下功能: 1

学习Spring必学的Java基础知识(2)----动态代理

学习Spring必学的Java基础知识(2)----动态代理 引述要学习Spring框架的技术内幕,必须事先掌握一些基本的Java知识,正所谓“登高必自卑,涉远必自迩”.以下几项Java知识和Spring框架息息相关,不可不学(我将通过一个系列分别介绍这些Java基础知识,希望对大家有所帮助.): [1] Java反射知识-->Spring IoC :http://www.iteye.com/topic/1123081 [2] Java动态代理-->Spring AOP :http://www

Java中两种动态代理的实现

本文介绍了java中两种动态代理的实现方法,Spring的动态代理也是基于这两种方法的.直接附上源码: 1.JDK实现 使用JDK实现动态代理必须使用接口 接口Work.java public interface Work { public void work(); } 实现类WorkImpl.java public class WorkImpl implements Work { @Override public void work() { System.out.println("我在工作&q

简谈Java 反射机制,动态代理

谈谈 Java 反射机制,动态代理是基于什么原理?小编整理了一些java进阶学习资料和面试题,需要资料的请加JAVA高阶学习Q群:701136382 这是小编创建的java高阶学习交流群,加群一起交流学习深造.群里也有小编整理的2019年最新最全的java高阶学习资料! 反射机制 Java 语言提供的一种基础功能,赋予程序在运行时自省(introspect,官方用语)的能力.可以在运行时通过提供完整的"包名+类名.class"得到某个对象的类型. 功能 在运行时能判断任意一个对象所属的

Java进阶之 JDK动态代理与Cglib动态代理

一.动态代理概述: 与静态代理对照(关于静态代理的介绍 可以阅读上一篇:JAVA设计模式之 代理模式[Proxy Pattern]), 动态代理类的字节码是在程序运行时由Java反射机制动态生成. 注意: 1.AspectJ是采用编译时生成AOP代理类,具有更好的性能,但是需要使用特定的编译器进行处理 2.Spring AOP采用运行时生成AOP代理类,无需使用特定编译器进行处理,但是性能相对于AspectJ较差 二.JDK动态代理 [对有实现接口的对象做代理] 1.JDK动态代理中 需要了解的

Java重要技术(27)动态代理之查看代理对象的类型信息

1.1. 查看动态代理类的各种信息 通过反射机制可以获取Proxy创建的动态代理类的各种信息,包括实现的接口,构造函数,方法等.查看到这些信息后,可以更好的理解为什么动态代理这么使用. /** * @Title: ProxyTest.java * @Package com.test.javatechnology.proxy * @Description: * @author http://www.cnblogs.com/coe2coe/ * @date 2017年3月26日 下午5:36:28

JAVA学习--反射之动态代理模式

1 import java.lang.reflect.InvocationHandler; 2 import java.lang.reflect.Method; 3 import java.lang.reflect.Proxy; 4 5 //动态代理的使用,体会反射是动态语言的关键 6 interface Subject { 7 void action(); 8 } 9 10 // 被代理类 11 class RealSubject implements Subject { 12 public

Java核心技术点之动态代理

本篇博文会从代理的概念出发,介绍Java中动态代理技术的使用,并进一步探索它的实现原理.由于个人水平有限,叙述中难免出现不清晰或是不准确的地方,希望大家可以指正,谢谢大家:) 一.概述 1. 什么是代理 我们大家都知道微商代理,简单地说就是代替厂家卖商品,厂家“委托”代理为其销售商品.关于微商代理,首先我们从他们那里买东西时通常不知道背后的厂家究竟是谁,也就是说,“委托者”对我们来说是不可见的:其次,微商代理主要以朋友圈的人为目标客户,这就相当于为厂家做了一次对客户群体的“过滤”.我们把微商代理