Spring AOP - การนำไปใช้งาน

สปริงรองรับ @AspectJ annotation style แนวทางและ schema-based แนวทางในการใช้แง่มุมที่กำหนดเอง

ตาม XML Schema

แง่มุมถูกนำไปใช้โดยใช้คลาสปกติพร้อมกับการกำหนดค่าตาม XML

ในการใช้แท็กเนมสเปซ AOP ที่อธิบายในส่วนนี้คุณต้องนำเข้าสคีมา AOP แบบสปริงซึ่งอธิบายไว้ดังต่อไปนี้ -

<?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"
   xsi:schemaLocation = "http://www.springframework.org/schema/beans
   http://www.springframework.org/schema/beans/spring-beans-3.0.xsd 
   http://www.springframework.org/schema/aop 
   http://www.springframework.org/schema/aop/spring-aop-3.0.xsd ">

   <!-- bean definition & AOP specific configuration -->

</beans>

การประกาศมุมมอง

อัน aspect ถูกประกาศโดยใช้ <aop:aspect> องค์ประกอบและ backing bean อ้างอิงโดยใช้ ref แอตทริบิวต์ดังนี้

<aop:config>
   <aop:aspect id = "myAspect" ref = "aBean">
   ...
   </aop:aspect>
</aop:config>

<bean id = "aBean" class = "...">
   ...
</bean>

ที่นี่ "aBean" จะได้รับการกำหนดค่าและการพึ่งพาการฉีดเช่นเดียวกับ Spring bean อื่น ๆ ตามที่คุณเห็นในบทก่อน ๆ

การประกาศ PointCut

PointCutช่วยในการกำหนดจุดเข้าร่วม (เช่นวิธีการ) ที่น่าสนใจที่จะดำเนินการด้วยคำแนะนำที่แตกต่างกัน ในขณะที่ทำงานกับการกำหนดค่าตาม XML Schema PointCut จะถูกกำหนดดังนี้ -

<aop:config>
   <aop:aspect id = "myAspect" ref = "aBean">

   <aop:PointCut id = "businessService"
      expression = "execution(* com.xyz.myapp.service.*.*(..))"/>
      ...
   </aop:aspect>
</aop:config>

<bean id = "aBean" class = "...">
   ...
</bean>

ตัวอย่างต่อไปนี้กำหนด PointCut ชื่อ 'businessService' ที่จะจับคู่การดำเนินการของเมธอด getName () ที่มีอยู่ในคลาสนักเรียนภายใต้แพ็คเกจ com.tutorialspoint

<aop:config>
   <aop:aspect id = "myAspect" ref = "aBean">

   <aop:PointCut id = "businessService"
      expression = "execution(* com.tutorialspoint.Student.getName(..))"/>
   ...
   </aop:aspect>
</aop:config>

<bean id = "aBean" class = "...">
   ...
</bean>

การประกาศคำแนะนำ

คุณสามารถประกาศคำแนะนำใด ๆ จากห้าคำแนะนำภายใน <aop: ด้าน> โดยใช้องค์ประกอบ <aop: {ADVICE NAME}> ดังนี้

<aop:config>
   <aop:aspect id = "myAspect" ref = "aBean">
      <aop:PointCut id = "businessService"
         expression = "execution(* com.xyz.myapp.service.*.*(..))"/>

      <!-- a before advice definition -->
      <aop:before PointCut-ref = "businessService" 
         method = "doRequiredTask"/>

      <!-- an after advice definition -->
      <aop:after PointCut-ref = "businessService" 
         method = "doRequiredTask"/>

      <!-- an after-returning advice definition -->
      <!--The doRequiredTask method must have parameter named retVal -->
      <aop:after-returning PointCut-ref = "businessService"
         returning = "retVal"
         method = "doRequiredTask"/>

      <!-- an after-throwing advice definition -->
      <!--The doRequiredTask method must have parameter named ex -->
      <aop:after-throwing PointCut-ref = "businessService"
        throwing = "ex"
         method = "doRequiredTask"/>

      <!-- an around advice definition -->
      <aop:around PointCut-ref = "businessService" 
         method = "doRequiredTask"/>
   ...
   </aop:aspect>
</aop:config>

<bean id = "aBean" class = "...">
   ...
</bean>

คุณสามารถใช้เช่นเดียวกัน doRequiredTaskหรือวิธีการต่างๆสำหรับคำแนะนำต่างๆ วิธีการเหล่านี้จะถูกกำหนดให้เป็นส่วนหนึ่งของโมดูลด้าน

@AspectJ อิง

@AspectJ หมายถึงรูปแบบของการประกาศแง่มุมเป็นคลาส Java ปกติที่มีคำอธิบายประกอบ Java 5 การสนับสนุน @AspectJ เปิดใช้งานโดยการรวมองค์ประกอบต่อไปนี้ไว้ในไฟล์คอนฟิกูเรชันตาม XML Schema ของคุณ

<aop:aspectj-autoproxy/>

การประกาศมุมมอง

คลาส Aspects ก็เหมือนกับ bean ทั่วไปทั่วไปและอาจมีเมธอดและฟิลด์เหมือนกับคลาสอื่น ๆ ยกเว้นว่าจะมีการใส่คำอธิบายประกอบด้วย @Aspect ดังนี้

package org.xyz;

import org.aspectj.lang.annotation.Aspect;

@Aspect
public class AspectModule {

}

พวกเขาจะถูกกำหนดค่าใน XML เหมือนกับ bean อื่น ๆ ดังต่อไปนี้

<bean id = "myAspect" class = "org.xyz.AspectModule">
   <!-- configure properties of aspect here as normal -->
</bean>

การประกาศ PointCut

PointCutช่วยในการกำหนดจุดเข้าร่วม (เช่นวิธีการ) ที่น่าสนใจที่จะดำเนินการด้วยคำแนะนำที่แตกต่างกัน ในขณะที่ทำงานกับการกำหนดค่าตาม @AspectJ การประกาศ PointCut มีสองส่วน -

  • นิพจน์ PointCut ที่กำหนดวิธีการดำเนินการที่เราสนใจ

  • ลายเซ็น PointCut ประกอบด้วยชื่อและพารามิเตอร์จำนวนเท่าใดก็ได้ เนื้อหาที่แท้จริงของวิธีการไม่เกี่ยวข้องและในความเป็นจริงควรว่างเปล่า

ตัวอย่างต่อไปนี้กำหนด PointCut ชื่อ 'businessService' ซึ่งจะจับคู่การดำเนินการของทุกวิธีที่มีอยู่ในคลาสภายใต้แพ็คเกจ com.xyz.myapp.service

import org.aspectj.lang.annotation.PointCut;

@PointCut("execution(* com.xyz.myapp.service.*.*(..))") // expression 
private void businessService() {}  // signature

ตัวอย่างต่อไปนี้กำหนด PointCut ชื่อ 'getname' ที่จะจับคู่การดำเนินการของเมธอด getName () ที่มีอยู่ในคลาส Student ภายใต้แพ็คเกจ com.tutorialspoint

import org.aspectj.lang.annotation.PointCut;

@PointCut("execution(* com.tutorialspoint.Student.getName(..))") 
private void getname() {}

การประกาศคำแนะนำ

คุณสามารถประกาศคำแนะนำใด ๆ จากห้าข้อโดยใช้คำอธิบายประกอบ @ {ADVICE-NAME} ตามที่ระบุด้านล่าง สิ่งนี้ถือว่าคุณได้กำหนดเมธอดลายเซ็น PointCut businessService () ไว้แล้ว

@Before("businessService()")
public void doBeforeTask(){
   ...
}

@After("businessService()")
public void doAfterTask(){
   ...
}

@AfterReturning(PointCut = "businessService()", returning = "retVal")
public void doAfterReturnningTask(Object retVal){
   // you can intercept retVal here.
   ...
}

@AfterThrowing(PointCut = "businessService()", throwing = "ex")
public void doAfterThrowingTask(Exception ex){
   // you can intercept thrown exception here.
   ...
}

@Around("businessService()")
public void doAroundTask(){
   ...
}

คุณสามารถกำหนด PointCut แบบอินไลน์สำหรับคำแนะนำใด ๆ ต่อไปนี้เป็นตัวอย่างในการกำหนด PointCut แบบอินไลน์สำหรับคำแนะนำก่อน

@Before("execution(* com.xyz.myapp.service.*.*(..))")
public doBeforeTask(){
   ...
}