spring的bean工廠
Bean Factory
依賴注入的方式
1.通過註解的方式 2.通過構造方法的方式 3.通過Application.getBean(引數) 的方式
通過註解的方式
常用註解@Autowire在介面上來注入;
通過構造方法的方式
private TaxDetailInfoRepository taxDetailInfoRepository;
private HscsHspRemoteFeign hscsHspRemoteFeign;
private HpfmRemoteFeign hpfmRemoteFeign;
public TaxDetailInfoServiceImpl(TaxDetailInfoRepository taxDetailInfoRepository, HscsHspRemoteFeign hscsHspRemoteFeign, HpfmRemoteFeign hpfmRemoteFeign) { this.taxDetailInfoRepository = taxDetailInfoRepository; this.hscsHspRemoteFeign = hscsHspRemoteFeign; this.hpfmRemoteFeign = hpfmRemoteFeign; }
通過Application.getBean(引數)
我們知道可以通過ApplicationContext的getBean方法來獲取Spring容器中已初始化的bean。
使用示例:
ItfDomainValidatingServiceImpl selfProxy = ApplicationContextHelper.getContext().getBean(this.getClass());
getBean一共有以下四種方法原型:
下來我們分別來探討以上四種方式獲取bean的區別。
(1)getBean(String name)
引數name表示IOC容器中已經例項化的bean的id或者name,且無論是id還是name都要求在IOC容器中是唯一的不能重名。那麼這種方法就是通過id或name去查詢獲取bean.
(2)getBean(Class type)
引數Class type表示要載入的Bean的型別。如果該型別沒有繼承任何父類(Object類除外)和實現介面的話,那麼要求該型別的bean在IOC容器中也必須是唯一的。比如applicationContext.xml配置兩個型別完全一致的bean,且都沒有配置id和name屬性
我們可以總結getBean(String name)和getBean(Class type)的異同點。
相同點:都要求id或者name或者型別在容器中的唯一性。
不同點:getBean(String name)獲得的物件需要型別轉換而getBean(Class type)獲得的物件無需型別轉換。
(3)getBean(String name,Class type)
這種方式比較適合當型別不唯一時,再通過id或者name來獲取bean。
(4)getBean(String name,Object[] args)
這種方式本質還是通過bean的id或者name來獲取bean,通過第二個引數Object[] args可以給bean的屬性賦值,賦值的方式有兩種:構造方法和工廠方法。但是通過這種方式獲取的bean必須把scope屬性設定為prototype,也就是非單例模式。
以下舉例說明
用SpringContextUtil實現 ApplicationContextAware
package util;
import java.util.Locale;
import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
public class SpringContextUtil
implements ApplicationContextAware
{
private static ApplicationContext context;
@Override
public void setApplicationContext(ApplicationContext contex)
throws BeansException
{
System.out.println("--------------------contex---------"+contex);
SpringContextUtil.context = contex;
}
public static ApplicationContext getApplicationContext() {
return context;
}
public static Object getBean(String beanName) {
return context.getBean(beanName);
}
public static String getMessage(String key) {
return context.getMessage(key, null, Locale.getDefault());
}
}
工具類
package redis;
import redis.clients.jedis.JedisPool;
import util.SpringContextUtil;
public class RedisUtil {
private static JedisPool jedisPool;
static{
jedisPool = (JedisPool)SpringContextUtil.getBean("jedisPool");
}
public static JedisPool getJedisPool(){
if(jedisPool == null){
jedisPool = (JedisPool)SpringContextUtil.getBean("jedisPool");
}
return jedisPool;
}
public void flusDB(){
jedisPool.getResource().flushDB();
}
public static String set(String key,String value){
return jedisPool.getResource().set(key, value);
}
public static String get(String key){
return jedisPool.getResource().get(key);
}
public static Long del(String key){
return jedisPool.getResource().del(key);
}
}
在Spring的配置檔案中配置這個類,Spring容器會在載入完Spring容器後把上下文物件呼叫這個物件中的setApplicationContext方法
<context:component-scan base-package=“com.first,com.util” />
classpath:jdbc.properties classpath:redis.properties
在web專案中的web.xml中配置載入Spring容器的Listener
<property name="maxActive" value="200" />
<property name="minIdle" value="10"/>
<property name="maxWait" value="300" />
<property name="testOnReturn" value="true" />
<property name="testWhileIdle" value="true" />
</bean>
<bean id="jedisPool" class="redis.clients.jedis.JedisPool">
<constructor-arg name="poolConfig" ref="jedisPoolConfig" />
<constructor-arg name="host" value="${redis_addr}" />
<constructor-arg name="port" value="${redis_port}" type="int" />
<constructor-arg name="timeout" value="${redis_timeout}" type="int" />
<constructor-arg name="password" value="#{'${redis_password}'!=''?'${redis_password}':null}" />
<constructor-arg name="database" value="${redis_db_index}" type="int" />
</bean>