【原】通过抽象泛型模板优雅的设计可扩展代码

2018-06-18 02:23:00来源:未知 阅读 ()

新老客户大回馈,云服务器低至5折

  前言:

      通过学习码云上开源优秀项目后,发现一个较为不错的代码结构,所以在此进行总结。

  过程:

                   大致过程是从Service层获取数据,首先先经过redis,如果redis没有再去查询db。看似很简单的一个步骤,实际可以写出不同风格的方式。一般来说首先想到的无非是下面这样(伪代码):

    

Object val = redisutils.get("key");
if(val == null || val == ""){
    Object obj = userdaoMapper.getUserById("key");
BeanUtils.cover(obj,User.class); }
else{ return val; }

          上面的伪代码一看通俗易懂,但是在实际开发中,特别是多模块开发的场景下大量充斥着如上的代码的话那么维护起来是一件费劲的事情,不仅要编写大量的重复代码,还要对不同的实体类型进行一个转换,那么有什么办法可以避免这个问题呢?第一个办法想到的就是大名鼎鼎的AOP,但在这里不打算介绍AOP,而是如博客标题所示(通过抽象泛型模板)

 

      实现方式:

#首先创建一个缓存抽象模板,这个抽象模板属于基类, 最好定义成泛型,方便后续的扩展。在基类里实现了最核心的功能---》从redis获取数据,如果有则取出来,如果没有则从redis获取。    

public abstract class CacheWorker<P, R>
{
    private static Log logger = LogFactory.getLog(CacheWorker.class);

    @Autowired
    protected RedisUtil redisUtil;

    /**
     * get方式获取缓存
     * 
     * @param params 查询参数
     * @param expireSeconds 缓存过期时间
     * @return 结果
     * @throws SQLException
     */
    @SuppressWarnings("unchecked")
    public R get(P params, Class<R> clazz)
    {
        // 获取key,由继承者拼接
        String key = getKey(params);
        Object res = getCache(key, clazz);

        // 如果缓存中存在,直接返回
        if (res != null)
        {
            if (logger.isDebugEnabled())
            {
                StringBuilder sb = new StringBuilder();
                sb.append("从redis获取数据 (key:{").append(key).append("})");

                logger.debug(sb.toString());
            }
            return (R) res;
        }

        if (logger.isDebugEnabled())
        {
            StringBuilder sb = new StringBuilder();
            sb.append("从redis获取数据失败(key:{").append(key).append("}), 准备从DB获取.");

            logger.debug(sb.toString());
        }

        // 否则去DB中取
        R dataFromDb = getDataWhenNoCache(params);
        // 回写cache
        if (dataFromDb != null)
        {
            setCache(getExpireSeconds(), key, dataFromDb);
        }

        return dataFromDb;
    }

    /**
     * 获取过期时间
     * 
     * @return
     */
    protected abstract int getExpireSeconds();

    /**
     * set操作 设定缓存
     * 
     * @param expireSeconds
     * @param key
     * @param dataFromDb
     */
    protected void setCache(int expireSeconds, String key, R dataFromDb)
    {
        redisUtil.set(key, dataFromDb, expireSeconds);
    }

    /**
     * set操作 从缓存中取值
     * 
     * @param key
     * @return
     */
    protected Object getCache(String key, Class<R> clazz)
    {
        // 尝试获取缓存值
        return redisUtil.get(key, clazz);
    }

    public void del(P params)
    {
        // 获取key,由继承者拼接
        String key = getKey(params);
        redisUtil.delete(key);
    }

    /**
     * 当获取不到缓存时,使用该方法去DB或其他途径取数据
     * 
     * @param params
     * @return
     * @throws SQLException
     */
    protected abstract R getDataWhenNoCache(P params);

    /**
     * 获取key
     * 
     * @param params
     * @return
     */
    protected abstract String getKey(P params);

     

#创建一个子类并继承上面的基类,这个子类是一个具体的工作组件。在这个子类里扩展自己的方法,例如下面的代码是一个商品缓存组件,然后把对应的Mapper注入进去,如果要有一个订单缓存组件,那么同理把缓存订单组件的Mapper注入进去。

@Component
public class GoodsInfoCacheWorker extends CacheWorker<Integer, Goods>
{
    @Autowired
    private GoodsMapper goodsMapper;

    @Override
    protected Goods getDataWhenNoCache(Integer goodsId)
    {
        return goodsMapper.selectByPrimaryKey(goodsId);
    }

    @Override
    protected String getKey(Integer goodsId)
    {
        String key = MessageFormat.format(CommonConstant.RedisKey.GOODS_INFO_BY_ID, new Object[] { goodsId });
        return key;
    }

    @Override
    protected int getExpireSeconds()
    {
        return CommonConstant.RedisKeyExpireSeconds.GOODS_STORE_BY_ID;
    }

}

         

#使用方,通过spring注入后直接调用get方法并传入2个参数分别是key和需要转换的bean,在内部已经封装好了查询redis和查询db以及转换的代码。

    @Autowired
    private GoodsInfoCacheWorker goodsInfoCacheWorker;


    public String getGoodsRandomName(Integer goodsId)
    {
        Goods goods = goodsInfoCacheWorker.get(goodsId, Goods.class);
        long now = System.currentTimeMillis();

        // 已经开始了活动,则输出抢购链接
        if (goods.getStartTime().getTime() < now && now < goods.getEndTime().getTime())
        {
            return goods.getRandomName();
        }

        return StringUtils.EMPTY;
    }

 

 

  ·

标签:

版权申明:本站文章部分自网络,如有侵权,请联系:west999com@outlook.com
特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有

上一篇:基本编成结构笔记

下一篇:注解就这么简单