設(shè)計模式-行為模式-模板模式

提要

前幾天做項目時,遇到一個類設(shè)計的問題,請教同事后,參考已有代碼完成我需要的設(shè)計,這時回想設(shè)計模式,終于能體會它的意義了??

問題

簡單描述問題就是:

我希望有一個從Redis取特定String緩存的工具,可以依據(jù)不同參數(shù)做不同取緩存操作

public class Pattern {

    public String getStringById(String keyPrefix, Long id) {
        ...
            
        A
            
        ...
            
    }
    
    private String getStringByIdWithRedisLock(String keyPrefix, Long id){
        ...
            
            
        B   
            
        ...
    }
    
}

整體模板就是這樣,現(xiàn)在問題是,在調(diào)用private時B部分代碼會依據(jù)參數(shù)不同走不同的調(diào)用,也就是我需要一種能利用這個模板,僅僅暴露出public方法便可完成調(diào)用

可以想一想自己會怎么做?

解決

有想法了嗎?

最簡單最容易想到的就是設(shè)計接口,不同實現(xiàn),構(gòu)造器傳入這樣的

廢話不多說上代碼

public interface RedisService {
    /**
     * 通過id獲取string,流程:redis-db
     * 使用包括,通過id獲取username,通過id獲取title,通過id獲取categoryName,通過id獲取feedbackTypeName。。。
     * @param keyPrefix
     * @param id
     * @return
     */
    String getStringById(String keyPrefix, Long id);

    String getStringByIdFromDB(Long id);
}

設(shè)計接口,其實也不是很必要,直接設(shè)計抽象類也可以

@Component
@Slf4j
public abstract class AbstractRedisService implements RedisService {

    @Autowired
    StringRedisTemplate stringRedisTemplate;

    @Override
    public String getStringById(String keyPrefix, Long id) {
        String value = stringRedisTemplate.opsForValue().get(keyPrefix + id);
        if (StringUtils.isEmpty(value)) {
            log.info("緩存未命中。。。將要查詢數(shù)據(jù)庫。。。");
            value = getStringByIdWithRedisLock(keyPrefix, id);
            stringRedisTemplate.opsForValue().set(keyPrefix + id, value, RedisKeyConstants.DEFAULT_TTL, TimeUnit.SECONDS);
            return value;
        }
        log.info("緩存命中。。。直接讀取。。。");
        return value;
    }


    private String getStringByIdWithRedisLock(String keyPrefix, Long id) {
        String key = keyPrefix + id;
        String uuid = UUID.randomUUID().toString();
        Boolean lock = stringRedisTemplate.opsForValue()
                .setIfAbsent(RedisLockUtils.LOCK + key, uuid, RedisLockUtils.EXPIRE, TimeUnit.SECONDS);
        if (Objects.nonNull(lock) && lock) {
            log.info("獲取分布式鎖成功。。。");
            String value = "";
            try {

                value = this.getStringByIdFromDB(id);
            } catch (Exception e) {
                log.error("查詢數(shù)據(jù)庫失敗", e);
            } finally {
                stringRedisTemplate.execute(new DefaultRedisScript<>(RedisLockUtils.UNLOCK_LUA, Boolean.class)
                        , Collections.singletonList(RedisLockUtils.LOCK + key), uuid);
            }
            return value;
        } else {
            log.info("獲取分布式鎖失敗。。。等待重試。。。");
            try {
                Thread.sleep(RedisLockUtils.RETRY_TIME);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            return getStringByIdWithRedisLock(keyPrefix, id);
        }
    }

    @Override
    public abstract String getStringByIdFromDB(Long id);

}

上面的已經(jīng)很清晰了自己看就能明白

@Component
@Slf4j
public class UserRedisService extends AbstractRedisService {
    @Autowired
    UserMapper userMapper;

    @Override
    public String getStringByIdFromDB(Long id) {
        String userName = stringRedisTemplate.opsForValue().get(RedisKeyConstants.USER_NAME + id);
        if (!StringUtils.isEmpty(userName)) {
            return userName;
        }
        log.info("查詢了數(shù)據(jù)庫。。。");

        User user = userMapper.selectByPrimaryKey(id);
        if (Objects.isNull(user)) {
            return "";
        }
        stringRedisTemplate.opsForValue().set(RedisKeyConstants.USER_NAME + id, user.getUserName()
                , RedisKeyConstants.DEFAULT_TTL, TimeUnit.SECONDS);
        return user.getUserName();
    }

    public void deleteUserName(Long id) {
        stringRedisTemplate.delete(RedisKeyConstants.USER_NAME + id);
    }
}

抽象類的一個實現(xiàn)

完成這個設(shè)計還是很自豪,轉(zhuǎn)頭一想,這是不是哪種設(shè)計模式???

果然,查到了,它屬于設(shè)計模式-行為模式-模板模式,感興趣的可以查資料好好看一看

小結(jié)

果然吶,一些設(shè)計模式只有你在生產(chǎn)工作中才會有更深的體會和理解!

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容