- 浏览: 1150865 次
- 性别:
- 来自: 北京
文章分类
最新评论
-
RebeccaZhong:
严重: StandardWrapper.Throwableco ...
三步发布java方式的rest服务 -
RebeccaZhong:
严重: StandardWrapper.Throwableco ...
三步发布java方式的rest服务 -
冷酷月光:
楼主。请教一下。arcgis for android 有提供地 ...
ArcGIS API For Android离线地图的实现 -
winney117:
请问如何GET已有网页上的指定内容?比如百度文库中的某一篇文章 ...
三步发布java方式的rest服务 -
zige1012:
您好,我想问问我想换个自己地图的切片,也有4层(L0-L3), ...
ArcGIS API For Android离线地图的实现
从JDK1.3开始,Java就引入了动态代理的概念。动态代理(Dynamic Proxy)可以帮助你减少代码行数,真正提高代码的可复用度。例如,你不必为所有的类的方法里面都写上相同的Log代码行,取而代之的是实用类的动态代理类。当然,这种便利是有条件的。本文简单介绍Java动态代理的原理,并实现一个被代理的Servlet创建,和调用的过程。
1.代理模式(Proxy Pattern)
在JDK1.3以前,代理模式就已流行,所以得代理模式是生成一个和类相同接口的代理类,用户通过使用代理类来封装某个实现类。如图1,其目的是加强实现类的某个方法的功能,而不必改变原有的源代码。
2.动态代理(Dynamic Proxy)
随着Proxy的流行,Sun把它纳入到JDK1.3实现了Java的动态代理。动态代理和普通的代理模式的区别,就是动态代理中的代理类是由java.lang.reflect.Proxy类在运行期时根据接口定义,采用Java反射功能动态生成的。和java.lang.reflect.InvocationHandler结合,可以加强现有类的方法实现。如图2,图中的自定义Handler实现InvocationHandler接口,自定义Handler实例化时,将实现类传入自定义Handler对象。自定义Handler需要实现invoke方法,该方法可以使用Java反射调用实现类的实现的方法,同时当然可以实现其他功能,例如在调用实现类方法前后加入Log。而Proxy类根据Handler和需要代理的接口动态生成一个接口实现类的对象。当用户调用这个动态生成的实现类时,实际上是调用了自定义Handler的invoke方法。
3.动态代理Servlet
虽然Web Application Server的产品很多,但Servlet的处理原理是相似的:动态加载Servlet,调用Servlet的init方法(只被调用一次),并保存到Servlet容器;Servlet使用时,调用Servlet的service方法。本文动态代理Servlet接口,使其init和service被调用时会在控制台打出方法调用前后信息。
首先实现2个Servlet,DefaultServlet和UserServlet
然后实现InvocationHandler
实现Servlet的调用
其中的ServletConfig保存Servlet相关信息。ServletContext保存所有的Servlet对象。WebAppClassLoader为自定义class loader,参见http://blog.csdn.net/tyrone1979/archive/2006/09/03/1164262.aspx。
最后编写测试类Main,该类模拟10个用户访问Servlet,5人访问DefaultServlet,5人访问UserServlet。
测试结果如下
org.colimas.servlet.DefaultServlet:Init servlet starting...
org.colimas.servlet.DefaultServlet:Running init
org.colimas.servlet.DefaultServlet:Init servlet ending...
org.colimas.servlet.UserServlet:Init servlet starting...
org.colimas.servlet.UserServlet:Running init
org.colimas.servlet.UserServlet:Init servlet ending...
org.colimas.servlet.DefaultServlet:service starting...
org.colimas.servlet.DefaultServlet:service ending...
org.colimas.servlet.DefaultServlet:service starting...
org.colimas.servlet.DefaultServlet:service ending...
org.colimas.servlet.UserServlet:service starting...
org.colimas.servlet.UserServlet:Do UserSErvlet Get
org.colimas.servlet.UserServlet:service ending...
org.colimas.servlet.UserServlet:service starting...
org.colimas.servlet.UserServlet:Do UserSErvlet Get
org.colimas.servlet.UserServlet:service ending...
org.colimas.servlet.UserServlet:service starting...
org.colimas.servlet.UserServlet:Do UserSErvlet Get
org.colimas.servlet.UserServlet:service ending...
org.colimas.servlet.UserServlet:service starting...
org.colimas.servlet.DefaultServlet:service starting...
org.colimas.servlet.UserServlet:service starting...
org.colimas.servlet.DefaultServlet:service starting...
org.colimas.servlet.DefaultServlet:service starting...
org.colimas.servlet.UserServlet:Do UserSErvlet Get
org.colimas.servlet.DefaultServlet:service ending...
org.colimas.servlet.UserServlet:Do UserSErvlet Get
org.colimas.servlet.DefaultServlet:service ending...
org.colimas.servlet.DefaultServlet:service ending...
org.colimas.servlet.UserServlet:service ending...
org.colimas.servlet.UserServlet:service ending...
2个Servlet第一次Load时初始化,被调用init,之后保存到ServletContext中。第二次直接从ServletContext获得,执行service。红字表示代理类里增加的输出结果。
4.动态代理的限制
JDK的动态代理并不能随心所欲的代理所有的类。Proxy.newProxyInstance方法的第二个参数只能是接口数组, 也就是Proxy只能代理接口。
1.代理模式(Proxy Pattern)
在JDK1.3以前,代理模式就已流行,所以得代理模式是生成一个和类相同接口的代理类,用户通过使用代理类来封装某个实现类。如图1,其目的是加强实现类的某个方法的功能,而不必改变原有的源代码。
2.动态代理(Dynamic Proxy)
随着Proxy的流行,Sun把它纳入到JDK1.3实现了Java的动态代理。动态代理和普通的代理模式的区别,就是动态代理中的代理类是由java.lang.reflect.Proxy类在运行期时根据接口定义,采用Java反射功能动态生成的。和java.lang.reflect.InvocationHandler结合,可以加强现有类的方法实现。如图2,图中的自定义Handler实现InvocationHandler接口,自定义Handler实例化时,将实现类传入自定义Handler对象。自定义Handler需要实现invoke方法,该方法可以使用Java反射调用实现类的实现的方法,同时当然可以实现其他功能,例如在调用实现类方法前后加入Log。而Proxy类根据Handler和需要代理的接口动态生成一个接口实现类的对象。当用户调用这个动态生成的实现类时,实际上是调用了自定义Handler的invoke方法。
3.动态代理Servlet
虽然Web Application Server的产品很多,但Servlet的处理原理是相似的:动态加载Servlet,调用Servlet的init方法(只被调用一次),并保存到Servlet容器;Servlet使用时,调用Servlet的service方法。本文动态代理Servlet接口,使其init和service被调用时会在控制台打出方法调用前后信息。
首先实现2个Servlet,DefaultServlet和UserServlet
package org.colimas.servlet; import javax.servlet.Servlet; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; public class DefaultServlet extends HttpServlet implements Servlet { public void init() throws ServletException { super.init(); System.out.println(DefaultServlet.class.getName()+":Running init"); } public String getServletInfo() { return DefaultServlet.class.getName(); } } package org.colimas.servlet; import java.io.IOException; import javax.servlet.Servlet; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; public class UserServlet extends HttpServlet implements Servlet { private static final long serialVersionUID = -7016554795165038652L; public void init() throws ServletException { super.init(); System.out.println(UserServlet.class.getName()+":Running init"); } protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { System.out.println(UserServlet.class.getName()+":Do UserSErvlet Get"); } public String getServletInfo() { return UserServlet.class.getName(); } }
然后实现InvocationHandler
package org.colimas.webapp; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import javax.servlet.Servlet; public class ServletHandler implements InvocationHandler { private Servlet obj; public ServletHandler(Servlet obj){ this.obj=obj; } public Object invoke(Object arg0, Method arg1, Object[] arg2) throws Throwable { if(arg1.getName().compareTo("init")==0){ //调用init时 System.out.println(obj.getServletInfo()+":Init servlet starting..."); //增加控制台输出。 arg1.invoke(obj,arg2); //调用init方法 System.out.println(obj.getServletInfo()+":Init servlet ending..."); //增加控制台输出。 else if(arg1.getName().compareTo("service")==0){ //调用service时 System.out.println(obj.getServletInfo()+":service starting..."); //增加控制台输出。 arg1.invoke(obj,arg2); //调用service方法。 System.out.println(obj.getServletInfo()+":service ending..."); //增加控制台输出。 } return null; } }
实现Servlet的调用
package org.colimas.webapp; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Proxy; import javax.servlet.Servlet; import javax.servlet.ServletConfig; import javax.servlet.ServletContext; import javax.servlet.ServletException; public class ServletWrapperImp { private Class servletClass; private ServletConfig config; private String _servletname; private Servlet _theServlet; private ServletContext context; public ServletWrapperImp(ServletConfig config){ this.config=config; this._servletname=this.config.getServletName(); this.context=this.config.getServletContext(); } public Servlet getServlet() throws ServletException{ destroy(); try { WebAppClassLoader loader=new WebAppClassLoader(this.getClass().getClassLoader()); //自定义class loader String name=getServletName(); //从ServletConfig中获得Servlet Name synchronized (context) { Servlet theServlet=context.getServlet(name); //在ServletContext中查找Servlet if(theServlet==null){ //如果ServletContext没有。 servletClass = loader.loadClass(name); //由Class loader 加载Servlet class。 theServlet = (Servlet) servletClass.newInstance(); //Servlet实例化。 WebAppContext.addServlet(name,theServlet); //将Servlet实例存入ServletContext。 InvocationHandler handler=new ServletHandler(theServlet); //自定义ServletHandler,参见ServletHandler类。 _theServlet=(Servlet)Proxy.newProxyInstance(theServlet.getClass().getClassLoader(), new Class[]{Servlet.class},handler); //代理类实例化。 _theServlet.init(config); //Servlet代理对象调用init方法。参见ServletHandler的invoke方法。 }else{ //ServletContext里已存在。 InvocationHandler handler=new ServletHandler(theServlet); //自定义ServletHandler,参见ServletHandler类。 _theServlet=(Servlet)Proxy.newProxyInstance(theServlet.getClass().getClassLoader(), new Class[]{Servlet.class},handler); //代理Servlet接口,动态生成代理类,并实例化。 } } return _theServlet; //返回Servlet代理对象 } catch( ClassNotFoundException ex1 ) { } catch( InstantiationException ex ) { }catch(IllegalAccessException ex2){ } return null; } public void destroy() { if (_theServlet != null) { _theServlet.destroy(); } } protected String getServletName(){ return _servletname; } }
其中的ServletConfig保存Servlet相关信息。ServletContext保存所有的Servlet对象。WebAppClassLoader为自定义class loader,参见http://blog.csdn.net/tyrone1979/archive/2006/09/03/1164262.aspx。
最后编写测试类Main,该类模拟10个用户访问Servlet,5人访问DefaultServlet,5人访问UserServlet。
package org.colimas.main; import java.io.IOException; import javax.servlet.Servlet; import javax.servlet.ServletConfig; import javax.servlet.ServletException; import org.colimas.webapp.HttpServletRequestWrapper; import org.colimas.webapp.HttpServletResponseWrapper; import org.colimas.webapp.ServletConfigImpl; import org.colimas.webapp.ServletWrapper; import org.colimas.webapp.ServletWrapperImp; import org.colimas.webapp.WebAppContext; public class Main { private ThreadGroup _threadGroup; private Thread[] _threads; String defaultServletName="org.colimas.servlet.DefaultServlet"; String userServletName="org.colimas.servlet.UserServlet"; WebAppContext context=WebAppContext.newInstance(); public void doStart(){ _threadGroup=new ThreadGroup("SERVLETS"); int i=0; _threads=new ServletThread[10]; //模拟10位用户。 for(i=0;i<5;i++){ _threads[i]=new ServletThread(_threadGroup,new Integer(i).toString(), defaultServletName); _threads[i].start(); } for(i=5;i<10;i++){ _threads[i]=new ServletThread(_threadGroup,new Integer(i).toString(), userServletName); _threads[i].start(); } } /** * @param args */ public static void main(String[] args) { Main _main=new Main(); _main.doStart(); } //模拟用户线程 private class ServletThread extends Thread{ private String servletName; public ServletThread(ThreadGroup group,String threadname,String servletname){ super(group,threadname); servletName=servletname; } //调用Servlet的service. public void run() { //用户调用Servlet ServletConfig config=new ServletConfigImpl(servletName,context); //调用的Servlet信息。 ServletWrapperImp wrapper=new ServletWrapperImp(config); try { Servlet defaultServlet=wrapper.getServlet(); //获得Servlet对象,实际是Servlet的代理对象 defaultServlet.service(new HttpServletRequestWrapper(), new HttpServletResponseWrapper()); 调用代理对象的service方法,参见ServletHandler的invoke方法。 } catch (ServletException e) { e.printStackTrace(); } catch(IOException e){ } } } }HttpServletRequestWrapper和HttpServletResponseWrapper实现HttpServletRequest,和HttpServletResponse。
测试结果如下
org.colimas.servlet.DefaultServlet:Init servlet starting...
org.colimas.servlet.DefaultServlet:Running init
org.colimas.servlet.DefaultServlet:Init servlet ending...
org.colimas.servlet.UserServlet:Init servlet starting...
org.colimas.servlet.UserServlet:Running init
org.colimas.servlet.UserServlet:Init servlet ending...
org.colimas.servlet.DefaultServlet:service starting...
org.colimas.servlet.DefaultServlet:service ending...
org.colimas.servlet.DefaultServlet:service starting...
org.colimas.servlet.DefaultServlet:service ending...
org.colimas.servlet.UserServlet:service starting...
org.colimas.servlet.UserServlet:Do UserSErvlet Get
org.colimas.servlet.UserServlet:service ending...
org.colimas.servlet.UserServlet:service starting...
org.colimas.servlet.UserServlet:Do UserSErvlet Get
org.colimas.servlet.UserServlet:service ending...
org.colimas.servlet.UserServlet:service starting...
org.colimas.servlet.UserServlet:Do UserSErvlet Get
org.colimas.servlet.UserServlet:service ending...
org.colimas.servlet.UserServlet:service starting...
org.colimas.servlet.DefaultServlet:service starting...
org.colimas.servlet.UserServlet:service starting...
org.colimas.servlet.DefaultServlet:service starting...
org.colimas.servlet.DefaultServlet:service starting...
org.colimas.servlet.UserServlet:Do UserSErvlet Get
org.colimas.servlet.DefaultServlet:service ending...
org.colimas.servlet.UserServlet:Do UserSErvlet Get
org.colimas.servlet.DefaultServlet:service ending...
org.colimas.servlet.DefaultServlet:service ending...
org.colimas.servlet.UserServlet:service ending...
org.colimas.servlet.UserServlet:service ending...
2个Servlet第一次Load时初始化,被调用init,之后保存到ServletContext中。第二次直接从ServletContext获得,执行service。红字表示代理类里增加的输出结果。
4.动态代理的限制
JDK的动态代理并不能随心所欲的代理所有的类。Proxy.newProxyInstance方法的第二个参数只能是接口数组, 也就是Proxy只能代理接口。
发表评论
-
Java中文排序
2011-06-14 13:11 1774所谓中文排序就是按照汉语拼音的顺序进行排序,在Java中进行排 ... -
三步发布java方式的rest服务
2010-10-18 09:47 19483大家好: 最近大家都想知道怎么发布一个java的rest方式的 ... -
JDK路径设置全解
2010-05-28 23:44 1692PATH:.;D:\Java\jdk1.6.0_10\bin ... -
java.util.Properties类的使用
2009-02-10 10:21 1826Properties 类已不是新东西了,它在 Java 编程的 ... -
java导入导出excel操作(jxl),整理中。。。
2009-02-10 10:15 4061jxl.jar 包 下载地址: http://www.and ... -
使用Struts和Velocity开发web应用
2008-12-25 23:31 1420http://www.edu999.com/IT/JAVA/2 ... -
关于JAVA断点续传
2008-10-18 10:20 2184SiteFileFetch.java负责整个文件的抓取,控制内 ... -
Spring中Quartz的配置实例
2008-10-13 15:29 2085Quartz是一个强大的企业级任务调度框架,Spring中继承 ... -
Spring中Quartz的Cron配置说明
2008-10-13 10:43 3345一个Cron-表达式是一个 ... -
在Hibernate应用中如何处理批量更新和批量删除?
2008-10-07 10:03 1149批量更新是指在一个事务中更新大批量数据,批量删除是指在一个事务 ... -
Servlet和Filter的url匹配以及url-pattern详解
2008-09-22 16:33 1895Servlet和filter是J2EE开发 ... -
用Java动态代理实现AOP
2008-09-16 20:36 1085目前整个开发社区对AOP(Aspect Oriented Pr ... -
JAVA中的反射机制详解
2008-09-16 14:44 2144JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个 ... -
从查询串中分离请求参数——acegi security中SecurityContextHolderA
2008-09-03 17:57 1996大多数情况下我们是不需要单独处理HttpServletRequ ... -
使用RememberMeProcessingFilter来实现cookies自动登陆
2008-09-03 17:56 2187要使用cookies自动登陆,我们需要配置过滤器Remembe ... -
Acegi concurrent session
2008-09-03 16:33 1342. <bean id="concurrent ... -
Web开发中的Listener和Filter
2008-08-28 16:12 2988http://www.javadby.com/Web_Serv ... -
关于struts2无法加载struts.properties的问题
2008-08-15 15:17 1689最近整合了struts2和spring的问题,其中看到一个st ... -
Hibernate配置详解
2008-08-02 23:19 1733表 3.3. Hibernate配置属 ... -
jar 命令详解
2008-08-01 08:43 1295jar 是随 JDK 安装的,在 ...
相关推荐
1. **Dynamic Invocation Interface (DII)**:这是一种动态调用接口,允许开发者直接与SOAP消息交互,无需生成客户端存根类。 2. **Stubs方式**:也称为静态代理,通过 Axis 自动生成的客户端和服务器端代码,使得...
1. **Java高级特性**:在第13部分和14部分,可能会深入讲解Java的一些高级特性,如反射(Reflection)、注解(Annotation)、动态代理(Dynamic Proxy)等。反射允许程序在运行时检查类的信息,并能创建和调用类的...
随着学习的深入,你将接触Java的高级特性,如反射(Reflection)、动态代理(Dynamic Proxy)、注解(Annotation)、泛型(Generics)、并发编程(Concurrent Programming,包括synchronized、volatile、java.util....
动态代理(Dynamic Proxy)调用 动态代理允许在运行时创建接口实现,这使得调用Web服务更加灵活。使用`javax.xml.rpc.Stub`接口,你可以通过`setOperationName()`和`setNamespaceURI()`等方法来设置操作名称和命名...
Java WebService 是一种基于WSDL...通过动态调用接口(DII)和动态代理,开发人员可以灵活地创建和消费Web服务,而无需过多关注底层的通信细节。了解和熟练掌握这些技术对于构建可扩展和可维护的分布式系统至关重要。
30. **动态代理(Dynamic Proxy)**:Java允许创建动态代理类,实现接口并在运行时提供拦截处理。 以上是Java新手入门必须掌握的30个基本概念,这些知识构成了Java程序员的基础,也是进一步深入学习Java技术体系的...
7. **动态代理(Dynamic Proxy)**:介绍Java的反射API和Proxy类如何实现动态代理,用于创建接口的代理对象,实现方法调用的拦截和增强。 8. **Java Servlet**:讲解Servlet的基本概念、生命周期、请求响应处理流程...
动态代理是一种更加灵活的方式,它可以在运行时生成代理类来调用远程服务。 - **定义接口** 定义一个接口 `HelloClientInterface`,声明你想要调用的方法。 ```java public interface HelloClientInterface ...
(3)生成客户端代理:利用`javax.xml.rpc.Service`和`java.lang.reflect.Proxy`动态创建服务代理对象,调用接口方法以访问服务。 **三、发布Web服务** Axis提供了wsdd文件来配置和发布Web服务。你可以通过编辑...
3. **Dynamic Proxy方式**:类似于DII,但提供了更灵活的服务调用方式。 ### 六、编写示例代码 #### 1. **编写服务端程序** 示例中提到了一个简单的`HelloClient`类,它包含一个`getName`方法,返回一个问候语...
- **概念介绍**:Dynamic Proxy是一种利用代理模式访问Web服务的方式。这种方式允许用户定义一个接口,然后通过该接口创建代理对象来调用Web服务,从而隐藏了具体的调用细节。 - **服务端程序编写**: - **使用上...
2. **Dynamic Proxy方式** - 动态代理,通过代理对象调用Web服务。 3. **Stubs方式** - 通常推荐使用,因为它既通用又灵活。我们将专注于这个方法。 以下是使用Stubs方式开发Web服务的步骤: 1. **创建Web工程** ...
2. **Dynamic Proxy 方式**:动态代理方式。 3. **Stubs 方式**:该方式最常用,也是本文重点介绍的方式。 #### 三、编写WebService服务端与客户端 接下来,我们将通过具体示例来展示如何使用Apache Axis搭建一个...
尽管现有的 EmployeeMgmt 应用程序已经有了 Façade 接口,但这个接口是暴露给 Servlet 使用的,最好能再为 Flex 定义另一个接口 FlexService,并隐藏 Java 语言的特定对象(如 清单 1 所示): 清单 1. Flex...
现在,我们希望用 Flex 替换掉原有的 Servlet 和 JSP 页面,就需要让 Flex 和 Java EE 后端通信。Flex 支持多种远程调用方式,包括 HTTP,Web Services 和 AMF。不过,针对 Java EE 开发的服务器端应用,可以通过...
##### 3.3 动态代理方式 (Dynamic Proxy) 1. **定义服务接口**: ```java public interface HelloClientInterface extends java.rmi.Remote { public String getName(String name) throws java.rmi....
在提供的部分内容中,可以看到一段Java代码示例,这是在一个名为`Info.jsp`的页面中用于展示通过Nginx代理后,服务端接收到的各种请求信息。这些信息包括但不限于服务器名称、协议、端口、请求方法等。 - **`...
在服务器端的开发过程中,首先创建一个Dynamic Web Project,并确保在项目中生成了`web.xml`文件。接着,将Apache CXF的`lib`目录下的jar文件导入到项目中,包括必要的CXF和Spring库。在`web.xml`中,配置Spring容器...