不少java開發者在使用Spring框架中都見事後綴爲FactoryBean的類,好比Mybatis-Spring中的SqlSessionFactoryBean。說到這裏就不得不提BeanFactory。FactoryBean和BeanFactory特別容易讓人混淆,面試還常常問到這兩種概念。其實它們的做用和使用場景是不同的java
先來講說BeanFactory。 用於訪問Spring bean容器的根接口。這是Spring bean容器的基本客戶端視圖。原來是獲取Spring Bean的接口,也就是IoC容器。而後咱們看類圖面試
原來咱們更經常使用的ApplicationContext就是一個BeanFactory。咱們經過bean的名稱或者類型均可以從BeanFactory來獲取bean。對於BeanFactory這麼介紹相信都不陌生了。讓咱們把關注點轉向FactoryBean上。框架
FactoryBean 是個什麼玩意兒呢?來看看源碼ide
public interface FactoryBean<T> {
@Nullable
T getObject() throws Exception;
@Nullable
Class<?> getObjectType();
default boolean isSingleton() {
return true;
}
}
複製代碼
FactoryBean 用來建立一類bean。好比你有一些同屬鳥類的bean須要被建立,可是它們本身有各自的特色,你只須要把他們的特色注入FactoryBean中就能夠生產出各類鳥類的實例。舉一個更加貼近實際生產的例子。甚至這個例子你能夠應用到實際java開發中去。咱們須要本身造一個定時任務的輪子。用FactoryBean 再合適不過了。咱們來用代碼說話一步步來演示FactoryBean的使用場景。測試
咱們聲明定時任務通常具備下列要素:this
Task任務執行抽象接口的實現。實現包含兩個方面:spa
public class CustomTask implements Task {
private SomeService someService;
private String cronExpression;
public CustomTask(SomeService someService) {
this.someService = someService;
}
@Override
public void execute() {
//do something
someService.doTask();
}
@Override
public void setCron(String cronExpression) {
this.cronExpression = cronExpression;
}
@Override
public String getCron() {
return cronExpression;
}
}
複製代碼
經過以上的定義。任務的時間和任務的邏輯能夠根據不一樣的業務作到差別化配置。而後咱們實現一個關於Task的FactoryBean。debug
public class TaskFactoryBean implements FactoryBean<Task> {
private SomeService someService;
private String cronExpression;
@Override
public Task getObject() throws Exception {
CustomTask customTask = new CustomTask(someService);
customTask.setCron(cronExpression);
return customTask;
}
@Override
public Class<?> getObjectType() {
return CustomTask.class;
}
@Override
public boolean isSingleton() {
return true;
}
public SomeService getSomeService() {
return someService;
}
public void setSomeService(SomeService someService) {
this.someService = someService;
}
public String getCronExpression() {
return cronExpression;
}
public void setCronExpression(String cronExpression) {
this.cronExpression = cronExpression;
}
}
複製代碼
你可使用xml的注入方式,固然也可使用javaConfig的配置方式。這裏咱們使用javaConfig注入。咱們將兩個FactroyBean注入到Spring容器中去。code
@Configuration
public class Config {
@Bean
public TaskFactoryBean customTask() {
TaskFactoryBean taskFactoryBean = new TaskFactoryBean();
taskFactoryBean.setCronExpression("0 15 10 * * ?");
String word = "定時任務一";
SomeService someService = new SomeService();
someService.setWord(word);
taskFactoryBean.setSomeService(someService);
return taskFactoryBean;
}
@Bean
public TaskFactoryBean otherTask() {
TaskFactoryBean taskFactoryBean = new TaskFactoryBean();
taskFactoryBean.setCronExpression("0 15 17 * * ?");
String word = "定時任務二";
SomeService someService = new SomeService();
someService.setWord(word);
taskFactoryBean.setSomeService(someService);
return taskFactoryBean;
}
}
複製代碼
通常如上聲明後,@Bean註解若是不顯式聲明bean名稱則方法名做爲bean的名稱,並且返回值做爲注入的Bean。可是咱們經過debug發現倒是這樣的:cdn
也就是說經過方法名是返回FactoryBean 建立的Bean。那麼如何返回該FactoryBean呢?上圖中也給出了答案在方法前增長引用符「&」。具體的緣由還用從BeanFactory中尋找,真是否是冤家不聚頭
咱們對上面聲明的兩個bean進行測試,也出色地完成了不一樣的定時任務業務邏輯。
@Autowired
private Task customTask;
@Autowired
private Task otherTask;
@Test
public void task() {
customTask.execute();
otherTask.execute();
}
複製代碼
在後續的使用中你能夠經過聲明不一樣的cron表達式,以及不一樣SomeService來定製更多的定時任務。經過這個例子相信你會對FactoryBean有的清晰的認識。demo就不提供了,很是簡單,強烈建議你本身試一試以加深理解。
關注公衆號:碼農小胖哥,獲取更多資訊