`

Java动态代理详解

 
阅读更多

不定期整理硬盘内源代码、笔记、总结等,同时发上来分享一下。今天再发一篇关于Java动态代理的总结(貌似ItEye一天最多发5篇Blog,再多只能放草稿箱了?)

-----------------------------------------------------------

Java动态代理详解

说到动态代理,顾名思义就是动态的代理(真是废话)。

关于代理:想必大家都并不陌生,GOF的23种设计模式之一(结构型模式)。这里暂不多做介绍,有兴趣的可以关注我关于设计模式的文章。

什么是动态代理:

说起动态,其实不如先说什么是静态。所谓静态代理,个人理解为自己手写的代理类,或者用工具生成的代理类,或者别人帮你写的代理类(没说一样...)。总之,就是程序运行前就已经存在的编译好的代理类。

相反,如果代理类程序运行前并不存在,需要在程序运行时动态生成(无需手工编写代理类源码),那就是今天要说的动态代理了。

如何生成的:根据Java的反射机制动态生成。

不多说了,上程序。

目标接口TargetInterface:

public interface TargetInterface {
	public int targetMethodA(int number);
	public int targetMethodB(int number);
}

很简单,一个普通的接口,里面有若干方法(此处写2个示范一下)

实现该接口的委托类ConcreteClass:

public class ConcreteClass implements TargetInterface{

	public int targetMethodA(int number) {
		System.out.println("开始调用目标类的方法targetMethodA...");
		System.out.println("操作-打印数字:"+number);
		System.out.println("结束调用目标类的方法targetMethodA...");
		return number;
	}
	
	public int targetMethodB(int number){
		System.out.println("开始调用目标类的方法targetMethodB...");
		System.out.println("操作-打印数字:"+number);
		System.out.println("结束调用目标类的方法targetMethodB...");
		return number;
	}

}

很简单,一个普通的类,实现了目标接口。

代理处理器类ProxyHandler:

public class ProxyHandler implements InvocationHandler{
	private Object concreteClass;
	
	public ProxyHandler(Object concreteClass){
		this.concreteClass=concreteClass;
	}

	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		System.out.println("proxy:"+proxy.getClass().getName());
		System.out.println("method:"+method.getName());
		System.out.println("args:"+args[0].getClass().getName());
		
		System.out.println("Before invoke method...");
		Object object=method.invoke(concreteClass, args);//普通的Java反射代码,通过反射执行某个类的某方法
		//System.out.println(((ConcreteClass)concreteClass).targetMethod(10)+(Integer)args[0]);
		System.out.println("After invoke method...");
		return object;
	}

}

该类实现了Java反射包中的InvocationHandler接口。代理实例调用方法时,将对方法调用指派到它的代理处理器程序的invoke方法中。invoke方法内部实现预处理,对委托类方法调用,事后处理等逻辑。

最后是入口程序:

public class DynamicProxyExample {
	public static void main(String[] args){
		 ConcreteClass c=new ConcreteClass();//元对象(被代理对象)
		 InvocationHandler ih=new ProxyHandler(c);//代理实例的调用处理程序。
		 //创建一个实现业务接口的代理类,用于访问业务类(见代理模式)。
		 //返回一个指定接口的代理类实例,该接口可以将方法调用指派到指定的调用处理程序,如ProxyHandler。
		 TargetInterface targetInterface=
			 (TargetInterface)Proxy.newProxyInstance(c.getClass().getClassLoader(),c.getClass().getInterfaces(),ih);
		 //调用代理类方法,Java执行InvocationHandler接口的方法.
		 int i=targetInterface.targetMethodA(5);
		 System.out.println(i);
		 System.out.println();
		 int j=targetInterface.targetMethodB(15);
		 System.out.println(j);
	}
}

首先创建委托类对象,将其以构造函数传入代理处理器,代理处理器ProxyHandler中会以Java反射方式调用该委托类对应的方法。然后使用Java反射机制中的Proxy.newProxyInstance方式创建一个代理类实例,创建该实例需要指定该实例的类加载器,需要实现的接口(即目标接口),以及处理代理实例接口调用的处理器。

最后,调用代理类目标接口方法时,会自动将其转发到代理处理器中的invoke方法内,invoke方法内部实现预处理,对委托类方法调用,事后处理等逻辑。

 

使用Java动态代理机制的好处:

1、减少编程的工作量:假如需要实现多种代理处理逻辑,只要写多个代理处理器就可以了,无需每种方式都写一个代理类。

2、系统扩展性和维护性增强,程序修改起来也方便多了(一般只要改代理处理器类就行了)。

 

使用Java动态代理机制的限制:

目前根据GOF的代理模式,代理类和委托类需要都实现同一个接口。也就是说只有实现了某个接口的类可以使用Java动态代理机制。但是,事实上使用中并不是遇到的所有类都会给你实现一个接口。因此,对于没有实现接口的类,目前无法使用该机制。有人说这不是废话吗,本来Proxy模式定义的就是委托类要实现接口的啊!但是没有实现接口的类,该如何实现动态代理呢?

当然不是没有办法,这也是我后面抽时间要继续整理和总结原先使用过的一件神器,相关Blog会不定期发上来。那就是大名鼎鼎的CGLib...

PS:CGLib中的动态代理已经新鲜出炉,欢迎访问:http://shensy.iteye.com/blog/1873155 

 

分享到:
评论
4 楼 qq_15010155 2016-05-04  
OLDDAOG 写道
代理类和委托类都要实现同一接口是什么意思啊上面的代码不就是只有委托类实现了TargetInterface接口吗?

下面创建代理类实例的时候要传入目标接口
3 楼 OLDDAOG 2016-04-28  
代理类和委托类都要实现同一接口是什么意思啊上面的代码不就是只有委托类实现了TargetInterface接口吗?
2 楼 longhua2003 2015-09-21  
1 楼 小桔子 2015-01-20  
aop都要使用CGLib动态代理么、。?我启用<aop:aspectj-autoproxy/> 就404.。原始框架的
struts的配置。。。。
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE struts PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 2.3//EN" "http://struts.apache.org/dtds/struts-2.3.dtd">
<struts>

<!-- 指定由spring负责action对象的创建 -->
<constant name="struts.objectFactory" value="spring" />
<!-- 所有匹配*.do的请求都由struts2处理 -->
<constant name="struts.action.extension" value="do" />
<!-- 是否启用开发模式 -->
<constant name="struts.devMode" value="false" />
<!-- struts配置文件改动后,是否重新加载 -->
<constant name="struts.configuration.xml.reload" value="true" />
<!-- 设置浏览器是否缓存静态内容 -->
<constant name="struts.serve.static.browserCache" value="false" />
<!-- 请求参数的编码方式 -->
<constant name="struts.i18n.encoding" value="utf-8" />
<!-- 每次HTTP请求系统都重新加载资源文件,有助于开发 -->
<constant name="struts.i18n.reload" value="true" />
<!-- 文件上传最大值 -->
<constant name="struts.multipart.maxSize" value="104857600" />
<!-- 让struts2支持动态方法调用 -->
<constant name="struts.enable.DynamicMethodInvocation" value="true" />
<!-- Action名称中是否还是用斜线 -->
<constant name="struts.enable.SlashesInActionNames" value="false" />
<!-- 允许标签中使用表达式语法 -->
<constant name="struts.tag.altSyntax" value="true" />
<!-- 对于WebLogic,Orion,OC4J此属性应该设置成true -->
<constant name="struts.dispatcher.parametersWorkaround" value="false" />
    <constant name="struts.multipart.saveDir" value="/tmp"></constant>
   
<package name="basePackage" extends="struts-default">
<interceptors>
<!-- session拦截器 -->
<interceptor name="sessionInterceptor" class="com.szcatic.interceptor.SessionInterceptor" />
<interceptor-stack name="sessionStack">
<interceptor-ref name="defaultStack"></interceptor-ref>
<interceptor-ref name="sessionInterceptor">
<param name="excludeMethods">doNotNeedSession_*</param>
</interceptor-ref>
</interceptor-stack>
<!-- 权限拦截器 -->
<interceptor name="authInterceptor" class="com.szcatic.interceptor.AuthInterceptor" />
<interceptor-stack name="authStack">
<interceptor-ref name="sessionStack"></interceptor-ref>
<interceptor-ref name="authInterceptor">
<param name="excludeMethods">doNotNeedSession_*,doNotNeedAuth_*</param>
</interceptor-ref>
</interceptor-stack>

</interceptors>
<!-- 全局拦截器栈 -->
<default-interceptor-ref name="authStack"></default-interceptor-ref>

<global-results>
<result name="noAuth">/error/noAuth.jsp</result>
<result name="noSession">/error/noSession.jsp</result>
<result name="strutsException">/error/strutsException.jsp</result>
</global-results>

<global-exception-mappings>
<exception-mapping result="strutsException" exception="java.lang.Exception"></exception-mapping>
</global-exception-mappings>
  
</package>
</struts>

相关推荐

    java动态代理详解

    更好的了解java动态代理的原理!

    Java 动态代理详解(代理模式+静态代理+JDK动态代理+CGLIB动态代理)

    Java 动态代理详解(代理模式+静态代理+JDK动态代理+CGLIB动态代理) Java 动态代理是 Java 编程语言中的一种强大工具,广泛应用于 Spring AOP、Hibernate 数据查询、测试框架的后端 mock、RPC 远程调用、Java 注解...

    Java动态代理详解; CGLib实现动态代理1

    Java 动态代理详解与 CGLib 实现 在 Java 中,代理模式是一种设计模式,它允许我们在不修改原有对象的基础上,为对象增加额外的功能。代理模式的核心在于代理类(Proxy)作为真实对象(RealSubject)的替代品,提供...

    Java 动态代理详解(学习资料)

    JDK 动态代理JDK 提供了 java.lang.reflect.Proxy 类和 java.lang.reflect.InvocationHandler 接口来支持动态代理。Proxy 类用于创建一个代理对象,而 InvocationHandler 接口则定义了代理对象的方法调用处理逻辑。...

    Java代理模式Java动态代理

    ### Java代理模式与Java动态代理详解 #### 一、代理模式概述 代理模式是一种软件设计模式,它在客户端和目标对象之间提供了一种间接层。这种模式的主要目的是控制客户端对目标对象的访问,并且可以在不修改原有...

    java动态代理实现详解

    Java 动态代理是Java平台提供的一种强大的工具,它允许我们在运行时动态生成代理类,这些代理类可以实现一组指定的接口,同时还能在方法调用前后添加自定义的行为。这种机制大大简化了面向切面编程(AOP)和事件监听...

    Java动态代理机制详解[整理].pdf

    Java动态代理机制是一种在运行时创建代理类和代理对象的技术,它允许我们在不修改源代码的情况下,对已有的接口或类进行扩展,增加新的功能。动态代理主要应用于事件监听、事务处理、性能监控等场景,提供了更加灵活...

    Java动态代理两种实现方式

    Java动态代理技术是Java编程中一个非常重要的特性,它允许我们在运行时动态创建具有特定行为的对象。这种技术常用于AOP(面向切面编程)和框架中,如Spring AOP,用于实现方法拦截、事务管理等功能。Java提供了两种...

    java动态代理 经典文章(word 2007格式的)

    - `动态代理详解.docx`则可能深入讨论了Java动态代理的细节,包括实际应用场景和常见问题。 以上内容是对Java动态代理技术的概述,详细的学习和理解可以通过阅读给定的文档来深入探讨。在实践中,掌握动态代理能够...

    java代理机制 JDK动态代理和cglib代理 详解

    本文将深入探讨两种主要的Java代理实现:JDK动态代理和CGLIB代理。 一、JDK动态代理 JDK动态代理基于接口实现,它要求被代理的类必须实现至少一个接口。在运行时,Java会动态地创建一个新的类,这个类实现了与原始...

    Java动态代理机制详解_动力节点Java学院整理

    Java动态代理机制详解_动力节点Java学院整理,动力节点口口相传的Java黄埔军校

    jdk动态代理技术详解

    JDK 动态代理技术详解 JDK 动态代理技术是 Java 语言自身对动态代理的支持,类似于 JDK 中在 java.util 包中提供 Observable 类和 Observer 接口提供对观察者模式的语言级支持。动态代理的优点是可以动态地为软件...

    java 动态代理实现AOP

    ### Java动态代理实现AOP详解 #### 一、引言 随着软件开发复杂度的提升,传统的面向对象编程(OOP)已经难以满足现代软件工程的需求。为了更好地管理跨切面的关注点,如日志记录、性能监控、安全控制等,面向切面...

    java静态代理和动态代理详解

    在Java中,代理主要分为静态代理和动态代理。 1. 静态代理: 静态代理是在编译时就已经明确代理类与被代理类关系的一种方式。为了实现静态代理,我们需要手动创建一个代理类,这个代理类通常会实现与被代理类相同的...

    Java动态代理方法详解.docx

    Java动态代理是一种编程技术,它允许在运行时创建代理类,这些代理类可以拦截和增强对目标对象的调用。动态代理主要分为两种类型:JDK动态代理和CGLIB代理。 一、JDK动态代理 JDK动态代理是通过Java反射API实现的,...

    Java中的动态代理

    #### 四、动态代理流程详解 1. **创建接口实现类**:首先定义一个接口`HelloWorld`,并实现一个具体的类`HelloWorldImpl`。 2. **创建InvocationHandler实现类**:定义`HelloWorldHandler`类实现`InvocationHandler...

    java动态代理

    ### Java动态代理知识点详解 #### 一、Java动态代理简介 在Java中,动态代理是一种非常重要的机制,它允许我们在运行时动态地创建一个接口的实现类实例。这种技术广泛应用于AOP(面向切面编程)、RPC(远程过程...

    Java2编程详解(Special_Edition_Using_Java)

    最后,本书可能还会涉及Java的最新特性,比如注解、泛型、枚举、动态代理等,这些都是Java语言不断演进的体现,有助于开发者保持与时俱进。 通过阅读《Java2编程详解》,无论是初学者还是有经验的开发者,都能从中...

    spring 的动态代理详解

    ### Spring的动态代理详解 #### 引言 在软件工程领域,代理模式是一种常见的设计模式,主要用于增强对象的功能,而不改变其原有的结构。Spring框架利用动态代理技术为Bean提供了横切关注点(如事务管理、日志记录...

Global site tag (gtag.js) - Google Analytics