模仿spring-aop的功能,利用注解搭建自己的框架。
入JAVA坑7月有余,也尝试自己手动搭建框架,最近对spring aop的这种切面很着迷,为此记录下自己目前搭出来的小小的demo,后续有时间也会继续改进自己的demo。望大神们不吝赐教。
主要还是运用反射和java自带的代理类。理论知识就不说了,因为我目前也不是很清楚,避免误导,还是避而不谈吧。好了,直接根据代码撸吧。
结构:
接口
Person.java
public interface Person {
void say();
}
接口实现类
Man.java
public class Man implements Person {
@Override
public void say() {
System.out.println("男人say:....");
}
}
自定义注解
@interface WaterAOP
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.PARAMETER, ElementType.METHOD})
@Documented
public @interface WaterAOP {
enum METHOD{before,after,afterthrowing}
METHOD method() default METHOD.after;
String Name() default "类全名";
}
自定义注解类
WaterLog.java
public class WaterLog { @WaterAOP(Name = "com.water.aop.attempt3.Man",method = WaterAOP.METHOD.after)
public void afterAction(){
System.out.println("后置行为");
}
@WaterAOP(Name = "com.water.aop.attempt3.Man",method = WaterAOP.METHOD.before)
public void beforeAction(){
System.out.println("前置行为");
}
}
实现自定义代理类(就是在
Proxy.newProxyInstance()方法的第三个参数里做手脚。用了java8的lambda表达式。
)
ProxyFactory.java
public class ProxyFactory {
// 维持一个实现接口的被代理的对象,后面改为对象组,由浅入深
private Person person;
private WaterLog waterLog;
private Method beforeMethod=null,afterMethod=null;
public ProxyFactory(Person person,WaterLog waterLog){
this.person=person;
this.waterLog=waterLog;
}
public Object getProxyInstance(){
return Proxy.newProxyInstance(
person.getClass().getClassLoader(),
person.getClass().getInterfaces(),
// 第一个参数就是代理者,如果你想对代理者做一些操作可以使用这个参数;
// 第二个就是被执行的方法,
// 第三个是执行该方法所需的参数。
(Object proxyObj, Method method,Object[] args)->{
//如果没有传入aop 直接返回空
if(waterLog==null){
return null;
}
Class aop=waterLog.getClass();
Class c = person.getClass();
// 获取aop类的方法的注解并赋给自定义的一些变量,下面根据这些变量是否有值来确定是否有注解
getAnnotation(aop,c);
if(beforeMethod!=null){
beforeMethod.invoke(waterLog);
}
// 代理对象执行方法并且获得返回值
Object returnValue=method.invoke(person,args);
if(afterMethod!=null){
afterMethod.invoke(waterLog);
}
return returnValue;
}
);
}
private void getAnnotation(Class aop,Class proxy){
//如果有AOP的类
if(waterLog!=null){
// 获取切面类所有的方法
Method[] methodsAOP=aop.getMethods();
// 如果切入的日志类的方法不为空
if(methodsAOP!=null){
for(Method logMethod:methodsAOP){
// 取得WaterLog类的方法上WaterAOP注解
WaterAOP waterAOP=logMethod.getAnnotation(WaterAOP.class);
if(waterAOP!=null) {
// 如果AOP上的注解与传入的类名一致
if (proxy.toString().substring(6).equals(waterAOP.Name())) {
if (waterAOP.method() == WaterAOP.METHOD.before) {
// 赋值 ,后面再执行
beforeMethod=logMethod;
}else if(waterAOP.method() == WaterAOP.METHOD.after){
afterMethod=logMethod;
}
}
}
}
}
}
} }
测试类
Test.java (junit是个测试包,也可以直接用main方法)
public class Test {
@org.junit.Test
public void waterAOP(){
Person person=new Man();
Person proxyPerson=(Person) new ProxyFactory(person,new WaterLog()).getProxyInstance();
proxyPerson.say();
}
}
大致的流程就是:传入要被代理的类和自定义的注解类,运用反射获取注解类里方法上的注解属性的值,然后进行比对,再进行相应的操作。
模仿spring-aop的功能,利用注解搭建自己的框架。的更多相关文章
- Spring AOP中使用@Aspect注解 面向切面实现日志横切功能详解
引言: AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术.AOP是OOP的延续,是软件开发中的一 ...
- Spring AOP配置简单记录(注解及xml配置方式)
在了解spring aop中的关键字(如:连接点(JoinPoint).切入点(PointCut).切面(Aspact).织入(Weaving).通知(Advice).目标(Target)等)后进行了 ...
- Spring AOP 实现功能权限校验功能
版权声明:本文为博主原创文章,未经博主允许不得转载. 目录(?)[-] 使用拦截器实现未登录时跳转到登录界面的功能 1 拦截器SecurityInterceptor 2spring-mvcxml拦 ...
- Spring AOP(5)-- 注解
applicationContext.xml <?xml version="1.0" encoding="UTF-8"?><beans xml ...
- 【Spring AOP】Spring AOP之如何通过注解的方式实现各种通知类型的AOP操作进阶篇(3)
一.切入点表达式的各种类型 切入点表达式的作用:限制连接点的匹配(满足时对应的aspect方法会被执行) 1)execution:用于匹配方法执行连接点.Spring AOP用户可能最经常使用exec ...
- spring AOP (使用AspectJ的注解方式 的aop实现) (6)
目录 一.在 Spring 中启用 AspectJ 注解支持 二.AspectJ 支持 5 种类型的通知注解: 2.1.使用之前的 计算器接口和实现类 ArithmeticCalculator.jav ...
- 循序渐进之Spring AOP(6) - 使用@Aspect注解
前面几节的示例看起来让人沮丧,要记忆如此多的接口.类和继承关系,做各种复杂的配置.好在这些只是一种相对过时的实现方式,现在只需要使用@Aspect注解及表达式就可以轻松的使用POJO来定义切面,设计精 ...
- Spring AOP 和 动态代理技术
AOP 是什么东西 首先来说 AOP 并不是 Spring 框架的核心技术之一,AOP 全称 Aspect Orient Programming,即面向切面的编程.其要解决的问题就是在不改变源代码的情 ...
- Spring基础知识之基于注解的AOP
背景概念: 1)横切关注点:散布在应用中多处的功能称为横切关注点 2)通知(Advice):切面完成的工作.通知定了了切面是什么及何时调用. 5中可以应用的通知: 前置通知(Before):在目标方法 ...
随机推荐
- 【JavaScript运算符与表达式】
一.表达式 1.原始表达式:2.14,"test",true/false,null--复合表达式:10*20-- 2.数组.对象的初始化表达式:new Array(1,2),[1, ...
- linux下,文件的权限和数字对应关系详解
命令 chmod ABC file 其中A.B.C各为一个数字,分别表示User.Group.及Other的权限. A.B.C这三个数字如果各自转换成由"0"."1&qu ...
- POJ 1659 Frogs' Neighborhood(可图性判定—Havel-Hakimi定理)【超详解】
Frogs' Neighborhood Time Limit: 5000MS Memory Limit: 10000K Total Submissions: 9897 Accepted: 41 ...
- [51nod1502]苹果曼和纸
苹果曼有很大的一张纸.这张纸的形状是1×n的长方形.你的任务是帮助苹果曼来折叠这一张纸.有一些操作,这些操作有如下两个种形式: 1. 把这张纸在第pi个位置对折.经过对折后,左边的1×pi部分会盖到右 ...
- MFC获取可执行文件(exe)所在文件目录
可以应用函数GetModuleFileName(),举一个例子: CString strexe; ::GetModuleFileName(NULL,strexe.GetBufferSetLength( ...
- [国嵌笔记][004][Linux快速体验]
Linux文件系统 bin目录:可执行的程序 boot目录:与Linux启动相关的文件 dev目录:设备以文件的方式存放 etc目录:配置文件 home目录:用户文件 lib目录:与库相关的文件 ro ...
- Web前端:改变鼠标样式
<span style="cursor:auto">浏览器设置的光标</span><br /> <span style="cur ...
- dede被注入后台提示用户名不存在解决方法
如果已经发现/data,有很长一个txt记事本,说明已经被其他人SQL注入了,或是已经有人进行尝试SQL注入了了. 记事儿本如:75252sdaswfdfsfq538ef2ad3556_safe.tx ...
- javaScript中关于字符串的操作函数和方法
1.字符串转换 toString():可以将任何类型的数据都转换为字符串 var num= 19; //19 var myStr = num.toString(); //"19" ...
- Redis进阶实践之七Redis和Lua初步整合使用
一.引言 Redis学了一段时间了,基本的东西都没问题了.从今天开始讲写一些redis和lua脚本的相关的东西,lua这个脚本是一个好东西,可以运行在任何平台上,也可以嵌入到大多数语言当 ...