1、獲取bean實例的三種方式 1.1 id 屬性 1.1.1 jar <properties> <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> <spring.version>4.3.18.RELEASE</sp ...
1、獲取bean實例的三種方式
1.1 id 屬性
1.1.1 jar
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<spring.version>4.3.18.RELEASE</spring.version>
<lombok.version>1.16.18</lombok.version>
<junit.version>4.11</junit.version>
</properties>
<dependencies>
<!-- spring-beans begin -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-beans</artifactId>
<version>${spring.version}</version>
</dependency>
<!-- spring-beans end -->
<!-- spring-core begin -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-core</artifactId>
<version>${spring.version}</version>
</dependency>
<!-- spring-core end -->
<!-- spring-context begin -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>${spring.version}</version>
</dependency>
<!-- spring-context end -->
<!-- spring-expression begin -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-expression</artifactId>
<version>${spring.version}</version>
</dependency>
<!-- spring-expression end -->
</dependencies>
1.1.2 application.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
<!-- 基於spring的核心配置文件方式,手動配置一個bean,放到spring的容器中 -->
<bean id="userOne" class="com.kgc.spring.bean.User">
<property name="nickName" value="kh96_spring_one"></property>
</bean>
</beans>
<bean>標簽:定義一個實例對象,會自動被創建 並 交給spring容器進行管理:
-
id 屬性:被spring容器進行管理的實例的 唯一標識,整個容器中,只能是唯一的(不可重覆);
-
class屬性:指定創建當前實例對象的類型(全類名),spring底層是使用的 反射機制 ,根據指定的目標類型,創建目標實例(必須有 空參構造)
-
<property>子標簽:給當前創建對象的屬性值
- name:指定對象的屬性名
- value: 給屬性賦值
1.1.3 測試
@Test
public void testSpringPrimer(){
//1.創建spring的核心容器對象(引用上下文對象),解析spring的核心配置文件,將文件中的bean標簽進行實例化(創建對象並賦值)
ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
//2. 寫法一: 從容器中獲取實例對象,根據核心配置文件中,配置的bean標簽的id屬性值
// 不足:強制轉化,容易出現轉換錯誤
User userOne = (User) context.getBean("userOne");
userOne.sayHello();
//對比以前創建對象的寫法(對象必須手動new創建,並手動賦值)
User userOld = new User();
userOld.setNickName("kh96_old");
userOld.sayHello();
}
1.1.4 總結
通過id屬性,獲取實例對象後需要強制轉換,容易出現強制轉化錯誤;
1.2 class 屬性
1.2.1 測試
//寫法二:從容器中獲取實例對象,根據核心配置文件中,配置的bean標簽的class屬性指定的目標類型
User userTwo = context.getBean(User.class);
userTwo.sayHello();
1.2.2 總結
不足:如果存在同類型的多個實例,會報異常;
1.2.2.1 application.xml
當用多個同類型的實例:
...
<bean id="userOne" class="com.kgc.spring.bean.User">
<property name="nickName" value="kh96_spring_one"></property>
</bean>
<!-- Configuration problem: Bean name 'userOne' is already used in this <beans> element -->
<bean id="userTwo" class="com.kgc.spring.bean.User">
<property name="nickName" value="kh96_spring_two"></property>
</bean>
...
1.2.2.2 測試
User userTwo = context.getBean(User.class);
會發生異常:
NoUniqueBeanDefinitionException: No qualifying bean of type 'com.kgc.spring.bean.User' available: expected single matching bean but found 2: userOne,userTwo
主要原因是由於只用class屬性去獲取實例,但是有多個相同類型的實例,所以無法確定你要獲取的是哪一個;
1.3 id 屬性 和 class 屬性 (推薦)
1.3.1 測試
//寫法三:從容器中獲取實例對象,根據配置文件中,配置的bean標簽的id屬性值和class屬性指定的目標類型
User userTwoThree = context.getBean("userTwo", User.class);
1.3.2 總結
能夠確定唯一實例,推薦使用;
1.4 id 不能重覆註意點
配置文件中的id必須是唯一的;
如果id不唯一:兩個id一樣的實例
...
<bean id="userOne" class="com.kgc.spring.bean.User">
<property name="nickName" value="kh96_spring_one"></property>
</bean>
<bean id="userOne" class="com.kgc.spring.bean.User">
<property name="nickName" value="kh96_spring_one"></property>
</bean>
...
報錯:
BeanDefinitionParsingException: Configuration problem: Bean name 'userOne' is already used in this <beans> element
提示 id 為 userOne 的實例已經存在;
1.5 實例化時機
當初始化spring的容器對象時,會將核心配置文件中所有的bean實例化,不是使用哪個,創建哪個;
2、DI
IOC(控制反轉是一種思想),DI是IOC的一種實現方式;
2.1 set 方式註入
2.1.1 實體
@Data
@AllArgsConstructor
@NoArgsConstructor
public class User {
//用戶昵稱
//普通參數
private String nickName;
//私家車
//實體參數
private Car car;
//喜歡的書
//數組參數
private String[] books;
//愛好
//list集合參數
private List<String> hobbies;
//喜歡的游戲
//set 集合
private Set<String> games;
//卡包
//map參數
private Map<String,String> cards;
//在校信息
//properties參數
private Properties info;
//是否有女朋友 (主要是演示賦值為null,其他類型都可以)
//空參數註入
private Boolean wife;
}
2.1.2 參數 註入
...
<!-- 不同類型參數的set註入 -->
<bean id="user01" class="com.kgc.spring.bean.User">
<!-- 1.普通參數 註入 -->
<property name="nickName" value="小氣鬼"></property>
<!-- 2.實體參數 註入 -->
<!-- 2.1 實體參數 註入方式一 外部引入 ref -->
<!-- <property name="car" ref="carOne"></property>-->
<property name="car">
<!-- 2.2 實體參數註入方式二 內部註入 <bean> -->
<bean class="com.kgc.spring.bean.Car">
<property name="brand" value="Bmw325"></property>
<property name="factory" value="華晨"></property>
<property name="price" value="300000"></property>
</bean>
</property>
<!-- 3.數組參數 註入 -->
<property name="books">
<array>
<value>紅樓夢</value>
<value>水滸傳</value>
<value>三國演義</value>
<value>西游記</value>
</array>
</property>
<!-- 4.list集合參數 -->
<property name="hobbies" >
<list>
<value>跑步</value>
<value>尤克裡里</value>
<value>敲代碼</value>
</list>
</property>
<!-- 5.set 集合參數 註入 -->
<property name="games">
<set>
<value>唱歌</value>
<value>跳舞</value>
<value>喝酒</value>
</set>
</property>
<!-- 6.map參數 -->
<property name="cards">
<map>
<entry key="身份證" value="1212121212121212"></entry>
<entry key="銀行卡" value="1111111111111111"></entry>
</map>
</property>
<!-- 7.properties參數 -->
<property name="info">
<props>
<prop key="學號">123456</prop>
<prop key="性別">男</prop>
<prop key="姓名">化羽</prop>
</props>
</property>
<!-- 8.空參數註入 -->
<property name="wife">
<null></null>
</property>
</bean>
<bean id="carOne" class="com.kgc.spring.bean.Car">
<property name="brand" value="Bmw325"></property>
<property name="factory" value="華晨"></property>
<property name="price" value="300000"></property>
</bean>
...
2.1.3 測試
@Test
public void testPramsDI(){
User user01 = context.getBean("user01", User.class);
//輸入對象詳情
System.out.println(user01);
}
輸出結果:
User(
nickName=小氣鬼,
car=Car(brand=Bmw325, factory=華晨, price=300000.0),
books=[紅樓夢, 水滸傳, 三國演義, 西游記],
hobbies=[跑步, 尤克裡里, 敲代碼],
games=[唱歌, 跳舞, 喝酒],
cards={身份證=1212121212121212, 銀行卡=1111111111111111},
info={學號=1913001072, 性別=男, 姓名=化羽},
wife=null
)
2.2 Constructor 構造器註入
2.2.1 按照預設順序 註入
2.2.1.1 參數註入
<!--
實體構造方法:Car(String brand, String factory, Double price)
-->
<!-- 構造器註入:通過構造方法,預設是按照構造方法的參數定義順序賦值 -->
<bean id="carTwo" class="com.kgc.spring.bean.Car">
<constructor-arg value="AudiA4"></constructor-arg>
<constructor-arg value="一汽"></constructor-arg>
<constructor-arg value="320000"></constructor-arg>
</bean>
2.2.1.2 測試
@Test
public void testSpringDIConstructor(){
Car carTwo = context.getBean("carTwo", Car.class);
//輸出對象詳情
System.out.println(carTwo);
}
2.2.2 根據 參數的下標 和 類型 註入
2.2.2.1 參數註入
<!-- 根據構造器中參數的下標 和 類型 賦值 -->
<bean id="carThree" class="com.kgc.spring.bean.Car">
<constructor-arg index="0" value="BenzC200"></constructor-arg>
<constructor-arg index="1" value="北京"></constructor-arg>
<constructor-arg index="2" type="java.lang.Double" value="350000"></constructor-arg>
<!--
如果參數列表中,該類型的參數只用一個,也可以只指定參數類型
<constructor-arg type="java.lang.Double" value="350000"></constructor-arg>
-->
</bean>
2.2.2.2 測試
@Test
public void testSpringDIConstructor2(){
Car carTwo = context.getBean("carThree", Car.class);
//輸入對象詳情
System.out.println(carTwo);
}
2.3 自定義 實體工廠bean
自定義實體工廠bean ,必須實現FactoryBean介面;
普通bean 與 工廠bean 的區別:
- 普通 bean:在配置文件中定義 bean 類型 就是 返回類型
- 工廠 bean:在配置文件定義 bean 類型 和 返回類型 不一樣
2.3.1 返回bean
@Data
@ToString
public class Car {
/*
品牌
*/
private String brand;
/*
廠商
*/
private String factory;
/*
價格
*/
private Double price;
}
2.3.2 實體工廠 bean
- 在spring容器初始化時,創建當前工廠bean的實例對象,但是真實返回的不是當前類的實例對象,而是當前類的實例對象返回的目標實例對象(自定義);
- 作用:可以在程式中,實現自定義創建實例(還可以增加業務邏輯處理),放入容器;
- 存在的原因:spring框架對外開放了一個入口,可以讓其他的開發人員參與到spring底層創建bean的實例過程中去,給整合其他框架使用的,比如mybatis;
public class CarFactoryBean implements FactoryBean<Car> {
@Override
public Car getObject() throws Exception {
System.out.println("通過CarFactoryBean的實例對象的getObject方法:返回一個自定義的car的實例");
return new Car("Byd唐","南京",2500000.0);
}
@Override
public Class<?> getObjectType() {
//指定給Object方法返回的目標類型
return Class.class;
}
@Override
public boolean isSingleton() {
//是否單例
return false;
}
}
2.3.3 工廠bean容器中添加自定義實例對象
...
<!-- 使用工廠bean容器,添加自定義實例對象 -->
<bean id="carFactoryBean" class="com.kgc.spring.bean.CarFactoryBean"></bean>
...
2.3.4 測試
@Test
public void testSpringFactoryBean(){
//從容器中獲取工廠bean的實例對象
Car car1 = context.getBean("carFactoryBean", Car.class);
Car car2 = context.getBean("carFactoryBean", Car.class);
//輸出對象詳情
System.out.println(car 1);
//在容器中添加的實例是 CarFactoryBean 類型,返回的是
//Car(brand=Byd唐, factory=南京, price=2500000.0)
System.out.println("car1 == car2 : " + ( car1 == car2));
//false 說明工廠bean是多例的
}
3、scope作用域
- singleton 單例 (預設) 容器初始化之前創建;
- prototype 多例 (手動設置) 使用到的時候才創建;
3.1 singleton 單例
這裡區別於,整個程式運行期間,有且只有唯一的實例(單例模式-懶漢和餓漢);
而容器中bean的單例,不是指當前類的實例在容器中,只有唯一的實例,而是當創建bean的實例時,此實例是單例(容器內唯一),但是同一個類的實例,容器中可以創建多個,每個都是單例的;
3.1.1 配置bean
<!-- scope="singleton" 不寫預設也是單例的 -->
<bean id="busTwo" class="com.kgc.spring.bean.Bus" scope="singleton">
<property name="brand" value="金龍2"></property>
<property name="factory" value="廈門2"></property>
<property name="price" value="1200000"></property>
</bean>
3.1.2 測試
@Test
public void testScopeSingleton(){
//從容器中,獲取Bus的實例對象
Bus busThree = context.getBean("busTwo", Bus.class);
Bus busFour = context.getBean("busTwo", Bus.class);
System.out.println(busThree);
System.out.println(busFour);
System.out.println("singleton busThree == busFour:"+(busThree == busFour));
//true
}
3.2 prototype 多例
多例:prototype,不是spring容器中的預設作用,需要單獨指定;
spring容器創建時,不會自動創建指定作用域為多例的bean的實例,而是每次通過getBean方法,獲取bean的實例,才會創建bean的實例;
3.2.1 配置bean
<!-- scope="prototype" 多例需要手動設置 -->
<bean id="busThree" class="com.kgc.spring.bean.Bus" scope="prototype">
<property name="brand" value="中通3"></property>
<property name="factory" value="山東3"></property>
<property name="price" value="1200000"></property>
</bean>
3.2.2 測試
@Test
public void testScopePrototype(){
Bus busOne = context.getBean("busThree", Bus.class);
Bus busTwo = context.getBean("busThree", Bus.class);
//輸入詳情
System.out.println(busOne);
System.out.println(busTwo);
System.out.println("prototype busOne == busTwo:"+(busOne == busTwo));
//false
}