`
AutomaticThoughts
  • 浏览: 165545 次
社区版块
存档分类
最新评论

Cglib介绍

 
阅读更多

1.CGLIB包的介绍
    代理为控制要访问的目标对象提供了一种途径。当访问对象时,它引入了一个间接的层。JDK自从1.3版本开始,就引入了动态代理,并且经常被用来动态地创建代理。JDK的动态代理用起来非常简单,当它有一个限制,就是使用动态代理的对象必须实现一个或多个接口。如果想代理没有实现接口的继承的类,该怎么办?现在我们可以使用CGLIB包
    CGLIB是一个强大的高性能的代码生成包。它广泛的被许多AOP的框架使用,例如Spring AOP和dynaop,为他们提供方法的interception(拦截)。最流行的OR Mapping工具hibernate也使用CGLIB来代理单端single-ended(多对一和一对一)关联(对集合的延迟抓取,是采用其他机制实现的)。EasyMock和jMock是通过使用模仿(moke)对象来测试java代码的包。它们都通过使用CGLIB来为那些没有接口的类创建模仿(moke)对象。
    CGLIB包的底层是通过使用一个小而快的字节码处理框架ASM,来转换字节码并生成新的类。除了CGLIB包,脚本语言例如Groovy和BeanShell,也是使用ASM来生成java的字节码。当不鼓励直接使用ASM,因为它要求你必须对JVM内部结构包括class文件的格式和指令集都很熟悉。
   
       Figure 1: CGLIB Library and ASM Bytecode Framework
图一显示了和CGLIB包和一些框架和语言的关系图。需要注意的是一些框架例如Spring AOP和Hibernate,它们为了满足需要经常同时使用JDK的动态代理和CGLIB包。Hiberater使用JDK的动态代理实现一个专门为WebShere应用服务器的事务管理适配器;Spring AOP,如果不强制使用CGLIB包,默认情况是使用JDK的动态代理来代理接口。

2.CGLIB 代理的APIS
         CGLIB包的基本代码很少,当学起来有一定的困难,主要是缺少文档,这也是开源软件的一个不足之处。目前CGLIB的版本是(2.1.2),主要由一下部分组成:        

·  net.sf.cglib.core
Low-level bytecode manipulation classes; Most of them are related to ASM.

·  net.sf.cglib.transform
Classes for class file transformations at runtime or build time

·  net.sf.cglib.proxy
Classes for proxy creation and method interceptions

·  net.sf.cglib.reflect
Classes for a faster reflection and C#-style delegates

·  net.sf.cglib.util
Collection sorting utilities

·  net.sf.cglib.beans
JavaBean related utilities

大多时候,仅仅为了动态地创建代理,你仅需要使用到代理包中很少的一些API。

正如我们先前所讨论的,CGLIB包是在ASM之上的一个高级别的层。对代理那些没有实现接口的类非常有用。本质上,它是通过动态的生成一个子类去覆盖所要代理类的不是final的方法,并设置好callback,则原有类的每个方法调用就会转变成调用用户定义的拦截方法(interceptors),这比JDK动态代理方法快多了。

·  Figure 2: CGLIB library APIs commonly used for proxying classes

图2为我们演示了创建一个具体类的代理时,通常要用到的CGLIB包的APIs。net.sf.cglib.proxy.Callback接口在CGLIB包中是一个很关键的接口,所有被net.sf.cglib.proxy.Enhancer类调用的回调(callback)接口都要继承这个接口。net.sf.cglib.pr用oxy.MethodInterceptor接口是最通用的回调(callback)类型,它经常被基于代理的AOP用来实现拦截(intercept)方法的调用。这个接口只定义了一个方法

    public Object intercept(Object object, java.lang.reflect.Method method,
        Object[] args, MethodProxy proxy) throws Throwable;

(原文请参照http://www.ociweb.com/jnb/jnbNov2005.html

 

 

     当net.sf.cglib.proxy.MethodInterceptor做为所有代理方法的回调(callback)时,当对基于代理的方法调用时,在调用原对象的方法的之前会调用这个方法,如图3所示。第一个参数是代理对像,第二和第三个参数分别是拦截的方法和方法的参数。原来的方法可能通过使用java.lang.reflect.Method对象的一般反射调用,或者使用net.sf.cglib.proxy.MethodProxy对象调用。net.sf.cglib.proxy.MethodProxy通常被首选使用,因为它更快。在这个方法中,我们可以在调用原方法之前或之后注入自己的代码。
 
Figure 3: CGLIB MethodInterceptor 

    net.sf.cglib.proxy.MethodInterceptor能够满足任何的拦截(interception )需要,当对有些情况下可能过度。为了简化和提高性能,CGLIB包提供了一些专门的回调(callback)类型。例如:

·  net.sf.cglib.proxy.FixedValue 
为提高性能,FixedValue回调对强制某一特别方法返回固定值是有用的。

·  net.sf.cglib.proxy.NoOp 
NoOp回调把对方法调用直接委派到这个方法在父类中的实现。

·  net.sf.cglib.proxy.LazyLoader 
当实际的对象需要延迟装载时,可以使用LazyLoader回调。一旦实际对象被装载,它将被每一个调用代理对象的方法使用。

·  net.sf.cglib.proxy.Dispatcher 
Dispathcer回调和LazyLoader回调有相同的特点,不同的是,当代理方法被调用时,装载对象的方法也总要被调用。

·  net.sf.cglib.proxy.ProxyRefDispatcher 
ProxyRefDispatcher回调和Dispatcher一样,不同的是,它可以把代理对象作为装载对象方法的一个参数传递。

  如图3所示,代理类的所以方法经常会用到回调(callback),当是你也可以使用net.sf.cglib.proxy.CallbackFilter 有选择的对一些方法使用回调(callback),这种考虑周详的控制特性在JDK的动态代理中是没有的。在JDK代理中,对java.lang.reflect.InvocationHandler方法的调用对代理类的所以方法都有效。

除了代理类外,CGLIB通过提供一个对java.lang.reflect.Proxy的drop-in替代来实现对对接口的代理。因为这种代理能力的替代很少被用到,因此相应的APIs也很少提到。

CGLIB的代理包也对net.sf.cglib.proxy.Mixin提供支持。基本上,它允许多个对象被绑定到一个单个的大对象。在代理中对方法的调用委托到下面相应的对象中。

接下来我们看看如何使用CGLIB代理APIs创建代理。

创建一个简单的代理
  CGLIB代理最核心的是net.sf.cglib.proxy.Enhancer类,为了创建一个代理,最起码你要用到这个类。首先,让我们使用NoOp回调创建一个代理:
    /**
     * Create a proxy using NoOp callback. The target class
     * must have a default zero-argument constructor.
     *
     * @param targetClass the super class of the proxy
     * @return a new proxy for a target class instance
     */
    public Object createProxy(Class targetClass) {
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(targetClass);
        enhancer.setCallback(NoOp.INSTANCE);
        return enhancer.create();
   }
返回值是target类一个实例的代理。在这个例子中,我们为net.sf.cglib.proxy.Enhancer 配置了一个单一的回调(callback)。我们可以看到很少直接创建一个简单的代理,而是创建一个net.sf.cglib.proxy.Enhancer的实例,在net.sf.cglib.proxy.Enhancer类中你可使用静态帮助方法创建一个简单的代理。一般推荐使用上面例子的方法创建代理,因为它允许你通过配置net.sf.cglib.proxy.Enhancer实例很好的控制代理的创建。

要注意的是,target类是作为产生的代理的父类传进来的。不同于JDK的动态代理,它不能在创建代理时传target对象,target对象必须被CGLIB包来创建。在这个例子中,默认的无参数构造器时用来创建target实例的。如果你想用CGLIB来创建有参数的实例,用net.sf.cglib.proxy.Enhancer.create(Class[], Object[])方法替代net.sf.cglib.proxy.Enhancer.create()就可以了。方法中第一个参数定义了参数的类型,第二个是参数的值。在参数中,基本类型应被转化成类的类型。

Use a MethodInterceptor
  为了更好的使用代理,我们可以使用自己定义的MethodInterceptor类型回调(callback)来代替net.sf.cglib.proxy.NoOp回调。当对代理中所有方法的调用时,都会转向MethodInterceptor类型的拦截(intercept)方法,在拦截方法中再调用底层对象相应的方法。
  下面我们举个例子,假设你想对目标对象的所有方法调用进行权限的检查,如果没有经过授权,就抛出一个运行时的异常AuthorizationException。其中AuthorizationService.java接口的代码如下:

    package com.lizjason.cglibproxy;

    import java.lang.reflect.Method;

    /**
     *  A simple authorization service for illustration purpose.
     *
     * @author Jason Zhicheng Li (jason@lizjason.com)
     */
    public interface AuthorizationService {
        /**
         * Authorization check for a method call. An AuthorizationException
         * will be thrown if the check fails.
         */
        void authorize(Method method);
    }

对net.sf.cglib.proxy.MethodInterceptor接口的实现的类AuthorizationInterceptor.java代码如下:

    package com.lizjason.cglibproxy.impl;

 

    import java.lang.reflect.Method;

    import net.sf.cglib.proxy.MethodInterceptor;

    import net.sf.cglib.proxy.MethodProxy;

    import com.lizjason.cglibproxy.AuthorizationService;

 

    /**

     * A simple MethodInterceptor implementation to

     * apply authorization checks for proxy method calls.

     *

     * @author Jason Zhicheng Li (jason@lizjason.com)

     *

     */

    public class AuthorizationInterceptor implements MethodInterceptor {

        private AuthorizationService authorizationService;

 

        /**

         * Create a AuthorizationInterceptor with the given

         * AuthorizationService

         */

        public AuthorizationInterceptor (AuthorizationService authorizationService) {

            this.authorizationService = authorizationService;

        }

 

        /**

         * Intercept the proxy method invocations to inject authorization check.

         * The original method is invoked through MethodProxy.

         * @param object the proxy object

         * @param method intercepted Method

         * @param args arguments of the method

         * @param proxy the proxy used to invoke the original method

         * @throws Throwable any exception may be thrown; if so, super method will not be invoked

         * @return any value compatible with the signature of the proxied method.

         */

        public Object intercept(Object object, Method method, Object[] args, MethodProxy methodProxy ) throws Throwable {

            if (authorizationService != null) {

                //may throw an AuthorizationException if authorization failed

                authorizationService.authorize(method);

            }

            return methodProxy.invokeSuper(object, args);

        }

    }

  
  我们可以看到在拦截方法中,首先进行权限的检查,如果通过权限的检查,拦截方法再调用目标对象的原始方法。由于性能的原因,
对原始方法的调用我们使用CGLIB的net.sf.cglib.proxy.MethodProxy对象,而不是反射中一般使用
java.lang.reflect.Method对象。


Use a CallbackFilter

net.sf.cglib.proxy.CallbackFilter允许我们在方法层设置回调(callback)。假如你有一个PersistenceServiceImpl类,
它有两个方法:save和load,其中方法save需要权限检查,而方法load不需要权限检查。



package com.lizjason.cglibproxy.impl;

 

    import com.lizjason.cglibproxy.PersistenceService;

 

    /**

     * A simple implementation of PersistenceService interface

     *

     * @author Jason Zhicheng Li (jason@lizjason.com)

     */

    public class PersistenceServiceImpl implements PersistenceService {

 

        public void save(long id, String data) {

            System.out.println(data + " has been saved successfully.");

        }

 

        public String load(long id) {

            return "Jason Zhicheng Li";

        }

    }

    
注意到PersistenceServiceImpl类实现了PersistenceService 接口,因此没有要求要使用CGLIB创建代理。
net.sf.cglib.proxy.CallbackFilter 接口的实现如下:

    package com.lizjason.cglibproxy.impl;

 

    import java.lang.reflect.Method;

    import net.sf.cglib.proxy.CallbackFilter;

 

    /**

     * An implementation of CallbackFilter for PersistenceServiceImpl

     *

     * @author Jason Zhicheng Li (jason@lizjason.com)

     */

    public class PersistenceServiceCallbackFilter implements CallbackFilter {

 

        //callback index for save method

        private static final int SAVE = 0;

 

        //callback index for load method

        private static final int LOAD = 1;

 

        /**

         * Specify which callback to use for the method being invoked.

         * @method the method being invoked.

         * @return the callback index in the callback array for this method

         */

        public int accept(Method method) {

            String name = method.getName();

            if ("save".equals(name)) {

                return SAVE;

            }

            // for other methods, including the load method, use the

            // second callback

            return LOAD;

        }

    }


accept方法中对代理方法和回调进行了匹配,返回的值是某方法在回调数组中的索引。下面是PersistenceServiceImpl类代理的
实现。


       ...

        Enhancer enhancer = new Enhancer();

        enhancer.setSuperclass(PersistenceServiceImpl.class);

 

        CallbackFilter callbackFilter = new PersistenceServiceCallbackFilter();

        enhancer.setCallbackFilter(callbackFilter);

 

        AuthorizationService authorizationService = ...

        Callback saveCallback = new AuthorizationInterceptor(authorizationService);

        Callback loadCallback = NoOp.INSTANCE;

        Callback[] callbacks = new Callback[]{saveCallback, loadCallback };

        enhancer.setCallbacks(callbacks);

        ...

        return (PersistenceServiceImpl)enhancer.create();




在这个例子中save方法使用了AuthorizationInterceptor实例,load方法使用了NoOp实例。此外,你也可以通过
net.sf.cglib.proxy.Enhancer.setInterfaces(Class[])方法指定代理对象所实现的接口。

  除了为net.sf.cglib.proxy.Enhancer指定回调数组,你还可以通过net.sf.cglib.proxy.Enhancer.setCallbackTypes(Class[]) 方法指定回调类型数组。当创建代理时,如果你没有回调实例的数组,就可以使用回调类型。象使用回调一样,你必须使用,net.sf.cglib.proxy.CallbackFilter
为每一个方法指定一个回调类型索引。你可以从http://www.lizjason.com/downloads/下载设置回调类型和接口的完整代码。

  小结     

    GLIB是一个功能强大,高性能的代码生成包。它为没有实现接口的类提供代理,为JDK的动态代理提供了很好的补充。
它底层使用字节码处理框架ASM。其原理是,生产一个要代理类的子类,子类覆盖要代理的类的所有不是final的方法。
 它比使用java反射的JDK动态代理要快。通常情况下,你可以使用JDK的动态代理创建代理,当你要代理的类没有实现接口
 或者为了更好的性能,CGLIB是一个好的选择。

分享到:
评论

相关推荐

    cglib.jar下载

    CGLIB介绍与原理(部分节选自网络) 一、什么是CGLIB? CGLIB是一个功能强大,高性能的代码生成包。它为没有实现接口的类提供代理,为JDK的动态代理提供了很好的补充。通常可以使用Java的动态代理创建代理,但当要...

    cglib动态代理介绍

    CGlib是Java开发中一个非常重要的库,它是一个强大的、高性能的代码生成库,主要用于创建子类,从而实现动态代理。在Java中,动态代理通常用于AOP(面向切面编程)或者为已有接口提供额外的功能,如日志、事务管理等...

    cglib代理模式要使用的相关jar包

    本文将重点介绍使用CGLIB库实现的动态代理。 CGLIB(Code Generation Library)是一个强大的高性能的代码生成库,它在许多AOP(面向切面编程)框架中被广泛应用,如Spring AOP。CGLIB通过字节码技术生成代理类,...

    转载:cglib动态代理介绍(一)

    CGlib是Java编程语言中的一个库,主要用于创建子类,也称为子类代理或动态代理。这个库在很多场合被广泛使用,特别是在那些需要在运行时动态创建对象或增强已有对象功能的场景下,比如Spring AOP(面向切面编程)...

    cglib代理的依赖jar包

    1. **原理介绍** CGlib代理机制主要是通过动态创建一个目标类的子类,并重写其方法来实现代理。当调用代理对象的方法时,实际上是调用了子类的方法,从而可以在不修改原有代码的情况下对目标类的方法进行增强。 2....

    hibernate-cglib-repack-2.1_3.jar.zip

    《深入理解Hibernate-CGLIB-2.1_3在Java开发中的应用》 在Java开发领域,Hibernate作为一款强大的对象关系映射(ORM)框架,极大地简化了数据库操作。而CGLIB则是一个代码生成库,它允许开发者在运行时动态创建子类...

    cglib(code generation library)

    code generation library

    AOP、CGLIB

    **AOP(面向切面编程)** 面向切面编程(Aspect Oriented...以上就是关于"AOP、CGLIB"的知识点介绍,Spring框架中的AOP和CGLIB结合使用,为Java开发者提供了强大的代码组织和管理工具,让软件开发变得更加高效和灵活。

    Java JDK代理、CGLIB、AspectJ代理分析比较

    接下来将详细介绍三种常用的Java代理技术:JDK代理、CGLIB代理以及AspectJ代理,并对比它们各自的优缺点。 #### 二、静态代理实例 静态代理可以通过下面的例子进行说明: ```java public interface Calculator { ...

    spring-cglib-repack-3.1.jar,spring-objenesis-repack-2.1.jar

    本文将详细介绍这两个jar包以及它们与Spring框架的关系。 首先,"spring-cglib-repack-3.1.jar"是一个针对CGLIB(Code Generation Library)的打包版本,用于在运行时动态创建Java类的子类。CGLIB是一个强大的代码...

    Spring源码缺失的spring-cglib-repack-3.2.6.jar和spring-objenesis-repack-2.6.jar

    本文将详细介绍这两个库以及它们在Spring框架中的作用。 **CGlib(Code Generation Library)** CGlib是一个强大的、高性能的代码生成库,主要用于在运行期扩展Java类与实现Java接口。在Spring框架中,CGlib主要被...

    cglib-2.2.jar

    5. **API介绍**: - `Enhancer`:这是CGLIB的核心类,用于创建代理对象。通过设置`setSuperclass`指定要增强的类,`setCallback`指定回调函数,然后调用`create`方法即可生成代理对象。 - `MethodInterceptor`:这...

    SpringMVC3.0-Jar全量包含cglib几个包-aop的aspectj几个包

    下面将详细介绍这些关键组件及其在SpringMVC 3.0中的作用。 1. **SpringMVC**:Spring MVC是Spring框架的一个模块,用于构建基于模型-视图-控制器(MVC)架构的Web应用。它提供了一个灵活的处理模型,允许开发者将...

    cglib.zip

    下面将详细介绍CGLib的相关知识点。 1. **动态代理**: CGLib常用于为Java对象创建代理,以在方法调用前后添加额外的行为。比如,在Spring AOP框架中,CGLib被用来创建代理对象,实现在方法执行前后的拦截器逻辑,...

    CGlib动态代理

    下面将详细介绍CGlib动态代理的工作原理、使用方法以及相关知识点。 **1. 动态代理概念** 动态代理是一种在运行时创建代理对象的技术,它可以让我们在不修改原有代码的基础上,为已有对象添加额外的功能。Java中的...

    CglibProxy

    - CGLIB库的介绍和使用场景 - ASM库的作用和工作原理 - Enhancer类的配置与使用 - MethodInterceptor接口和intercept()方法的实现 - CGLIB代理对象的创建和调用过程 - 对比CGLIB和Java动态代理的优缺点 - 实战示例,...

    cglib代理资料类,静态代理动态代理

    以上就是关于静态代理、动态代理(JDK)和CGlib代理的基本介绍。它们在Java开发中都有广泛的应用,尤其是在AOP(面向切面编程)和框架设计中,如Spring AOP就使用了这两种代理技术。了解并熟练掌握这些代理机制,...

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

    本文主要介绍 Java 中两种常见的动态代理方式:JDK 原生动态代理和 CGLIB 动态代理。 一、 代理模式 代理模式是指程序通过代理类来访问目标对象,以达到对目标对象的控制和增强。代理模式的优点是可以在不改变目标...

Global site tag (gtag.js) - Google Analytics