从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
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.
最后编写测试类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){
}
}
}
}
发表评论
-
深入分析 Java 中的中文编码问题
2011-11-16 07:45 0几种常见的编码格式 ... -
Java 编码
2011-11-16 07:44 0http://zhidao.baidu.com/quest ... -
java字符编码原理解析
2011-11-16 07:43 0可以理解为计算机没 ... -
HttpClient
2011-11-03 11:07 836From http://www.blogjava.net/Al ... -
ECLIPSE ANT OutOfMemoryError
2011-08-04 17:23 1018ANT BUILD MEMORY ERROR: [cl ... -
JDBC BATCH
2011-07-05 14:58 0PreparedStatement ps = conn.pre ... -
OUT OF MEMORY WHEN BUILD
2011-02-22 17:47 01, ANT BUILD: In Eclipse op ... -
spring weblogic jndi
2011-02-16 09:18 1844weblogic:weblogic8.1 数据库:MySql ... -
log4j 邮件
2011-01-24 15:54 0<!-- 设置上下文参数 --> ... -
tomcat weblogic
2010-12-01 11:25 1868EJB 层基本搞定,以前测试 EJB 也都是写一个 appli ... -
ant weblogic “local class incompatible: stream classdesc serialVersionUI”
2010-11-29 12:41 2239weblogic.management.Management ... -
Debugging with the Maven Jetty Plugin in Eclipse
2010-11-15 17:42 1043debug: http://docs.codehaus.or ... -
maven tomcat eclipse debug
2010-11-15 17:36 1956from: http://bandaidprogrammin ... -
maven app tomcat 部署
2010-11-11 15:56 1312修改pom.xml,添加如下配置: <build ... -
Maven Cargo Tomcat 部署
2010-11-11 15:49 1774pom.xml中<build>下添加如下代码: ... -
java中读取配置文件各种方法
2010-09-07 12:31 01。使用Java.util.Properties类的load( ... -
ThreadGroup
2010-05-25 08:47 0在Java中每个线程都属于某个线程组(ThreadGroup) ... -
java Excel 导出
2010-03-28 20:06 0public void createExcel(OutputS ... -
java小数保留两位小数
2009-11-19 16:49 2341方式一: 四舍五入 double f = ... -
java中实现xml schema 验证文件
2009-11-16 20:05 3884XML 是可扩展标记语言,也就是说其中的标记我们可以按照我们 ...
相关推荐
2. 创建一个Java客户端类,通过动态代理实现对服务的调用。 ### 结语 了解并掌握Axis的安装配置以及开发Web服务的方法,对于开发者来说至关重要。无论是使用DII、Stubs还是Dynamic Proxy,都可以帮助我们快速地...
1. **Dynamic Invocation Interface (DII)**:这是一种动态调用接口,允许开发者直接与SOAP消息交互,无需生成客户端存根类。 2. **Stubs方式**:也称为静态代理,通过 Axis 自动生成的客户端和服务器端代码,使得...
3. **Dynamic Proxy方式**:动态代理,可以快速创建服务客户端。 **1. DII方式开发服务** (1)编写服务端程序:创建一个名为HelloClient的Java类,提供一个getName方法返回问候语。 (2)部署服务:将HelloClient...
- **概念介绍**:Dynamic Proxy是一种利用代理模式访问Web服务的方式。这种方式允许用户定义一个接口,然后通过该接口创建代理对象来调用Web服务,从而隐藏了具体的调用细节。 - **服务端程序编写**: - **使用上...
7. **动态代理(Dynamic Proxy)**:介绍Java的反射API和Proxy类如何实现动态代理,用于创建接口的代理对象,实现方法调用的拦截和增强。 8. **Java Servlet**:讲解Servlet的基本概念、生命周期、请求响应处理流程...
Java WebService 是一种基于WSDL...通过动态调用接口(DII)和动态代理,开发人员可以灵活地创建和消费Web服务,而无需过多关注底层的通信细节。了解和熟练掌握这些技术对于构建可扩展和可维护的分布式系统至关重要。
1. **Java高级特性**:在第13部分和14部分,可能会深入讲解Java的一些高级特性,如反射(Reflection)、注解(Annotation)、动态代理(Dynamic Proxy)等。反射允许程序在运行时检查类的信息,并能创建和调用类的...
动态代理(Dynamic Proxy)调用 动态代理允许在运行时创建接口实现,这使得调用Web服务更加灵活。使用`javax.xml.rpc.Stub`接口,你可以通过`setOperationName()`和`setNamespaceURI()`等方法来设置操作名称和命名...
2. **Dynamic Proxy方式** - 动态代理,通过代理对象调用Web服务。 3. **Stubs方式** - 通常推荐使用,因为它既通用又灵活。我们将专注于这个方法。 以下是使用Stubs方式开发Web服务的步骤: 1. **创建Web工程** ...
- **`request.getServletPath()`**: 获取当前Servlet的路径。 - **`request.getRealPath("/")`**: 获取应用程序根目录的物理路径。 - **`request.getQueryString()`**: 获取请求的查询字符串。 - **`request....
30. **动态代理(Dynamic Proxy)**:Java允许创建动态代理类,实现接口并在运行时提供拦截处理。 以上是Java新手入门必须掌握的30个基本概念,这些知识构成了Java程序员的基础,也是进一步深入学习Java技术体系的...
CGLIB 用来生成动态代理类(dynamic proxy classes),供核心 DI 和 AOP 实现之用。由于大量 Spring 功能的实现都要用到 CGLIB,因此你的应用几乎总需包含 CGLIB。 coscos.jar COS 代表 com.oreilly.servlet,是个...
随着学习的深入,你将接触Java的高级特性,如反射(Reflection)、动态代理(Dynamic Proxy)、注解(Annotation)、泛型(Generics)、并发编程(Concurrent Programming,包括synchronized、volatile、java.util....
##### 3.3 动态代理方式 (Dynamic Proxy) 1. **定义服务接口**: ```java public interface HelloClientInterface extends java.rmi.Remote { public String getName(String name) throws java.rmi....
在服务器端的开发过程中,首先创建一个Dynamic Web Project,并确保在项目中生成了`web.xml`文件。接着,将Apache CXF的`lib`目录下的jar文件导入到项目中,包括必要的CXF和Spring库。在`web.xml`中,配置Spring容器...
动态代理是一种更加灵活的方式,它可以在运行时生成代理类来调用远程服务。 - **定义接口** 定义一个接口 `HelloClientInterface`,声明你想要调用的方法。 ```java public interface HelloClientInterface ...
需要在`proxy-config.xml`文件中配置代理地址。 - `url`: 指定访问的Servlet地址。 - `result`: 处理请求结果的方法。 3. **配置proxy-config.xml文件**: - **示例**: ```xml <dynamic-url> ...