java开发AOP面向切面编程入门

目录
  • 引言
  • 不好的解决方案
    • 面向过程的解决方案
    • 使用继承解决方案
    • 使用聚合的解决方案
  • 面向切面的编程基本概念
  • 基于Spring面向切面程序实现
  • 小结

引言

在实际应用场景中,我们封装一个学生的类,这个类用于封装学生的日常行为,如:上学、吃饭、上课等。然而,在疫情期间,学生上学时入校、吃饭时进入餐厅,需要测温查验证件等行为,拿到这样的需求我们怎么办?

不好的解决方案

面向过程的解决方案

遇到问题解决问题,在上学、吃饭方法中加上测温、查验证件方法,或者在学生类中提炼一个测温查验证件私有的方法,在需要时调用。

代码如下:

public class Student {
    public void toSchool(){
        check();
        System.out.println("高高兴兴去上学!");
    }
    public void toEat(){
        check();
        System.out.println("一号餐厅去吃饭!");
    }

    public void toClass(){
        System.out.println("排排坐,听讲座!");
    }

    public void getUp(){
        System.out.println("起床啦!");
    }

    private void check(){
        System.out.println("查验证件、测体温");
    }
}

这种方式存在问题只是头痛医头、脚痛医脚,代码硬拷贝,现在教师进校门也要查验证件测温,如何办?将check代码复制到教师类中,再者,如果check()中业务规则发生变化,则需要到处改代码,显然是一种非常蹩脚的解决方案。

使用继承解决方案

创建一个man的基类,将check()方法放到该类中,教师、学生均继承此类,子类调用这个方法即可,

代码如下:

public class Man {
    protected void check(){
        System.out.println("查验证件、测体温");
    }
}
public class Student extends Man {

    public void toSchool(){
        check();
        System.out.println("高高兴兴去上学!");
    }

    public void toEat(){
        check();
        System.out.println("一号餐厅去吃饭!");
    }
}

public class Teacher extends Man {
    public void toSchool(){
        check();
        System.out.println("高高兴兴去上班!");
    }

    public void toEat(){
        check();
        System.out.println("教工餐厅去吃饭!");
    }

    public void toClass(){
        System.out.println("排排坐,听讲座!");
    }
}

这种方式,虽然解决了代码复制问题,但违背了面向对象程序设计的单一职责原则,因为查验证件测温等均不是学生或教师应该拥有的职责,类的划分职责不清,增加了代码扩展维护的难度。

使用聚合的解决方案

将查验证件测温等行为封装一个独立的类,学生和教师的依然只封装他们固有的方法,

代码如下:

public class Checktor {
    public void takeTemperature(){
        System.out.println("查验证件、测体温");
    }
}
public class Student {
    private Checktor checktor = new Checktor();

    public void toSchool(){
        checktor.takeTemperature();
        System.out.println("高高兴兴去上学!");
    }

    public void toEat(){
        checktor.takeTemperature();
        System.out.println("一号餐厅去吃饭!");
    }
}

这种方法很好的解决了面向对象程序设计单一职责原则,体现了类的封装性,但是代码侵入性很强,而且代码僵化,维护性差。如疫情结束了,我们要求取消查验证件和测温,就需要改原有的代码,破坏了开闭原则,增加了程序员的工作量。有些人写配置开关变量,在调用时使用ifelse进行判断是否调用也能遵循开关变量,但是代码中包含了一些可能用不到代码,不优雅,不是很好解决方案。

面向切面的编程基本概念

面向切面的编程是不破坏原有类封装性的前提下,动态在其方法前面(before)、后边(after)及周围(Around)(前面和后边)增强功能。

  • 连接点(Joinpoint):目标对象中每个方法前面、后面均为连接点。如学生类中toSchool、toEat、toClass、getUp方法前面后面均是切入点;
  • 切入点(Point):需要增强功能方法前面或后面,是切入点,如进学校前、吃饭前需要测体温,则toSchool、toEat两个方法前是切入点,其余的不是;

  • 切面类(Aspect):也称通知类,封装了需要增强功能的类,如:Checktor;
  • 通知方法(advice):需要增强的功能,如:测温方法;
  • 通知类型:前置通知(Before)、后置通知(After)、环绕通知(around)、异常通知(throwing)、最终通知(after (finally) advice);

  • 代理(proxy):将通知应用到目标对象实现机制,是通过创建代理对象完成的
  • 织入(Weaving):将切面代码插入到目标对象上,生成代理对象的过程

基于Spring面向切面程序实现

针对学生上学场景,Spring的使用XML和注解两设计种方式完成面向切面的编程。
工程依赖项:在maven pom文件中,需要引入以下依赖项

     <dependency>
      <groupId>org.aspectj</groupId>
      <artifactId>aspectjtools</artifactId>
      <version>1.8.9</version>
    </dependency>

    <dependency>
      <groupId>org.aspectj</groupId>
      <artifactId>aspectjrt</artifactId>
      <version>1.5.4</version>
    </dependency>

    <dependency>
      <groupId>org.aspectj</groupId>
      <artifactId>aspectjweaver</artifactId>
      <version>1.9.5</version>
      <scope>runtime</scope>
    </dependency>

XML方式:
学生类:

public class Student {

    public void toSchool(){
        System.out.println("高高兴兴去上学!");
    }

    public void toEat(){
        System.out.println("一号餐厅去吃饭!");
    }

    public void toClass(){
        System.out.println("排排坐,听讲座!");
    }

    public void getUp(){
        System.out.println("起床啦!");
    }
}

切面类:

public class Checktor {
    public void takeTemperature(){
        System.out.println("测体温啦!");
    }

    public void washHand(){
        System.out.println("洗洗手更健康!");
    }
}

xml文件

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/aop https://www.springframework.org/schema/aop/spring-aop.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd">
    <context:component-scan base-package="com.bjwl.*"/>
    <!--目标(业务)对象-->
    <bean id="student" class="com.bjwl.xmltest.Student"/>
    <bean id="teacher" class="com.bjwl.xmltest.Teacher"/>
    <!--通知(切面)对象-->
    <bean id="checktor" class="com.bjwl.xmltest.Checktor"></bean>

    <aop:config>
        <!--定义(切面)对象-->
        <aop:aspect ref="checktor">
            <!--定义切点-->
            <aop:pointcut id="point_test_d" expression="execution(* com.bjwl.xmltest.Student.to*(..))"/>
            <!--定义通知-->
            <aop:before method="takeTemperature" pointcut-ref="point_test_d"></aop:before>
            <aop:after method="washHand" pointcut-ref="point_test_d"></aop:after>
        </aop:aspect>
    </aop:config>
</beans>

其中:目标对象是student对象,切点是student对象中方法名前两个字母是to的所有方法,事前执行测温(takeTemperature)、事后执行洗手(washHand)
测试代码如下:

public class StudentTest {
    @Test
    public void toSchool() {
        ApplicationContext applicationContext = new ClassPathXmlApplicationContext(
                "bean02.xml"
        );
        Student student = (Student)applicationContext.getBean("student");
        student.toSchool();
    }
 }

运行结果如下:

如果教师需也需要测温,编写教师类,代码如下:

public class Teacher {
    public void toSchool(){
        System.out.println("高高兴兴去上班!");
    }

    public void toEat(){
        System.out.println("教工餐厅去吃饭!");
    }

    public void toClass(){
        System.out.println("排排坐,听讲座!");
    }
}

xml文件更改如下:

    <aop:config>
        <!--定义(切面)对象-->
        <aop:aspect ref="checktor">
            <!--定义切点-->
            <aop:pointcut id="point_test_d" expression="execution(* com.bjwl.xmltest.*.to*(..))"/>
            <!--定义通知-->
            <aop:before method="takeTemperature" pointcut-ref="point_test_d"></aop:before>
            <aop:after method="washHand" pointcut-ref="point_test_d"></aop:after>
        </aop:aspect>
    </aop:config>

则表示com.bjwl.xmltest中的所有类的对象均为目标对象,前两个字母带to的方法前后均为切点。

注解方式:
学生类代码:

切面类的代码

xml文件:

运行结果等同,XML文件。

这样做的好处,学生类、教师类、切面类具有职责单一性,提现业务逻辑封装性,动态增强符合开闭原则。

小结

以上简单的介绍了面向切面编程的基本概念和基本用法。用好面向切面的编程技术很不易,因为切点只能设置到需要增强功能前面或后面,对程序设计人员要求很高,要求能够根据编程经验和技巧设计出合理的切点,需要提现类的层次性。如:执行SQL语句前,在控制台输出完整的SQL语句的切点,切点就不能设置到你自己定义的Dao中,而是设置在PrepareStatement的exectuce的方法前。不过也不用担心,使用各种框架时,常用的日志等,已经给我们设置了开关变量,我们直接使用即可,非常简单。

以上就是java开发AOP面向切面编程入门的详细内容,更多关于java面向切面编程的资料请关注我们其它相关文章!

(0)

相关推荐

  • MVC AOP面向切面编程简单介绍及实例

    MVC AOP面向切面编程 AOP这个词相信大家都没有接触太多过,但是实际上你们已经有所接触了,就在设计模式中.AOP所用的思想其实和设计模式是一样的,即在不修改原代码的情况下统一增加或者修改功能.还有,AOP大多用在spring里面,但是本文所写的只是在MVC中的应用,要注意. 一.简介 所谓AOP(Aspect Oriented Programming的缩写)意为面向切面的编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术.AOP是OOP的延续,是软件开发中的一个热点,也是

  • Spring AOP面向切面编程实现原理方法详解

    1. 什么是AOP AOP (Aspect Oriented Programming)意为:面向切面编程,通过预编译方式和运行期动态代理实现在不修改源代码的情况下,给程序动态统一添加功能的一种技术,可以理解成动态代理.是Spring框架中的一个重要内容.利用 AOP 可以对业务逻辑的各个部分进行隔离,使业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高开发的效率 2. Spring AOP ①. AOP 在Spring中的作用 提供声明式事务:允许用户自定义切面 ②. AOP 的基本概

  • Java实现AOP面向切面编程的实例教程

    介绍 众所周知,AOP(面向切面编程)是Spring框架的特色功能之一.通过设置横切关注点(cross cutting concerns),AOP提供了极高的扩展性.那AOP在Spring中是怎样运作的呢?当你只能使用core java,却需要AOP技术时,这个问题的解答变得极为关键.不仅如此,在高级技术岗位的面试中,此类问题也常作为考题出现.这不,我的朋友最近参加了一个面试,就被问到了这样一个棘手的问题--如何在不使用Spring及相关库,只用core Java的条件下实现AOP.因此,我将在

  • 面向切面编程(AOP)的理解

    在传统的编写业务逻辑处理代码时,我们通常会习惯性地做几件事情:日志记录.事务控制及权限控制等,然后才是编写核心的业务逻辑处理代码.当代码编写完成回头再看时,不禁发现,扬扬洒洒上百行代码中,真正用于核心业务逻辑处理才那么几行,如图6-4所示.方法复方法,类复类,就这样子带着无可奈何遗憾地度过了多少个春秋.这倒也罢,倘若到了项目的尾声,突然决定在权限控制上需要进行大的变动时,成千上万个方法又得一一"登门拜访",痛苦"雪上加霜". 如果能把图6-4中众多方法中的所有共有代

  • Spring 面向切面编程AOP实现详解

    简介 1.什么叫做面向切面编程? 概念:把一个个的横切关注点(某种业务的实现代码)放到某个模块中去,称之为切面.每个切面影响业务的一种功能,切面的目的就是为了功能增强,将需要增强的方法做成切面,实现对业务的增强,就是面向切面编程. 目的:将与业务本身无关,却被业务模块所共同调用的功能代码封装成切面,以减少系统的重复代码,降低耦合,提高可扩展性. 优势:把多个方法前/后的共同代码抽离出来,使用动态代理机制来控制,先执行抽离出来的代码,再执行每一个真实方法. 2.Spring中的AOP使用动态代理来

  • java开发AOP面向切面编程入门

    目录 引言 不好的解决方案 面向过程的解决方案 使用继承解决方案 使用聚合的解决方案 面向切面的编程基本概念 基于Spring面向切面程序实现 小结 引言 在实际应用场景中,我们封装一个学生的类,这个类用于封装学生的日常行为,如:上学.吃饭.上课等.然而,在疫情期间,学生上学时入校.吃饭时进入餐厅,需要测温查验证件等行为,拿到这样的需求我们怎么办? 不好的解决方案 面向过程的解决方案 遇到问题解决问题,在上学.吃饭方法中加上测温.查验证件方法,或者在学生类中提炼一个测温查验证件私有的方法,在需要

  • Java aop面向切面编程(aspectJweaver)案例详解

    面向切面编程的目的就是:在不改变别人的代码的前提下,在别人代码方法执行前或后,执行(切入自己的逻辑) 准备:idea+maven+aspectjweaver-1.8.9.jar 结构图: pom.xml内容 <dependencies> <dependency> <groupId>org.aspectj</groupId> <artifactId>aspectjrt</artifactId> <version>1.8.9&

  • SpringBoot整合aop面向切面编程过程解析

    这篇文章主要介绍了SpringBoot整合aop面向切面编程过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术.AOP是Spring框架中的一个重要内容,它通过对既有程序定义一个切入点,然后在其前后切入不同的执行内容,比如常见的有:打开数据库连接/关闭数据库连接.打开事务/关闭事务.记录日

  • Spring使用AspectJ的注解式实现AOP面向切面编程

    1.认识Spring AOP 1.1 AOP的简介 AOP:面向切面编程,相对于OOP面向对象编程. Spring的AOP的存在目的是为了解耦.AOP可以让一组类共享相同的行为.在OOP中只能通过继承类和实现接口,来使代码的耦合度增强,而且类的继承只能为单继承,阻碍更多行为添加到一组类上,AOP弥补了OOP的不足. 1.2 AOP中的概念 切入点(pointcut): 切入点(pointcut):在哪些类.哪些方法上切入. 通知(advice):在方法前.方法后.方法前后做什么. 切面(aspe

  • Spring框架AOP面向切面编程原理全面分析

    目录 1.什么是AOP AOP面向切面的优势 AOP需要添加的依赖 2.简述AOP工作运行原理 动态创建的总结: 3.使用Spring创建AOP 测试类 Spring.xml 1.什么是AOP AOP:Aspect Oriented Programming ⾯向切⾯编程. AOP面向切面的优势 降低模块之间的耦合度. 使系统更容易扩展. 更好的代码复⽤. ⾮业务代码更加集中,不分散,便于统⼀管理. 业务代码更加简洁存粹,不参杂其他代码的影响. AOP 是对⾯向对象编程的⼀个补充,在运⾏时,动态地

  • Javascript aop(面向切面编程)之around(环绕)分析

    Aop又叫面向切面编程,其中"通知"是切面的具体实现,分为before(前置通知).after(后置通知).around(环绕通知),用过spring的同学肯定对它非常熟悉,而在js中,AOP是一个被严重忽视的技术点.但是利用aop可以有效的改善js代码逻辑,比如前端框架dojo和yui3中AOP则被提升至自定义事件的一种内在机制,在源码中随处可见.得益于这种抽象使得dojo的自定义事件异常强大和灵活.dojo中aop的实现在dojo/aspect模块中,主要有三个方法:before.

  • .NET Core利用动态代理实现AOP(面向切面编程)

    目录 1.介绍 1.1 动态代理作用 1.2 原生DispatchProxy类介绍 1.3简单介绍一下:IL代码 2.实现 2.1 继承DispatchProxy 2.2 定义handle接口 2.3 定义AOP特性 2.4 定义创建代理类的工厂 2.5 定义ServiceHelp 3.测试 3.1 定义handle实现 3.2 定义Service接口 3.3实现Service接口 3.4 大功告成 3.5 效果 4.Demo 1.介绍 1.1 动态代理作用 用动态代理可以做AOP(面向切面编程

随机推荐