`
whp0731
  • 浏览: 176004 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

java语言反射与动态代理学习笔记2(动态代理部分)

    博客分类:
  • J2SE
阅读更多

java语言反射与动态代理学习笔记(动态代理部分)。(学习材料张龙:“反射机制与动态代理”、jdk帮助文档、book:JAVA.2核心技术)

一、代理模式:
代理模式的作用是:为其他对象提供一种代理以控制对这个对象的访问。在某些情况下,一个客户不想或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用。
代理模式一般涉及到的角色有
抽象角色:声明真实对象和代理对象的共同接口
代理角色:代理对象角色内部含有对真实对象的引用,从而可以操作真实对象,同时代理对象提供与真实对象相同的接口以便在任何时刻都能代替真实对象。同时,代理对象可以在执行真实对象操作时,附加其他的操作,相当于对真实对象进行封装(面向对象的三大特性(继承、多态、封装)之一:)
真实角色:代理角色所代表的真实对象,是我们最终要引用的对象。
代理模式也是《设计模式》(由4个人书写很有名)中的23中模式之一。

1、程序Subject.java

// 抽象角色
abstract public class Subject
{
    abstract public void request();
} 

 2、程序RealSubject.java

public class RealSubject extends Subject
{
   public RealSubject()
    {
    }
    public void request()
    {
        System.out.println("From real subject.");
    }
}

 
3、程序ProxySubject.java

//代理角色

public class ProxySubject extends Subject

{
    private RealSubject realSubject; // 以真实角色作为代理角色的属性
    public ProxySubject()
    {
    }
    public void request() // 该方法封装了真实对象的request方法
    {
        preRequest();
        if (realSubject == null)
        {
            realSubject = new RealSubject();
        }
        realSubject.request(); // 此处执行真实对象的request方法
        postRequest();
    }

    private void preRequest()
    {
        // something you want to do before requesting
    }

    private void postRequest()
    {
        // something you want to do after requesting
    }

}

 4、程序Client.java  

//客户端调用
public class Client
{
    public static void main(String[] args)
    {
        Subject sub = new ProxySubject();
        sub.request();
    }
}

  

二、另外,如果要按照上述的方法使用代理模式,那么真实角色必须是事先已经存在的,并将其作为代理对象的内部属性。但是实际使用时,一个真实角色必须对应一个 代理角色,如果大量使用会导致类的急剧膨胀;此外,如果事先并不知道真实角色,该如何使用代理呢?这个问题可以通过Java的动态代理类来解决。
1、Java动态代理类位于java.lang.reflect包下,一般主要涉及到以下两个类:

(1)Interface InvocationHandler:该接口中仅定义了一个方法
public object invoke(Object obj,Method method, Object[] args)
在实际使用时,第一个参数obj一般是指代理类,method是被代理的方法,如上例中的request(),args为该方法的参数数组。 这个抽象方法在代理类中动态实现。

 

(2)Dynamic Proxy:该类即为动态代理类,作用类似于上例中的ProxySubject,其中主要包含以下内容
protected Proxy(InvocationHandler h):构造函数,用于给内部的h赋值。h是实现了 InvocationHandler 接口的对象。


static Class getProxyClass (ClassLoader loader, Class[] interfaces):获得一个代理类,其中loader是类装载器,interfaces是真实类所拥有的全部接口的数组。


static Object newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h):h是实现了 InvocationHandler 接口的对象,返回代理类的一个实例,返回后的动态代理类可以当作被代理类使用(可使用被代理类的在Subject接口中声明过的方法)以下实例中的Client.java类中就是用了该方法。

 

(3)所谓Dynamic Proxy动态代理类是这样一种class:它是在运行时生成的class,在生成它时你必须提供一组interface给它,然后该class就宣称它实现了这些 interface。你当然可以把该类的实例当作这些interface中的任何一个来用。(此处就是运用了类的动态多态性,用类的实例来赋值给该类的某个接口的意思)当然,这个Dynamic Proxy其实就是一个Proxy,它不会替你作实质性的工作,在生成它的实例时你必须提供一个h,由它接管实际的工作。

 

(4)所以在使用动态代理类时,我们必须实现接口InvocationHandler(即重写invoke方法)


程序 Subject.java

//抽象角色(之前是抽象类,此处应改为接口): 
public interface Subject
{
    abstract public void request();
}

 程序 RealSubject.java

//具体角色
public class RealSubject implements Subject
{
    public RealSubject()
    {
    }
    public void request()
    {
        System.out.println("From real subject.");
    }
}

 
程序 DynamicSubject.java

//代理处理器 
/**
 * 该代理类的内部属性为Object类,实际使用时通过该类的构造函数DynamicSubject(Object obj)对其赋值;
 * 此外,在该类还实现了invoke方法,该方法中的 method.invoke(sub,args);
 * 其实就是调用被代理对象的将要被执行的方法,方法参数sub是实际的被代理对象,
 * args为执行被代理对象相应操作所需的参数。
 * 通过动态代理类,我们可以在调用之前或之后执行一些相关操作
 */
public class DynamicSubject implements InvocationHandler
{
    private Object sub;
    public DynamicSubject()
    {
    }
    public DynamicSubject(Object obj)
    {
        sub = obj;
    }
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable//该方法java会去调用,我们程序中没有调用到
    {
        System.out.println("before calling " + method);//可以自己写一些方法执行前操作
        method.invoke(sub, args);
        System.out.println("after calling " + method);//可以自己写一些方法执行后的操作!
        return null;
 }
}



 程序 Client.java

//客户端
public class Client
{
 static public void main(String[] args) throws Throwable
 {
  RealSubject rs = new RealSubject(); // 在这里指定被代理类
  InvocationHandler ds = new DynamicSubject(rs);//生成一个上文所说的h
  Class<?> cls = rs.getClass();
//   以下是一次性生成代理
  Subject subject = (Subject) Proxy.newProxyInstance(
    cls.getClassLoader(), cls.getInterfaces(), ds);
  subject.request();//调用过程:request--->上一行的ds------>ds会动态调运它invoke----->invoke方法内真实对象
 }
}

Subject subject 指的是代理类

 通过这种方式,被代理的对象(RealSubject)可以在运行时动态改变,需要控制的接口(Subject接口)可以在运行时改变,控制的方式(DynamicSubject类)也可以动态改变,从而实现了非常灵活的动态代理关系
动态代理是指客户通过代理类来调用其它对象的方法
动态代理使用场合:
a、调试
b、远程方法调用(RMI)
调运顺序:客户---->代理接口(代理)---->接口(对象)

2总结:动态代理步骤
(1).创建一个实现接口InvocationHandler的类,它必须覆写invoke方法(一般为建立一个DynamicSubjet类)
(2).创建被代理的类以及接口
(3).通过Proxy的静态方法
newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h)创建一个代理
 (4).通过代理调用方法


3.程序实践

VectorProxy.java 

public class VectorProxy implements InvocationHandler
{
 private Object proxyobj;

 public VectorProxy(Object obj)
 {
  proxyobj = obj;
 }

 public static Object factory(Object obj)
 {
  Class<?> cls = obj.getClass();

  return Proxy.newProxyInstance(cls.getClassLoader(),
    cls.getInterfaces(), new VectorProxy(obj));
 }
 public Object invoke(Object proxy, Method method, Object[] args)
   throws Throwable
 {
  System.out.println("before calling " + method);

  if (args != null)
  {
   for (int i = 0; i < args.length; i++)
   {
    System.out.println(args[i] + "");
   }
  }
  Object object = method.invoke(proxyobj, args);

  System.out.println("after calling " + method);
  return object;
 }
 public static void main(String[] args)
 {
  List<String> v = (List<String>) factory(new Vector<String>(10));

  v.add("New");
  v.add("York");
  System.out.println(v.toString);
  v.remove(0);
  System.out.println(v);
 }
}

  

若修改该程序的main函数中的System.out.println(v.toString);为System.out.println(v.getClass);发现程序在执行中不会再去执行invoke方法了,而是直接打印出结果而已
为什么呢?参考jdk5.0的文档可以发现在java.lang.reflect.Proxy类中说明如下:
An invocation of the hashCode, equals, or toString methods declared in java.lang.Object on a proxy instance will be encoded and dispatched to the invocation handler's invoke method in the same manner as interface method invocations are encoded and dispatched, as described above. The declaring class of the Method object passed to invoke will be java.lang.Object. Other public methods of a proxy instance inherited from java.lang.Object are not overridden by a proxy class, so invocations of those methods behave like they do for instances of java.lang.Object.
翻译如下:
在代理实例上的 java.lang.Object 中声明的 hashCode、equals 或 toString 方法的调用将按照与编码和指派接口方法调用相同的方式进行编码,并被指派到调用处理程序的 invoke 方法,如上所述。传递到 invoke 的 Method 对象的声明类是 java.lang.Object。代理类不重写从 java.lang.Object 继承的代理实例的其他公共方法,所以这些方法的调用行为与其对 java.lang.Object 实例的操作一样。

 

即对于  java.lang.Object 中声明的 hashCode、equals 或 toString 方法的调用将按照invoke 方法,其他方法就像以前方法一样,正常进行,不代理。


             

 

 

 

 

 

 

 

 

 

分享到:
评论

相关推荐

    java语言反射与动态代理学习笔记

    ### Java语言反射与动态代理深度解析 #### 一、Java反射机制详解 Java反射机制是Java编程语言的一个强大特性,允许程序在运行时检查和修改自身结构与行为。这一机制为开发人员提供了高度的灵活性,尤其是在框架...

    Java基础 学习笔记 Markdownr版

    本学习笔记主要涵盖了Java的基础知识,包括面向对象、集合、IO流、多线程、反射与动态代理以及Java 8的新特性等方面,旨在帮助初学者或有经验的开发者巩固和提升Java编程技能。 1. 面向对象(OOP):Java的核心是...

    Java学习笔记(源码)

    学习笔记会讲解如何使用反射进行元编程,以及其在插件系统、序列化和动态代理中的应用。 9. **Java API和库**:Java标准库提供了大量的预定义类和接口,如JDBC(Java数据库连接)用于数据库操作,Swing和JavaFX用于...

    《Java学习笔记》

    2. **类与对象**:Java的核心是面向对象编程,这一部分会深入讲解类的定义、对象的创建、封装、继承和多态性等概念。还会涉及构造函数、访问修饰符、抽象类和接口等关键概念。 3. **异常处理**:Java的异常处理机制...

    java反射笔记

    Java反射是Java编程语言中的一个强大特性,它允许运行中的Java程序对自身进行检查并且可以直接操作程序的内部属性。在“java反射笔记”这个主题中,我们将深入探讨反射的基础知识和常见用法。 首先,我们需要了解...

    java学习笔记(java 反射机制 流 内存管理)

    Java反射机制是Java语言的一个强大特性,允许程序在运行时检查类、接口、字段和方法的信息,并能动态地创建对象和调用方法。通过java.lang.Class和java.lang.reflect包中的类,我们可以获取类的结构信息,包括构造器...

    JAVA学习笔记————————

    2. **面向对象编程**:JAVA是纯面向对象的语言,因此学习笔记中会详细讲解类的创建、继承、封装、多态等概念。同时,接口、抽象类以及访问修饰符也是重要的知识点。 3. **异常处理**:JAVA提供了一种结构化的异常...

    个人Java学习过程中所有学习笔记

    7. **反射与动态代理**:反射允许在运行时检查和操作类、接口和对象,而动态代理则用于创建代理对象以在运行时拦截方法调用。这些高级特性在框架开发和元编程中有着广泛应用。 8. **Java标准库**:Java的标准库...

    JAVA 私塾笔记整理——反射机制(Reflection)

    - 动态代理:Java的`java.lang.reflect.Proxy`类可以利用反射创建动态代理对象。 - 测试工具:JUnit等测试框架利用反射来调用私有方法进行测试。 - 数据库操作:ORM框架如Hibernate通过反射将数据库记录映射为...

    java面试题 学习笔记

    Java面试题及学习笔记概述 Java作为一种广泛应用的编程语言,其面试题库广泛且深入,涵盖了从基础语法到高级特性的各个层面。本篇将基于常见的Java面试问题,结合学习笔记,深入探讨Java的核心概念和技术。 一、...

    良葛格Java学习笔记

    【良葛格Java学习笔记】 本笔记主要涵盖了Java编程语言的核心概念和技术,旨在帮助初学者以及有一定基础的开发者深入理解并掌握Java。Java作为一种广泛应用于企业级应用开发、移动开发(尤其是Android)以及大数据...

    良葛格Java学习笔记html.rar

    9. **高级特性**:可能包括反射、动态代理、注解、JavaFX等。 10. **实战项目**:提供实际案例,让学习者将理论知识应用到实际编程中,提高问题解决能力。 通过这份详尽的HTML笔记,学习者可以系统地学习Java编程...

    Java反射笔记源代码

    Java反射是Java编程语言中的一个强大特性,它允许程序在运行时检查类、接口、字段和方法的信息,并能动态地创建对象和调用方法。在深入理解Java反射之前,我们首先要明白,它属于元编程的一部分,即在编写程序时能够...

    java学习笔记JDK6

    Java JDK 6学习笔记是Java开发者入门和进阶的重要参考资料,由知名作者林信良编著。本笔记主要涵盖了JDK 6版本的核心特性和关键概念,为读者提供了全面而深入的学习路径。以下是对其中重要知识点的详细阐述: 1. **...

    Java JDK 7学习笔记(国内第一本Java 7,前期版本累计销量5万册)

     《Java JDK 7学习笔记》详细介绍了JVM、JRE、Java SE API、JDK与IDE之间的对应关系。必须要时从Java SE API的源代码分析,了解各种语法在Java SE API中如何应用。  《Java JDK 7学习笔记》将IDE操作纳为教学内容...

    java学习笔记6

    8. **反射与动态代理**:Java反射允许程序在运行时检查和修改自身行为,而动态代理则可以创建在运行时决定行为的代理对象。 9. **JDBC数据库操作**:Java Database Connectivity(JDBC)是Java连接数据库的标准API...

    Java 学习笔记.zip

    这份"Java学习笔记.zip"文件显然包含了一些关于Java编程的基础到高级的学习资料,特别聚焦于JDK 6版本。JDK(Java Development Kit)是Java开发环境的核心组件,包括了Java编译器、Java运行时环境、以及丰富的API库...

    [Java学习笔记doc]之反射机制

    Java反射机制是Java编程语言中的一个强大特性,它允许运行中的Java程序对自身进行检查并且可以直接操作程序的内部属性。在深入理解这个知识点之前,我们首先需要了解什么是反射。反射是指在运行时,程序可以获取类的...

    Java公司培训经典学习笔记

    Java公司培训经典学习笔记是针对Java编程语言进行深入学习的一份宝贵资料,涵盖了从基础到高级的诸多知识点,旨在帮助开发者提升技能,适应企业级项目开发的需求。以下将详细阐述这些笔记中的关键点: 1. **Java...

Global site tag (gtag.js) - Google Analytics