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(){
...
}