前段时间在朋友的带动下,入门了LOL,由于有时候自制力不是很强,平时没什么活动,所以就经常在无聊的时候打机,连续几个月,偶尔还通宵了,游戏卸载了两次,但是终究还是装上去了,突然觉得自己好傻逼,卸载了后又去下载安装。。。
前几天,一个年纪比我小的朋友,聊天中我问,你平时的业余时间有什么兴趣爱好么,然而回答让我比较吃惊,因为平时觉得周围的人的业余生活男的很多就是打游戏,女的很多都是看电视剧。直接和我说,玩游戏是在浪费青春。听到这句话的时候,忽然觉得自己好惭愧,也同时很佩服这类人,觉悟很大,所以自己也就好好收拾一下心态吧,走回学习的旅程。
话题好像扯远了,抱歉抱歉,扯回来哈,关于AOP使用自定义Spring 的注解的话个人觉得还是比较简单的,无非就是通过AOP的面向切面功能,在指定的代码位置织入我们想要的代码,通过参数的形式传递给切面处理程序,然后执行解析,解析完毕后我们就可以做我们想要的操作啦, 例如通过BeanDefinitionVisitor修改Spring容器里面的Bean对象,或者发起其他的调用,通知等。下面举个例子,通过AOP的形式定义注解来,然后通过注解记录我们的日志信息。
java注解简单介绍
AOP的简单介绍
AOP基本概念
- 切面(Aspect) :通知和切入点共同组成了切面,时间、地点和要发生的“故事”。
连接点(Joinpoint) :程序能够应用通知的一个“时机”,这些“时机”就是连接点,例如方法被调用时、异常被抛出时等等。
通知(Advice) :通知定义了切面是什么以及何时使用。描述了切面要完成的工作和何时需要执行这个工作。
切入点(Pointcut) :通知定义了切面要发生的“故事”和时间,那么切入点就定义了“故事”发生的地点,例如某个类或方法的名称。
目标对象(Target Object) :即被通知的对象。
AOP代理(AOP Proxy) 在Spring AOP中有两种代理方式,JDK动态代理和CGLIB代理。默认情况下,TargetObject实现了接口时,则采用JDK动态代理;反之,采用CGLIB代理。
织入(Weaving)把切面应用到目标对象来创建新的代理对象的过程,织入一般发生在如下几个时机:
1)编译时:当一个类文件被编译时进行织入,这需要特殊的编译器才能做到,例如AspectJ的织入编译器;
2)类加载时:使用特殊的ClassLoader在目标类被加载到程序之前增强类的字节代码;
3)运行时:切面在运行的某个时刻被织入,SpringAOP就是以这种方式织入切面的,原理是使用了JDK的动态代理。
AOP通知类型
自定义Spring注解实现
定义注解类
/*
* Copyright 2002-2013 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package net.itaem.annotation;
import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
/**
* 说明:
* 定义日志注解
*
* @author Fighter168
* @date 2016-04-17
*/
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface FileLog {
String value() default "日志记录开始";
}
定义注解的处理程序
/*
* Copyright 2002-2013 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package net.itaem.aspect;
import net.itaem.annotation.FileLog;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.stereotype.Component;
/**
* 说明:
* 日志的切面处理程序
*
* @author Fighter168
*/
@Aspect
@Component
public class LogAspect {
@AfterReturning("within(net.itaem..*) && @annotation(fl)")
public void addSuccessLog(JoinPoint jp, FileLog fl){
Object[] parames = jp.getArgs();
//获取目标方法体参数
String className = jp.getTarget().getClass().toString();
//获取目标类名
String signature = jp.getSignature().toString();
//获取目标方法签名
String methodName = signature.substring(signature.lastIndexOf(".")+1, signature.indexOf("("));
//获取注解值
String desc=fl.value();
//把调用的信息写到日常记录信息里面去...
}
//标注该方法体为异常通知,当目标方法出现异常时,执行该方法体
@AfterThrowing(pointcut="within(net.itaem..*) && @annotation(fl)", throwing="e")
public void addExceptionLog(JoinPoint jp, FileLog fl, Exception e){
//把错误信息写到错误日志文件里面去...
}
}
编写测试用例
<?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.2.xsd
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.1.xsd ">
<aop:aspectj-autoproxy/>
<bean name="purchaseService" class="net.itaem.test.PurchaseService" />
<bean class="net.itaem.aspect.LogAspect"/>
</beans>
/**
*
* @author Fighter168
*/
public class PurchaseService {
/**
*
*/
public void purchaseProduct(String productName,int price,String type) {
System.out.println("购买商品。。。");
}
}
package net.itaem.test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class Test {
@org.junit.Test
public void test() {
ApplicationContext context=new ClassPathXmlApplicationContext("net/itaem/source/test.xml");
PurchaseService service=(PurchaseService) context.getBean("purchaseService");
service.purchaseProduct("电风扇", 98, "日用品");
}
}
本文介绍了如何使用AOP结合自定义Spring注解来记录日志信息,包括注解定义、处理程序编写及测试案例。
1581

被折叠的 条评论
为什么被折叠?



