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;
    }
}
AI 代码解读

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);
}
AI 代码解读

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";
    }

}
AI 代码解读

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;
    }

}
AI 代码解读

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");
    }
}
AI 代码解读

打印结果:


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");
    }
}

AI 代码解读

打印结果:


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";
    }
}
AI 代码解读

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;
    }
}
AI 代码解读

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");
    }
}
AI 代码解读

4). 打印结果


img_c90cdb49528869f0617e09a4185d4fa5.png
图3.png
目录
打赏
0
0
0
0
69
分享
相关文章
为什么虚拟线程将改变Java并发编程?
为什么虚拟线程将改变Java并发编程?
218 83
2025 年最新 Java 学习路线图含实操指南助你高效入门 Java 编程掌握核心技能
2025年最新Java学习路线图,涵盖基础环境搭建、核心特性(如密封类、虚拟线程)、模块化开发、响应式编程、主流框架(Spring Boot 3、Spring Security 6)、数据库操作(JPA + Hibernate 6)及微服务实战,助你掌握企业级开发技能。
44 3
|
14天前
|
Java编程:理解while循环的使用
总结而言, 使用 while 迴圈可以有效解决需要多次重复操作直至特定條件被触发才停止執行任务场景下问题; 它简单、灵活、易于实现各种逻辑控制需求但同时也要注意防止因邏各错误导致無限迁璇発生及及時處理可能発生异常以确保程序稳定运作。
87 0
Java:历久弥新的企业级编程基石
Java:历久弥新的企业级编程基石
Java编程的知识体系 | Java编程精要
Java是一种广泛使用的通用编程语言,具备面向对象、跨平台、安全简单等优势,适用于桌面、企业、Web、移动及大数据等多个领域。它功能强大且易于学习,是程序设计入门和面向对象思想学习的优选语言。本书系统讲解Java编程知识,涵盖技术核心与应用拓展两大模块,内容包括基础语法、面向对象设计、GUI、数据库、多线程、网络编程及Web开发等,帮助读者全面掌握Java开发技能。
52 0
|
1月前
|
Java编程探究:深入解析final关键字
1. **使用限制**: 对于 `final` 方法和类,可以限制其他开发人员对代码的使用,确保其按设计的方式工作而不会被子类意外改变。
59 0
深入讲解 Java 并发编程核心原理与应用案例
本教程全面讲解Java并发编程,涵盖并发基础、线程安全、同步机制、并发工具类、线程池及实际应用案例,助你掌握多线程开发核心技术,提升程序性能与响应能力。
67 0
2025 年 Java 秋招面试必看 Java 并发编程面试题实操篇
Java并发编程是Java技术栈中非常重要的一部分,也是面试中的高频考点。本文从基础概念、关键机制、工具类、高级技术等多个方面进行了介绍,并提供了丰富的实操示例。希望通过本文的学习,你能够掌握Java并发编程的核心知识,在面试中取得好成绩。同时,在实际工作中,也能够运用这些知识设计和实现高效、稳定的并发系统。
77 0
AI助理
登录插画

登录以查看您的控制台资源

管理云资源
状态一览
快捷访问

你好,我是AI助理

可以解答问题、推荐解决方案等