Spring-Boot基于配置按條件裝載Bean

背景

同一個(gè)接口有多種實(shí)現(xiàn),項(xiàng)目啟動(dòng)時(shí)按某種規(guī)則來(lái)選擇性的啟用其中一種實(shí)現(xiàn),再具體一點(diǎn)怪嫌,比如Controller初始化的時(shí)候,根據(jù)配置文件的指定的實(shí)現(xiàn)類(lèi)前綴柳沙,來(lái)記載具體Service岩灭,不同Service使用不同的Dao和數(shù)據(jù)庫(kù)。


業(yè)務(wù)背景示意圖.png

看到這里赂鲤,我們會(huì)想到使用SPI機(jī)制噪径,或Spring按條件加載Bean機(jī)制來(lái)實(shí)現(xiàn),下面主要討論后者数初。

定義接口

定義2個(gè)Service層接口:OrderService熄云、OrderPromotionService,分別有一個(gè)方法妙真,如下:

// OrderService.java

public interface OrderService {

    /**
     * 通過(guò)tid查詢(xún)訂單信息
     * @param tid 訂單主鍵
     */
    List<Order> findByTid(Long tid);
}

...

// OrderPromotionService.java

public interface OrderPromotionService {

    /**
     * 通過(guò)tid獲取促銷(xiāo)詳情.
     * @param tid 訂單唯一標(biāo)識(shí)
     */
    List<OrderPromotion> findByTid(Long tid);
}

默認(rèn)實(shí)現(xiàn)

分別實(shí)現(xiàn)上面2個(gè)接口的各自的方法缴允,包路徑:com.a.b

為達(dá)到根據(jù)規(guī)則裝載不同ServiceImpl的目的珍德,需要使用@Conditional注解练般,并且實(shí)現(xiàn)規(guī)則定義DefaultCondition。當(dāng)Spring掃描到@Service注解時(shí)锈候,會(huì)判斷DefaultCondition#matches()方法薄料,決定是否裝載該ServiceImpl。

DefaultCondition實(shí)現(xiàn)如下:

// DefaultCondition.java

public class DefaultCondition extends ParentCondition implements Condition  {

    @Override
    public boolean matches(ConditionContext conditionContext, AnnotatedTypeMetadata annotatedTypeMetadata) {
        Environment environment = conditionContext.getEnvironment();

        String interfaceName = getInterFaceName(annotatedTypeMetadata);
        String implPrefix = environment.getProperty(interfaceName);
        if (StringUtils.isEmpty(implPrefix)) {
            return true;
        }
        return Constant.DEFAULT_PREFIX.equals(implPrefix);
    }
}

...

// Constant.java

public class Constant {
    public static final String THIRD_PARTY_PREFIX = "ThirdParty";
    public static final String DEFAULT_PREFIX = "Default";

}

DefaultCondition實(shí)現(xiàn)了Spring的Condition接口泵琳,先通過(guò)方法ParentCondition#getInterFaceName()獲取ServiceImpl實(shí)現(xiàn)的接口名稱(chēng)interfaceName摄职,然后從配置文件中獲取該接口指定的實(shí)現(xiàn)類(lèi)的前綴implPrefix,然后判斷是否為Constant.DEFAULT_PREFIX获列,如果是谷市,則裝載該ServiceImpl。

獲取接口名稱(chēng)的實(shí)現(xiàn)定義在ParentCondition類(lèi)中(是自己實(shí)現(xiàn)的)击孩,是這里需要繼承ParentCondition的原因迫悠。

  • Service層實(shí)現(xiàn)
// DefaultOrderServiceImpl.java

@Conditional(DefaultCondition.class)
@Service
public class DefaultOrderServiceImpl implements OrderService {

    @Autowired
    private OrderDao orderDao;

    @Override
    public List<Order> findByTid(Long tid) {
        return orderDao.findByTid(tid);
    }
}

...

/**
 * 默認(rèn)實(shí)現(xiàn).
 */
@Conditional(DefaultCondition.class)
@Service
public class DefaultOrderPromotionServiceImpl implements OrderPromotionService {

    @Autowired
    private OrderPromotionDao promotionDao;

    @Override
    public List<OrderPromotion> findByTid(Long tid) {
        return promotionDao.findByTid(tid);
    }
}
  • Dao層實(shí)現(xiàn)
// OrderDao.java

@Repository
public class OrderDao {

    @Autowired
    @Qualifier("firstJdbcTemplate")
    private JdbcTemplate jdbcTemplate;

    public List<Order> findByTid(Long tid) {
       // 省略...
    }
}

...

@Repository
public class OrderPromotionDao {

    @Autowired
    @Qualifier("firstJdbcTemplate")
    private JdbcTemplate jdbcTemplate;

    public List<OrderPromotion> findByTid(final Long tid) {
       // 省略...
    }
}

Dao層使用的是自己的數(shù)據(jù)源,代碼注入的是firstJdbcTemplate巩梢,Spring-Boot多數(shù)據(jù)源配置不是今天討論的重點(diǎn)创泄,這里不再詳細(xì)說(shuō)明。

第三方實(shí)現(xiàn)

這里假設(shè)上面的2個(gè)Service接口還有一種第三方的實(shí)現(xiàn)括蝠,包路徑:com.c.d鞠抑。跟默認(rèn)實(shí)現(xiàn)類(lèi)似,這里也要定義自己的ServiceImpl裝載規(guī)則ThirdPartyCondition忌警,實(shí)現(xiàn)如下:

// ThirdPartyOrderServiceImpl.java

public class ThirdPartyCondition extends ParentCondition implements Condition {

    @Override
    public boolean matches(ConditionContext conditionContext, AnnotatedTypeMetadata annotatedTypeMetadata) {
        Environment environment = conditionContext.getEnvironment();

        String interfaceName = getInterFaceName(annotatedTypeMetadata);
        String implPrefix = environment.getProperty(interfaceName);
        return Constant.THIRD_PARTY_PREFIX.equals(implPrefix);
    }
}

先獲取ServiceImpl實(shí)現(xiàn)的接口名稱(chēng)interfaceName搁拙,然后從配置文件中獲取該接口指定的實(shí)現(xiàn)類(lèi)的前綴implPrefix,然后判斷是否為Constant.THIRD_PARTY_PREFIX,如果是感混,則裝載該ServiceImpl端幼。

  • Service層實(shí)現(xiàn)
// ThirdPartyOrderServiceImpl.java

@Conditional({ThirdPartyCondition.class})
@Service
public class ThirdPartyOrderServiceImpl implements OrderService {

    @Autowired
    private ThirdPartyOrderDao thirdPartyOrderDao;

    @Override
    public List<Order> findByTid(Long tid) {
        return thirdPartyOrderDao.findByTid(tid);
    }
}

...

//ThirdPartyOrderPromotionServiceImpl.java

@Conditional(ThirdPartyCondition.class)
@Service
public class ThirdPartyOrderPromotionServiceImpl implements OrderPromotionService {

    @Autowired
    private ThirdPartyOrderPromotionDao thirdPartyOrderPromotionDao;

    @Override
    public List<OrderPromotion> findByTid(Long tid) {
        return thirdPartyOrderPromotionDao.findByTid(tid);
    }
}
  • Dao層實(shí)現(xiàn)

省略,Dao層實(shí)現(xiàn)使用另一個(gè)數(shù)據(jù)源弧满,注入secondJdbcTemplate婆跑。

以上分別為OrderService和OrderPromotionService提供了兩種實(shí)現(xiàn),如下:

Service接口 Service默認(rèn)實(shí)現(xiàn) Service第三方實(shí)現(xiàn)
OrderService DefaultOrderServiceImp
使用Dao層OrderDao庭呜,數(shù)據(jù)源firstJdbcTemplate
ThirdPartyOrderServiceImpl
使用Dao層ThirdPartyOrderDao滑进,數(shù)據(jù)源secondJdbcTemplate
OrderPromotionService DefaultOrderPromotionServiceImpl
使用Dao層DefaultOrderPromotionServiceImpl,數(shù)據(jù)源firstJdbcTemplate
ThirdPartyOrderPromotionServiceImpl
使用Dao層ThirdPartyOrderPromotionDao募谎,數(shù)據(jù)源secondJdbcTemplate

配置規(guī)則

ServiceImpl定義完成扶关,裝載規(guī)則也定義了,下面我們?cè)赟pring-Boot中分別指定兩個(gè)類(lèi)的加載對(duì)象

// application.properties

com.free.spring.jdbc.demo.service.OrderPromotionService=ThirdParty
com.free.spring.jdbc.demo.service.OrderService=Default

如上数冬,定義了OrderPromotionService接口使用第三方的實(shí)現(xiàn)节槐,OrderService接口使用默認(rèn)實(shí)現(xiàn)
下面簡(jiǎn)單的寫(xiě)兩個(gè)Controller看一下效果。

運(yùn)行效果

1. 數(shù)據(jù)準(zhǔn)備

為first庫(kù)的2張表分別添加1條數(shù)據(jù)拐纱,如下:

fisrt庫(kù)數(shù)據(jù).png

為second庫(kù)的2張表添加數(shù)據(jù)

second庫(kù)數(shù)據(jù).png

2. Controller定義

// OrderController.java

@RestController
@RequestMapping("/order")
public class OrderController {

    @Autowired
    private OrderService orderService;

    @GetMapping("/{tid}")
    @ResponseBody
    public List<Order> findByTid(@PathVariable("tid") Long tid) {
        return orderService.findByTid(tid);
    }
}

...

// OrderPromotionController.java

@RestController
@RequestMapping("/promotion")
public class OrderPromotionController {

    @Autowired
    private OrderPromotionService promotionService;

    @GetMapping("/{tid}")
    @ResponseBody
    public List<OrderPromotion> query(@PathVariable("tid") Long tid) {
        try {
            return promotionService.findByTid(tid);
        } catch (Exception e) {
            e.printStackTrace();
        }
        return null;
    }
}

我們通過(guò)Controller分別請(qǐng)求OrderService和OrderPromotionService铜异,通過(guò)返回的數(shù)據(jù)判斷是否真的實(shí)現(xiàn)了選擇性裝載ServiceImpl。

然后我們分別請(qǐng)求上面兩個(gè)接口秸架,觀(guān)察結(jié)果:

運(yùn)行結(jié)果.png

OK揍庄!沒(méi)問(wèn)題!
如果大家有更簡(jiǎn)單的方式东抹,歡迎探討~
Github源碼:https://github.com/nmyphp/spring-cloud-demo (spring-jdbc-demo模塊)

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末蚂子,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子缭黔,更是在濱河造成了極大的恐慌食茎,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,451評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件试浙,死亡現(xiàn)場(chǎng)離奇詭異董瞻,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)田巴,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,172評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)挟秤,“玉大人壹哺,你說(shuō)我怎么就攤上這事∷腋眨” “怎么了管宵?”我有些...
    開(kāi)封第一講書(shū)人閱讀 164,782評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀(guān)的道長(zhǎng)。 經(jīng)常有香客問(wèn)我箩朴,道長(zhǎng)岗喉,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,709評(píng)論 1 294
  • 正文 為了忘掉前任炸庞,我火速辦了婚禮钱床,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘埠居。我一直安慰自己查牌,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,733評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布滥壕。 她就那樣靜靜地躺著纸颜,像睡著了一般。 火紅的嫁衣襯著肌膚如雪绎橘。 梳的紋絲不亂的頭發(fā)上胁孙,一...
    開(kāi)封第一講書(shū)人閱讀 51,578評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音称鳞,去河邊找鬼涮较。 笑死,一個(gè)胖子當(dāng)著我的面吹牛胡岔,可吹牛的內(nèi)容都是我干的法希。 我是一名探鬼主播,決...
    沈念sama閱讀 40,320評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼靶瘸,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼苫亦!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起怨咪,我...
    開(kāi)封第一講書(shū)人閱讀 39,241評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤屋剑,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后诗眨,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體唉匾,經(jīng)...
    沈念sama閱讀 45,686評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,878評(píng)論 3 336
  • 正文 我和宋清朗相戀三年匠楚,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了巍膘。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,992評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡芋簿,死狀恐怖峡懈,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情与斤,我是刑警寧澤肪康,帶...
    沈念sama閱讀 35,715評(píng)論 5 346
  • 正文 年R本政府宣布荚恶,位于F島的核電站,受9級(jí)特大地震影響磷支,放射性物質(zhì)發(fā)生泄漏谒撼。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,336評(píng)論 3 330
  • 文/蒙蒙 一雾狈、第九天 我趴在偏房一處隱蔽的房頂上張望廓潜。 院中可真熱鬧,春花似錦箍邮、人聲如沸茉帅。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,912評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)堪澎。三九已至,卻和暖如春味滞,著一層夾襖步出監(jiān)牢的瞬間樱蛤,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,040評(píng)論 1 270
  • 我被黑心中介騙來(lái)泰國(guó)打工剑鞍, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留昨凡,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,173評(píng)論 3 370
  • 正文 我出身青樓蚁署,卻偏偏與公主長(zhǎng)得像便脊,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子光戈,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,947評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容