Spring中的两种代理JDK和CGLIB的区别浅谈

一、原理区别:

Java动态代理是利用反射机制生成一个实现代理接口的匿名类,在调用具体方法前调用InvokeHandler来处理。

而cglib动态代理是利用asm开源包,对代理对象类的class文件加载进来,通过修改其字节码生成子类来处理。

1、如果目标对象实现了接口,默认情况下会采用JDK的动态代理实现AOP

2、如果目标对象实现了接口,可以强制使用CGLIB实现AOP

3、如果目标对象没有实现了接口,必须采用CGLIB库,spring会自动在JDK动态代理和CGLIB之间转换

如何强制使用CGLIB实现AOP?

(1)添加CGLIB库,SPRING_HOME/cglib/.jar

(2)在spring配置文件中加入<aop:aspectj-autoproxy proxy-target-class="true"/>

JDK动态代理和CGLIB字节码生成的区别?

(1)JDK动态代理只能对实现了接口的类生成代理,而不能针对类

(2)CGLIB是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的方法

因为是继承,所以该类或方法最好不要声明成final

二、代码实现

package com.fy.spring.proxy;
public interface UserManager {
  public void addUser(String id, String password);
  public void delUser(String id);
}  
package com.fy.spring.proxy;
  public class UserManagerImpl implements UserManager {  

  public void addUser(String id, String password) {
    System.out.println(".: 掉用了UserManagerImpl.addUser()方法! ");  

  }  

  public void delUser(String id) {
    System.out.println(".: 掉用了UserManagerImpl.delUser()方法! ");  

  }
}

JDK动态代理类

package com.fy.spring.proxy;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/**
 *
 * JDK动态代理类
 *
 *
 */
public class JDKProxy implements InvocationHandler {  

  private Object targetObject;//需要代理的目标对象  

  public Object newProxy(Object targetObject) {//将目标对象传入进行代理
    this.targetObject = targetObject;
    return Proxy.newProxyInstance(targetObject.getClass().getClassLoader(),
        targetObject.getClass().getInterfaces(), this);//返回代理对象
  }  

  public Object invoke(Object proxy, Method method, Object[] args)//invoke方法
      throws Throwable {
    checkPopedom();//一般我们进行逻辑处理的函数比如这个地方是模拟检查权限
    Object ret = null;   // 设置方法的返回值
    ret = method.invoke(targetObject, args);    //调用invoke方法,ret存储该方法的返回值
    return ret;
  }  

  private void checkPopedom() {//模拟检查权限的例子
    System.out.println(".:检查权限 checkPopedom()!");
  }
}

CGLibProxy动态代理类

package com.fy.spring.proxy;
import java.lang.reflect.Method;  

import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;  

/**
 * CGLibProxy动态代理类的实例
 *
 *
 */ public class CGLibProxy implements MethodInterceptor {  

  private Object targetObject;// CGLib需要代理的目标对象  

  public Object createProxyObject(Object obj) {
    this.targetObject = obj;
    Enhancer enhancer = new Enhancer();
    enhancer.setSuperclass(obj.getClass());
    enhancer.setCallback(this);
    Object proxyObj = enhancer.create();
    return proxyObj;// 返回代理对象
  }  

  public Object intercept(Object proxy, Method method, Object[] args,
      MethodProxy methodProxy) throws Throwable {
    Object obj = null;
    if ("addUser".equals(method.getName())) {// 过滤方法
      checkPopedom();// 检查权限
    }
    obj = method.invoke(targetObject, args);
    return obj;
  }  

  private void checkPopedom() {
    System.out.println(".:检查权限 checkPopedom()!");
  }
}

测试类:

public class Client {  

  public static void main(String[] args) {  

    UserManager userManager = (UserManager) new CGLibProxy()
        .createProxyObject(new UserManagerImpl());
    System.out.println("-----------CGLibProxy-------------");
    userManager.addUser("tom", "root");
    System.out.println("-----------JDKProxy-------------");
    JDKProxy jdkPrpxy = new JDKProxy();
    UserManager userManagerJDK = (UserManager) jdkPrpxy
        .newProxy(new UserManagerImpl());
    userManagerJDK.addUser("tom", "root");
  }  

}

运行结果:

-----------CGLibProxy-------------   
检查权限  checkPopedom()!   
掉用了UserManagerImpl.addUser()方法!    
-----------JDKProxy-------------   
检查权限  checkPopedom()!   
掉用了UserManagerImpl.addUser()方法!

JDK代理是不需要以来第三方的库,只要要JDK环境就可以进行代理,它有几个要求

  1. 实现InvocationHandler
  2. 使用Proxy.newProxyInstance产生代理对象
  3. 被代理的对象必须要实现接口

CGLib 必须依赖于CGLib的类库,但是它需要类来实现任何接口代理的是指定的类生成一个子类,覆盖其中的方法,是一种继承但是针对接口编程的环境下推荐使用JDK的代理

在Hibernate中的拦截器其实现考虑到不需要其他接口的条件Hibernate中的相关代理采用的是CGLib来执行。

希望本文所述对你有所帮助,Spring中的两种代理JDK和CGLIB的区别浅谈内容就给大家介绍到这里了。希望大家继续关注我们的网站!想要学习java可以继续关注本站。

(0)

相关推荐

  • spring cglib 与 jdk 动态代理

    1. 概述 JDK动态代理是利用java反射机制 生成一个实现接口的匿名类, 在调用具体方法前调用InvocationHandler来处理 Cglib动态代理是 利用asm开源包 把被代理类的class文件加载进来 通过修改其字节码生成子类来处理 如果目标对象实现了接口 那么默认使用jdk代理(可以强制使用cglib代理) 如果没有实现接口 必须使用cglib代理 强制使用cglib代理需要 *引入cglibjar包 *配置spring <aop:aspectj-autoproxy proxy-

  • Spring中的两种代理JDK和CGLIB的区别浅谈

    一.原理区别: Java动态代理是利用反射机制生成一个实现代理接口的匿名类,在调用具体方法前调用InvokeHandler来处理. 而cglib动态代理是利用asm开源包,对代理对象类的class文件加载进来,通过修改其字节码生成子类来处理. 1.如果目标对象实现了接口,默认情况下会采用JDK的动态代理实现AOP 2.如果目标对象实现了接口,可以强制使用CGLIB实现AOP 3.如果目标对象没有实现了接口,必须采用CGLIB库,spring会自动在JDK动态代理和CGLIB之间转换 如何强制使用

  • 详解Spring的两种代理方式:JDK动态代理和CGLIB动态代理

    代理模式 代理模式的英文叫做Proxy或Surrogate,中文都可译为"代理",所谓代理,就是一个人或者一个机构代表另一个人或者另一个机构采取行动.在一些情况下,一个客户不想或者不能够直接引用一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用 A. 抽象主题角色 声明了真实主题和代理主题的共同接口,这样一来在任何可以使用真实主题的地方都可以是使用代理主题 B. 代理主题(Proxy)角色: 代理主题角色内部含有对真实主题的引用,从而可以在任何时候操作真实主题对象:代理主题角

  • 关于spring aop两种代理混用的问题

    目录 spring aop两种代理混用问题 一.首先复习一下两种代理 二.我们项目是spring-boot项目 spring的aop和代理模式理解 代理模式代码的主要特点是 代理模式目前实现的方式有三种 Aop的最大意义是 spring aop两种代理混用问题 工作繁忙,但是遇到问题还是要总结积累下来,今天项目中出现了代理混用的问题,解决之后记录一下对两种代理方式的学习理解. 一.首先复习一下两种代理 JDK动态代理 和 cglib代理 1.如果目标对象实现了接口,默认情况下会采用JDK的动态代

  • Feign调用中的两种Header传参方式小结

    目录 Feign调用中的两种Header传参方式 在请求拦截器中统一配置 通过@RequestHeader注解 调用feign接口时,如何往header中添加参数 总结 Feign调用中的两种Header传参方式 在Spring Cloud Netflix栈中,各个微服务都是以HTTP接口的形式暴露自身服务的,因此在调用远程服务时就必须使用HTTP客户端. 我们可以使用JDK原生的URLConnection.Apache的Http Client.Netty的异步HTTP Client, Spri

  • 详解Java 中的三种代理模式

    代理模式 代理(Proxy)是一种设计模式,提供了对目标对象另外的访问方式;即通过代理对象访问目标对象.这样做的好处是:可以在目标对象实现的基础上,增强额外的功能操作,即扩展目标对象的功能. 这里使用到编程中的一个思想:不要随意去修改别人已经写好的代码或者方法,如果需改修改,可以通过代理的方式来扩展该方法. 举个例子来说明代理的作用:假设我们想邀请一位明星,那么并不是直接连接明星,而是联系明星的经纪人,来达到同样的目的.明星就是一个目标对象,他只要负责活动中的节目,而其他琐碎的事情就交给他的代理

  • Quartz与Spring集成的两种方法示例

    目录 Quartz与Spring集成有2种方式:1.MethodInvokeJobDetailFactoryBean2.JobDetailBean下面分别介绍这两种方式.1.MethodInvokeJobDetailFactoryBean1)创建一个Job类,此类不需要实现任何接口,不需要继承任何类 public class MyJobTest { public void execute(){ System.out.println("正在执行quartz任务的一个方法..."); }}

  • Spring  AOP的两种使用方法

    目录 前言 1 注解方式 1.1 声明目标类 UserDao 类 1.2 声明切面 AnnotationAspect 类 1.3 声明配置 1.4 测试用例 2 XML 配置方式 2.1 声明目标类 CompanyDao 2.2 声明切面拦截类 XmlAspect 2.3 声明 XML 配置 2.3 测试用例 前言 记录下 Spring AOP 的两种使用方式,不谈概念,只记录两种方式的使用例子 注解方式 xml 配置方式 1 注解方式 1.1 声明目标类 UserDao 类 @Reposito

  • 浅谈TreeSet中的两种排序方式

    直接上代码: package exercise1; public class Person implements Comparable{ private int id; private String name; public Person(int id, String name) { super(); this.id = id; this.name = name; } public int getId() { return id; } public void setId(int id) { th

  • MyBatis批量插入数据到Oracle数据库中的两种方式(实例代码)

    一.mybatis批量插入数据到Oracle中的两种方式: 第一种: <insert id="addList" parameterType="java.util.List" useGeneratedKeys="false"> INSERT ALL <foreach item="item" index="index" collection="list"> INTO

  • JS中的两种数据类型及实现引用类型的深拷贝的方法

    一.前言 我们知道,在JS中数据类型按照访问方式和存储方式的不同可分为基本类型和引用类型. 基本类型 基本类型有String.Boolean.Number,Undefined.Null,这些基本类型都是按值传递的,也称为值类型. 引用类型 引用类型有对象.数组.函数,它们都是按引用访问的. 二.存储方式区别 基本类型和引用类型由于两者在内存中存储的方式不同,造成两者访问的方式也不同.其中,基本类型存储在内存的栈中,是按值访问:引用类型存储在内存的堆中,是按引用访问.可如下图所示: 当有 var

随机推荐