1.利用AOP实现Redis缓存

1.1 为什么要使用AOP?

1).业务代码与Redis缓存服务,紧紧的耦合在一起. 不方便后期维护.
2).如果采用下列代码的形式添加缓存机制,则不方便.每次使用缓存时,都需要按照缓存的机制重新执行业务代码. 显得特别的繁琐.
3).利用AOP的形式实现该操作.!!

/**
	 * 数据的来源: 数据库中
	 * 数据库中的数据类型:  ItemCat对象信息   POJO
	 * 需要的类型:        EasyUITree对象信息.  VO对象
	 * 思路: 将ItemCat转化为EasyUITree对象.
	 * sql: parent_id=0 SELECT * FROM tb_item_cat WHERE parent_id=0
	 */
	@Override
	public List<EasyUITree> findItemCatByParentId(Long parentId) {
		//1.根据parentId 查询数据库信息  根据父级查询子级信息.
		QueryWrapper<ItemCat> queryWrapper = new QueryWrapper<>();
		queryWrapper.eq("parent_id", parentId);
		List<ItemCat> itemCatList = itemCatMapper.selectList(queryWrapper);
		
		//2.将数据库记录转化为VO数据.
		List<EasyUITree> treeList = new ArrayList<EasyUITree>();
		for (ItemCat itemCat : itemCatList) {
			Long id = itemCat.getId();
			String text = itemCat.getName();
			//如果是父级标题,则默认关闭 closed,否则开启. open
			String state = itemCat.getIsParent()?"closed":"open"; 
			EasyUITree uiTree = new EasyUITree(id, text, state);
			treeList.add(uiTree);
		}
		
		return treeList;
	}
	
	/**
	 * 思路:
	 * 	1.定义查询redis的key, key要求唯一的
	 *  2.第一次查询先查询redis.
	 *  	没有数据: 表示缓存中没有数据, 查询数据库,之后将数据保存到redis中
	 *  	有数据:   证明缓存中有值, 直接返回给用户即可.
	 */
	@SuppressWarnings("unchecked")
	@Override
	public List<EasyUITree> findItemCatCache(Long parentId) {
		//定义key值
		String key = "ITEM_CAT_LIST::" + parentId;
		List<EasyUITree> treeList = new ArrayList<EasyUITree>();
		//判断redis缓存中是否有值
		if (jedis.exists(key)) {
			//表明有值时,直接将数据返回到页面
			String json = jedis.get(key);
			//将json转化为对象,返回到页面上
			treeList = ObjectMapperUtil.toObject(json, treeList.getClass());
			System.out.println("redis缓存查询");
		}else {
			//表明没有值,先查询数据库
			treeList = findItemCatByParentId(parentId);
			//将对象转化为json
			String json = ObjectMapperUtil.toJson(treeList);
			//将json保存到redis缓存中
			jedis.set(key, json);
			System.out.println("查询数据库");
		}
		return treeList;
	}

1.2 AOP核心理念

1.2.1 AOP说明

AOP: 对原有的方法进行扩展,可以将重复的事情,但是右不得不做的事情,放到AOP中执行.可以减少代码的耦合性.降 低维护的成本

redis  redis ap_java


公式:AOP = 切入点表达式 + 通知方法

1.2.2 切入点表达式

Spring中通过切入点表达式定义具体切入点,其常用AOP切入点表达式定义及说明:

redis  redis ap_redis 收费_02


bean表达式(重点)

bean表达式一般应用于类级别,实现粗粒度的切入点定义,案例分析:

bean(“userServiceImpl”)指定一个userServiceImpl类中所有方法。

bean("*ServiceImpl")指定所有后缀为ServiceImpl的类中所有方法。

说明:bean表达式内部的对象是由spring容器管理的一个bean对象,表达式内部的名字应该是spring容器中某个bean的name。

within表达式(了解)
within表达式应用于类级别,实现粗粒度的切入点表达式定义,案例分析:
within(“aop.service.UserServiceImpl”)指定当前包中这个类内部的所有方法。
within(“aop.service.") 指定当前目录下的所有类的所有方法。
within("aop.service…
”) 指定当前目录以及子目录中类的所有方法。

execution表达式(了解)
execution表达式应用于方法级别,实现细粒度的切入点表达式定义,案例分析:
语法:execution(返回值类型 包名.类名.方法名(参数列表))。
execution(void aop.service.UserServiceImpl.addUser())匹配addUser方法。
execution(void aop.service.PersonServiceImpl.addUser(String)) 方法参数必须为String的addUser方法。
execution(* aop.service….(…)) 万能配置。

@annotation表达式(重点)
@annotaion表达式应用于方法级别,实现细粒度的切入点表达式定义,案例分析
@annotation(anno.RequiredLog) 匹配有此注解描述的方法。
@annotation(anno.RequiredCache) 匹配有此注解描述的方法。
其中:RequiredLog为我们自己定义的注解,当我们使用@RequiredLog注解修饰业务层方法时,系统底层会在执行此方法时进行扩展操作。

1.2.3 通知方法

在基于Spring AOP编程的过程中,基于AspectJ框架标准,spring中定义了五种类型的通知(通知描述的是一种扩展业务),它们分别是:

@Before

@AfterReturning

@AfterThrowing

@After

@Around.重点掌握(优先级最高)

redis  redis ap_java_03

1.2.3 AOP 相关术语

AOP 相关术语分析
切面(aspect): 横切面对象,一般为一个具体类对象(可以借助@Aspect声明)。
通知(Advice):在切面的某个特定连接点上执行的动作(扩展功能),例如around,before,after等。
连接点(joinpoint):程序执行过程中某个特定的点,一般指被拦截到的的方法。
切入点(pointcut):对多个连接点(Joinpoint)一种定义,一般可以理解为多个连接点的集合。

1.2.4 入门案例

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;

//1.将对象交给spring容器管理
@Component
//2.定义一个切面
@Aspect
public class CacheAop {
	//切入点表达值
	@Pointcut("bean(itemCatServiceImpl)") //按类匹配
	public void pointCut() {
		
	}
	
	//通知方法
	/**
	 * 记录程序的执行状态  获取哪个类,哪个方法执行的
	 */
	@Before("pointCut()")
	public void before(	JoinPoint joinPoint) {
		System.out.println("我是前置通知");
		//获取方法的所在的包名和类名
		String typeName = joinPoint.getSignature().getDeclaringTypeName();
		String methodName = joinPoint.getSignature().getName();//获取方法的方法名
		Object[] args = joinPoint.getArgs();//获取方法的参数
		Object target = joinPoint.getTarget();//获取目标对象
		System.out.println("方法执行的全路径:" + typeName + "." + methodName);
		System.out.println("方法的参数:" + args);
		System.out.println("目标对象:" + target);
		
	}
	
	//添加环绕通知  可以控制目标方法执行
	@Around("pointCut()")
	public Object around(ProceedingJoinPoint joinPoint) {
		System.out.println("我是环绕通知的开始");
		try {
			Object result = joinPoint.proceed();
			System.out.println("我是环绕通知的结束");
			return result;
		} catch (Throwable e) {
			e.printStackTrace();
			throw new RuntimeException(e);
		}
	}
}

2.实现AOP缓存处理

2.1 自定义注解

import static java.lang.annotation.ElementType.METHOD;
import static java.lang.annotation.RetentionPolicy.RUNTIME;

import java.lang.annotation.Retention;
import java.lang.annotation.Target;

@Retention(RUNTIME) //表示运行期有效  注解使用的声明周期
@Target(METHOD)//表示注解在方法上有效
public @interface CacheFind {
	public String key();  //表示存入redis的key的前缀
	public int seconds() default 0; //表示保存的时间单位是秒
}

2.2 自定义注解的标识

redis  redis ap_缓存_04

2.3 实现AOP的缓存处理

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

import com.jt.annotation.CacheFind;
import com.jt.util.ObjectMapperUtil;

import redis.clients.jedis.Jedis;

@Component
@Aspect
public class CacheRedis {
	@Autowired
	private Jedis jedis;
	/**
	 * 实现思路: 拦截@CacheFind标识的方法之后利用AOP进行缓存的控制
	 * 通知方法:环绕通知
	 * 实现步骤:
	 * 	1.准备redis查询的key    ITEM_CAT_PARENTID  
	 *  2.@annotation(cacheFind)动态获取注解的语法,拦截指定注解类型的注解并且将注解对象
	 *  	当做参数进行传递
	 */
	
	@SuppressWarnings("unchecked")
	@Around("@annotation(cacheFind)")
	public Object around(ProceedingJoinPoint joinPoint,CacheFind cacheFind) {
		//1.获取用户注解中的key
		String key = cacheFind.key();
		//2.动态获取第一个参数当做key
		String args1 = joinPoint.getArgs()[0].toString();
		key += args1;
		//3.根据key查询redis
		Object result = null;
		if (jedis.exists(key)) {
			//如果缓存中有数据
			//获取redis中的数据
			String json = jedis.get(key);
			//动态获取目标方法的返回值类型
			MethodSignature methodSignature = (MethodSignature) joinPoint.getSignature();
			Class returnType = methodSignature.getReturnType();
			//将json字符串转化为对象
			result = ObjectMapperUtil.toObject(json, returnType);
			System.out.println("查询AOP缓存");
		}else {
			try {
				//如果key不存在,则需要执行目标方法查询数据库
				result = joinPoint.proceed();
				//将查询到的数据转化为json字符串
				String json = ObjectMapperUtil.toJson(result);
				System.out.println("AOP执行数据库查询");
				//将json保存到redis中
				int seconds = cacheFind.seconds();
				if (seconds > 0) {
					jedis.setex(key, seconds, json);//添加超时时间
				}else {
					jedis.set(key, json);//不需要超时时间
				}
			} catch (Throwable e) {
				e.printStackTrace();
				throw new RuntimeException();
			}
		}
		return result;
	}
}