手动实现aop使用的动态代理和cglib代理

简介: spring的aop使用的是动态代理和cglib代理,在对象有实现接口的情况下使用动态代理,没有实现接口的情况下使用cglib代理。cglib代理是继承目标对象来创建代理,所以目标对象不能使用final修饰。

spring的aop使用的是动态代理和cglib代理,在对象有实现接口的情况下使用动态代理,没有实现接口的情况下使用cglib代理。

cglib代理是继承目标对象来创建代理,所以目标对象不能使用final修饰。目标对象:

public class TestServiceImpl implements TestService {
  public void save() {
    System.out.println("保存");
  }
  public void update() {
    System.out.println("更改");
  }
  public void delete() {
    System.out.println("删除");
  }
  public void select() {
    System.out.println("查找");
  }
}

动态代理:

package com.vhukze.proxy;
import java.lang.reflect.Method;
import org.springframework.cglib.proxy.InvocationHandler;
import org.springframework.cglib.proxy.Proxy;
import com.vhukze.service.TestService;
import com.vhukze.service.serviceimpl.TestServiceImpl;
/**
* @author zsz
* @version 
* @创建时间:2019年9月18日 上午9:09:17
*/
public class TestServiceProxyFactory {
  private TestServiceImpl service;
  public TestServiceProxyFactory(TestService service) {
    this.service = (TestServiceImpl) service;
  }
  public TestService getTestServiceProxy() {
    //生成动态代理              第一个参数是一个类加载器                        目标对象实现的接口的数组        
    TestService serviceProxy = (TestService) Proxy.newProxyInstance(TestServiceProxyFactory.class.getClassLoader(), TestServiceImpl.class.getInterfaces(), 
                new InvocationHandler() {
                  public Object invoke(Object arg0, Method method, Object[] arg2) throws Throwable {
                    System.out.println("开启事务");
                    //参数一是当前执行此方法的对象 参数二是此方法的参数
                    Object invoke = method.invoke(service, arg2);
                    System.out.println("提交事务");
                    return invoke;
                  }
                });
    return serviceProxy;
  }
}

动态代理:

package com.vhukze.proxy;
import java.lang.reflect.Method;
import org.springframework.cglib.proxy.InvocationHandler;
import org.springframework.cglib.proxy.Proxy;
import com.vhukze.service.TestService;
import com.vhukze.service.serviceimpl.TestServiceImpl;
/**
* @author zsz
* @version 
* @创建时间:2019年9月18日 上午9:09:17
*/
public class TestServiceProxyFactory {
  private TestServiceImpl service;
  public TestServiceProxyFactory(TestService service) {
    this.service = (TestServiceImpl) service;
  }
  public TestService getTestServiceProxy() {
    //生成动态代理              第一个参数是一个类加载器                        目标对象实现的接口的数组        
    TestService serviceProxy = (TestService) Proxy.newProxyInstance(TestServiceProxyFactory.class.getClassLoader(), TestServiceImpl.class.getInterfaces(), 
                new InvocationHandler() {
                  public Object invoke(Object arg0, Method method, Object[] arg2) throws Throwable {
                    System.out.println("开启事务");
                    //参数一是当前执行此方法的对象 参数二是此方法的参数
                    Object invoke = method.invoke(service, arg2);
                    System.out.println("提交事务");
                    return invoke;
                  }
                });
    return serviceProxy;
  }
}

测试方法:

1./**
* @author zsz
* @编辑时间 2019年9月18日上午9:44:03
* @编辑说明  动态代理测试
*/
@Test
public void fun1() {
  TestServiceProxyFactory proxy = new TestServiceProxyFactory(new TestServiceImpl());
  TestService serviceProxy = proxy.getTestServiceProxy();
  serviceProxy.select();
}

结果:

cglib代理:

package com.vhukze.proxy;
import java.lang.reflect.Method;
import org.springframework.cglib.proxy.Enhancer;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;
import com.vhukze.service.TestService;
import com.vhukze.service.serviceimpl.TestServiceImpl;
/**
* @author zsz
* @version 
* @创建时间:2019年9月18日 上午9:09:17
*/
public class TestServiceProxyFactory2{
  public TestService getTestServiceProxy(){
    //帮我们生成代理对象
    Enhancer en = new Enhancer();
    //设置对谁代理
    en.setSuperclass(TestServiceImpl.class);
    //设置代理要做什么
    en.setCallback(new MethodInterceptor() {
      public Object intercept(Object proxyObj, Method method, Object[] arg, MethodProxy methodProxy) throws Throwable {
        System.out.println("开启事务");
        Object returnValue = methodProxy.invokeSuper(proxyObj, arg);
        System.out.println("提交事务");
        return returnValue;
      }
    });
    return (TestService) en.create();
  }
}

结果:

相关文章
|
10月前
|
设计模式 Java 开发者
如何快速上手【Spring AOP】?从动态代理到源码剖析(下篇)
Spring AOP的实现本质上依赖于代理模式这一经典设计模式。代理模式通过引入代理对象作为目标对象的中间层,实现了对目标对象访问的控制与增强,其核心价值在于解耦核心业务逻辑与横切关注点。在框架设计中,这种模式广泛用于实现功能扩展(如远程调用、延迟加载)、行为拦截(如权限校验、异常处理)等场景,为系统提供了更高的灵活性和可维护性。
1378 0
|
XML 安全 Java
Spring AOP—深入动态代理 万字详解(通俗易懂)
Spring 第四节 AOP——动态代理 万字详解!
617 24
|
存储 缓存 Java
Spring高手之路23——AOP触发机制与代理逻辑的执行
本篇文章深入解析了Spring AOP代理的触发机制和执行流程,从源码角度详细讲解了Bean如何被AOP代理,包括代理对象的创建、配置与执行逻辑,帮助读者全面掌握Spring AOP的核心技术。
448 3
Spring高手之路23——AOP触发机制与代理逻辑的执行
|
Java 开发者 Spring
Spring AOP深度解析:探秘动态代理与增强逻辑
Spring框架中的AOP(Aspect-Oriented Programming,面向切面编程)功能为开发者提供了一种强大的工具,用以将横切关注点(如日志、事务管理等)与业务逻辑分离。本文将深入探讨Spring AOP的底层原理,包括动态代理机制和增强逻辑的实现。
383 4
|
设计模式 Java 测试技术
spring复习04,静态代理动态代理,AOP
这篇文章讲解了Java代理模式的相关知识,包括静态代理和动态代理(JDK动态代理和CGLIB),以及AOP(面向切面编程)的概念和在Spring框架中的应用。文章还提供了详细的示例代码,演示了如何使用Spring AOP进行方法增强和代理对象的创建。
spring复习04,静态代理动态代理,AOP
|
安全 Java 开发者
AOP中的JDK动态代理与CGLIB动态代理:深度解析与实战模拟
【11月更文挑战第21天】面向切面编程(AOP,Aspect-Oriented Programming)是一种编程范式,它通过将横切关注点(cross-cutting concerns)与业务逻辑分离,以提高代码的可维护性和可重用性。在Java开发中,AOP的实现离不开动态代理技术,其中JDK动态代理和CGLIB动态代理是两种常用的方式。本文将从背景、历史、功能点、业务场景、底层逻辑等多个维度,深度解析这两种代理方式的区别,并通过Java示例进行模拟和比较。
1083 5
|
缓存 安全 Java
Spring AOP 中两种代理类型的限制
【8月更文挑战第22天】
405 0
|
9月前
|
XML 安全 Java
使用 Spring 的 @Aspect 和 @Pointcut 注解简化面向方面的编程 (AOP)
面向方面编程(AOP)通过分离横切关注点,如日志、安全和事务,提升代码模块化与可维护性。Spring 提供了对 AOP 的强大支持,核心注解 `@Aspect` 和 `@Pointcut` 使得定义切面与切入点变得简洁直观。`@Aspect` 标记切面类,集中处理通用逻辑;`@Pointcut` 则通过表达式定义通知的应用位置,提高代码可读性与复用性。二者结合,使开发者能清晰划分业务逻辑与辅助功能,简化维护并提升系统灵活性。Spring AOP 借助代理机制实现运行时织入,与 Spring 容器无缝集成,支持依赖注入与声明式配置,是构建清晰、高内聚应用的理想选择。
817 0
|
8月前
|
监控 Java Spring
AOP 切面编程
AOP(面向切面编程)通过动态代理在不修改源码的前提下,对方法进行增强。核心概念包括连接点、通知、切入点、切面和目标对象。常用于日志记录、权限校验、性能监控等场景,结合Spring AOP与@Aspect、@Pointcut等注解,实现灵活的横切逻辑管理。
1937 6
AOP 切面编程
|
8月前
|
XML Java 数据格式
《深入理解Spring》:AOP面向切面编程深度解析
Spring AOP通过代理模式实现面向切面编程,将日志、事务等横切关注点与业务逻辑分离。支持注解、XML和编程式配置,提供五种通知类型及丰富切点表达式,助力构建高内聚、低耦合的可维护系统。