Java AOP(面向切面编程)实现

简介: 动态代理AOP概念解释AOP用在哪些方面:AOP能够将那些与业务无关,却为业务模块所共同调用的逻辑或责任,例如事务处理、日志管理、权限控制,异常处理等,封装起来,便于减少系统的重复代码,降低模块间的耦合度,并有利于未来的可操作性和可维护性。
动态代理

AOP概念解释

AOP用在哪些方面:AOP能够将那些与业务无关,却为业务模块所共同调用的逻辑或责任,例如事务处理、日志管理、权限控制,异常处理等,封装起来,便于减少系统的重复代码,降低模块间的耦合度,并有利于未来的可操作性和可维护性。

AOP中的概念

Aspect(切面):指横切性关注点的抽象即为切面,它与类相似,只是两者的关注点不一样,类是对物体特征的抽象,而切面是横切性关注点的抽象。
joinpoint(连接点):所谓连接点是指那些被拦截到的点(可以是方法、属性、或者类的初始化时机(可以是Action层、Service层、dao层))。在Spring中,这些点指的是方法,因为Spring只支持方法类型的连接点,实际上joinpoint还可以是field或类构造器。
Pointcut(切入点):所谓切入点是指我们要对那些joinpoint进行拦截的定义,也即joinpoint的集合。
Advice(通知):所谓通知是指拦截到joinpoint之后所要做的事情就是通知。通知分为前置通知、后置通知、异常通知、最终通知、环绕通知。我们就以CGlibProxyFactory类的代码为例进行说明:

public class CGlibProxyFactory implements MethodInterceptor {
    private Object targetObject; // 代理的目标对象

    public Object createProxyInstance(Object targetObject) {
        this.targetObject = targetObject; 

        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(this.targetObject.getClass()); // 设置目标类为代理对象的父类
        enhancer.setCallback(this);

        return enhancer.create();
    }

    // 从另一种角度看: 整个方法可看作环绕通知
    @Override
    public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
        PersonServiceBean bean = (PersonServiceBean)this.targetObject;
        Object result = null; 
        if (bean.getUser() != null) { // 有权限
            // ...... advice() ----> 前置通知(所谓通知,就是我们拦截到业务方法之后所要干的事情)
            try {
                result = methodProxy.invoke(targetObject, args); // 把方法调用委派给目标对象
                // ...... afteradvice() ----> 后置通知
            } catch (RuntimeException e) {
                // ...... exceptionadvice() ----> 异常通知
            } finally {
                // ...... finallyadvice() ----> 最终通知
            }
        }
        return result;
    }
}

Target(目标对象):代理的目标对象。
Weave(织入):指将aspects应用到target对象并导致proxy对象创建的过程称为织入。
Introduction(引入):在不修改类代码的前提下,Introduction可以在运行期为类(代理类)动态地添加一些方法或Field。
AOP带来的好处:降低模块的耦合度;使系统容易扩展;更好的代码复用性。

2. JDK实现

1). 创建Person接口

public interface PersonService {
    /**
     * 保存
     * @param name 名称
     */
    public void save(String name);
    /**
     * 根据ID更新名称
     * @param name 姓名
     * @param personId 人员ID
     */
    public void update(String name, Integer personId);
    /**
     * 根据ID获取名称
     * @param personId 人员ID
     * @return 名称
     */
    public String getPersonName(Integer personId);
}

2). 创建实现Person接口的实现类PersonImpl


public class PersonServiceImpl implements PersonService{
    private String user = null;
    
    public void setUser(String user) {
        this.user = user;
    }
    
    public String getUser() {
        return user;
    }
    
    public PersonServiceImpl() {    }
    
    public PersonServiceImpl(String user){
        this.user = user;
    }
    
    @Override
    public void save(String name) {
        System.out.println("我是save方法");
    }

    @Override
    public void update(String name, Integer personId) {
        System.out.println("我是update方法");
    }

    @Override
    public String getPersonName(Integer personId) {
        System.out.println("我是getPersonName方法");
        return "xxx";
    }

}

3). 创建代理类PersonServiceImplProxy

public class PersonServiceImplProxy implements InvocationHandler {
    private PersonService personService;
    
    public PersonService createProxy(PersonService personService) {
        return (PersonService) Proxy.newProxyInstance(PersonServiceImplProxy.class.getClassLoader(), personService.getClass().getClass().getInterfaces(), this);
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        PersonServiceImpl pImpl = (PersonServiceImpl) this.personService;
        Object result = null;
        // 如果不等于null则表示有权限
        if (null != pImpl.getUser()) {
            // 执行方法
            result = method.invoke(personService, args);
        }
        return result;
    }

}

4). 创建Demo类测试

public class Demo {
    public static void main(String[] args) {
        PersonServiceImplProxy proxy = new PersonServiceImplProxy();
        PersonService service = proxy.createProxy(new PersonServiceImpl("mazaiting"));
        service.save("123");
    }
}

打印结果:


img_ca0e299488614528fc451f48b4161d2a.png
图1.png

5). 更改测试代码

public class Demo {
    public static void main(String[] args) {
        PersonServiceImplProxy proxy = new PersonServiceImplProxy();
        PersonService service = proxy.createProxy(new PersonServiceImpl());
        service.save("123");
    }
}

打印结果:


img_c421a47be207ed9702e326055ebbef75.png
图2.png

3. CGlib实现AOP功能

动态代理技术只能是基于接口,那如果这个对象没有接口,就要使用CGlib这个工具包。

1). 创建PersonService类

public class PersonService {
    private String user = null;
    
    public void setUser(String user) {
        this.user = user;
    }
    
    public String getUser() {
        return user;
    }
    
    public PersonService() {    }
    
    public PersonService(String user){
        this.user = user;
    }
    
    public void save(String name) {
        System.out.println("我是save方法");
    }

    public void update(String name, Integer personId) {
        System.out.println("我是update方法");
    }

    public String getPersonName(Integer personId) {
        System.out.println("我是getPersonName方法");
        return "xxx";
    }
}

2). 创建CGlibProxyFactory类

public class CGlibProxyFactory implements MethodInterceptor {
    // 代理的目标对象
    private Object targetObject;
    
    public Object createProxyInstance(Object targetObject) {
        this.targetObject = targetObject;
        
        // 该类用于生成代理对象
        Enhancer enhancer = new Enhancer();
        // 设置目标类为代理对象的父类
        enhancer.setSuperclass(this.targetObject.getClass());
        // 设置回调用对象本身
        enhancer.setCallback(this);
        return enhancer.create();
    }

    @Override
    public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
        PersonService service = (PersonService) this.targetObject;
        Object result = null;
        // 有权限
        if (null != service.getUser()) {
            // 把方法调用委派给目标对象
            result = methodProxy.invoke(targetObject, args);
        }
        return result;
    }
}

3). 创建Demo测试类

public class Demo {
    public static void main(String[] args) {
        CGlibProxyFactory factory = new CGlibProxyFactory();
        PersonService service = (PersonService) factory.createProxyInstance(new PersonService("mazaiting"));
        service.save("999");
    }
}

4). 打印结果


img_c90cdb49528869f0617e09a4185d4fa5.png
图3.png
目录
相关文章
|
2天前
|
XML 监控 安全
Spring特性之一——AOP面向切面编程
Spring特性之一——AOP面向切面编程
15 1
|
1天前
|
存储 关系型数据库 MySQL
《MySQL 入门教程》第 05 篇 账户和权限,Java高并发编程详解深入理解pdf
《MySQL 入门教程》第 05 篇 账户和权限,Java高并发编程详解深入理解pdf
|
1天前
|
Java
java使用AOP切面获取请求日志并记录
java使用AOP切面获取请求日志并记录
|
1天前
|
NoSQL Dubbo Java
StringBoot编程式事务与声明式事务java工程师面试突击第一季
StringBoot编程式事务与声明式事务java工程师面试突击第一季
|
2天前
|
安全 Java 开发者
深入理解Java并发编程:线程安全与性能优化
【5月更文挑战第15天】本文将深入探讨Java并发编程的核心概念,包括线程安全和性能优化。我们将通过实例分析,理解线程安全的重要性,并学习如何通过各种技术和策略来实现它。同时,我们也将探讨如何在保证线程安全的同时,提高程序的性能。
|
2天前
|
前端开发 Java 开发者
【JavaEE】面向切面编程AOP是什么-Spring AOP框架的基本使用
【JavaEE】面向切面编程AOP是什么-Spring AOP框架的基本使用
8 0
|
2天前
|
Java 编译器 开发者
Java并发编程中的锁优化策略
【5月更文挑战第15天】 在Java的多线程编程中,锁机制是实现线程同步的关键。然而,不当的锁使用往往导致性能瓶颈甚至死锁。本文深入探讨了Java并发编程中针对锁的优化策略,包括锁粗化、锁消除、锁分离以及读写锁的应用。通过具体实例和性能分析,我们将展示如何有效避免竞争条件,减少锁开销,并提升应用程序的整体性能。
|
2天前
|
消息中间件 并行计算 Java
Java中的多线程编程:基础知识与实践
【5月更文挑战第15天】 在现代计算机编程中,多线程是一个复杂但必不可少的概念。特别是在Java这种广泛使用的编程语言中,理解并掌握多线程编程是每个开发者必备的技能。本文将深入探讨Java中的多线程编程,从基础概念到实际应用场景,为读者提供全面的理论支持和实践指导。
|
2天前
|
Java 程序员 调度
Java中的多线程编程:从理论到实践
【5月更文挑战第14天】在现代计算机技术中,多线程编程是一个重要的概念。它允许多个线程并行执行,从而提高程序的运行效率。本文将从理论和实践两个角度深入探讨Java中的多线程编程,包括线程的基本概念、创建和控制线程的方法,以及如何处理线程同步和通信问题。
时间轮-Java实现篇
在前面的文章《[时间轮-理论篇](https://developer.aliyun.com/article/910513)》讲了时间轮的一些理论知识,然后根据理论知识。我们自己来实现一个简单的时间轮。