spring實(shí)現(xiàn)bean對象創(chuàng)建代碼詳解
我以一個簡單的示例解構(gòu)spring是怎樣管理java對象的。
首先,定義一個簡單的pojo,代碼如下:
package com.jvk.ken.spring;
public class Demo {
private String name;
public Demo() {
name="I'm Demo.";
}
public void printName() {
System.out.println(name);
}
public void setName(String name) {
this.name = name;
}
}
對應(yīng)的spring配置文件如下:
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation=" http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.1.xsd http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.1.xsd"> <bean id="demo" class="com.jvk.ken.spring.DemoFactory" /> </beans>
簡單的測試代碼如下:
package com.jvk.ken.spring;
import org.springframework.beans.factory.BeanFactory;
import org.springframework.beans.factory.xml.XmlBeanFactory;
import org.springframework.core.io.ClassPathResource;
public class Test {
public static void main(String[] args) throws Exception {
testSpring();
}
private static void testSpring() throws Exception {
BeanFactory bf = new XmlBeanFactory(new ClassPathResource("applicationContext.xml"));
Demo bean = (Demo) bf.getBean("demo");
System.out.println(bean.getClass());
bean.printName();
}
}
運(yùn)行Test類,輸出如下信息,說明一個簡單的spring示例成功運(yùn)行了。
2012-3-28 22:18:07 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions 信息: Loading XML bean definitions from class path resource [applicationContext.xml] class com.jvk.ken.spring.Demo I'm Demo.
從簡短的Java代碼和xml配置文件可知,XmlBeanFactory通過讀取xml配置文件組裝javabean,當(dāng)用戶調(diào)用getBean方法時返回所需對象。為了模仿它的行為,我定義一個簡單的beanFactory。
package com.jvk.ken.spring;
import java.util.HashMap;
import java.util.Map;
public class MyBeanFactory {
// 保存bean的定義
Map<String, Class> beans = new HashMap<String, Class>();
public Object getBean(String id) throws InstantiationException,
IllegalAccessException {
return beans.get(id).newInstance();
}
private String xmlFile;
public MyBeanFactory(String xmlFile) throws ClassNotFoundException {
super();
this.xmlFile = xmlFile;
init();
}
private void init() throws ClassNotFoundException {
// 初始化與解析XML,這里略去實(shí)際解析XML的情況,使用硬編碼模仿
System.out.println("配置文件:"+xmlFile);
String className = "com.jvk.ken.spring.Demo";
Class<?> loadClass = this.getClass().getClassLoader().loadClass(
className);
beans.put("demo", loadClass);
}
}
測試代碼如下:
package com.jvk.ken.spring;
public class Test {
public static void main(String[] args) throws Exception {
testNotSpring();
}
private static void testNotSpring() throws Exception {
MyBeanFactory bf = new MyBeanFactory("applicationContext.xml");
Demo bean = (Demo) bf.getBean("demo");
System.out.println(bean.getClass());
bean.printName();
}
}
運(yùn)行后輸出如下信息:
配置文件:applicationContext.xml class com.jvk.ken.spring.Demo I'm Demo.
以上短簡代碼展現(xiàn)了spring是怎樣充當(dāng)起最簡單的bean工廠。下面稍微調(diào)整一下代碼,分析一下spring會出現(xiàn)怎樣的現(xiàn)象。首先把Demo類的無參構(gòu)造方法改成private。
private Demo() {
name="I'm Demo.";
}
運(yùn)行測試代碼發(fā)現(xiàn),spring測試結(jié)果沒有任何差別,但我自定義的MyBeanFactory就報了如下錯誤信息:
Exception in thread "main" java.lang.IllegalAccessException: Class com.jvk.ken.spring.MyBeanFactory can not access a member of class com.jvk.ken.spring.Demo with modifiers "private" at sun.reflect.Reflection.ensureMemberAccess(Reflection.java:65) at java.lang.Class.newInstance0(Class.java:349) at java.lang.Class.newInstance(Class.java:308) at com.jvk.ken.spring.MyBeanFactory.getBean(MyBeanFactory.java:12) at com.jvk.ken.spring.Test.testNotSpring(Test.java:25) at com.jvk.ken.spring.Test.main(Test.java:9)
spring如此神奇?非也,是我寫的代碼過于簡陋而已,稍作修改,也是可以直接運(yùn)行的。
public Object getBean(String id) throws Exception {
Class class1 = beans.get(id);
Constructor declaredConstructor = class1.getDeclaredConstructor();
declaredConstructor.setAccessible(true);
return declaredConstructor.newInstance();
}
以上是spring容器管理的最純粹的javabean。spring還支持另外一種bean,叫工廠bean,示例勝千言,請看代碼
package com.jvk.ken.spring;
import org.springframework.beans.factory.FactoryBean;
public class DemoFactory implements FactoryBean {
@Override
public Object getObject() throws Exception {
return new Demo();
}
@Override
public Class getObjectType() {
return Demo.class;
}
@Override
public Boolean isSingleton() {
return false;
}
}
增加了DemoFactory類后,同時修改spring的配置文件
<bean id="demo" class="com.jvk.ken.spring.DemoFactory" />
其它代碼不作修改,運(yùn)行測試代碼后,輸出結(jié)果和原來完全一致。為什么明明配置了ID為demo的class為com.jvk.ken.spring.DemoFactory,返回的結(jié)果卻是Demo實(shí)例呢,這是因為spring檢測到DemoFactory是一種實(shí)現(xiàn)了FactoryBean接口的特殊bean,返回結(jié)果前會調(diào)用getObject方法,所以最后得到的是Demo對象。當(dāng)然,如果我們真的需要得到工廠bean,可以這樣寫bf.getBean("&demo")。
總結(jié)
以上就是本文關(guān)于spring實(shí)現(xiàn)bean對象創(chuàng)建代碼詳解的全部內(nèi)容,希望對大家有所幫助。感興趣的朋友可以繼續(xù)參閱本站其他相關(guān)專題,如有不足之處,歡迎留言指出。感謝朋友們對本站的支持!
相關(guān)文章
Java發(fā)送form-data請求的實(shí)例代碼
在Java中發(fā)送form-data請求,可以使用Apache?HttpClient或OkHttp這樣的HTTP客戶端庫來發(fā)送請求,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友參考下吧2023-10-10
Spring使用@Autowired注解靜態(tài)實(shí)例對象方式
這篇文章主要介紹了Spring使用@Autowired注解靜態(tài)實(shí)例對象方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-08-08
idea配置tomcat,idea配置web下lib的包詳解
這篇文章主要介紹了idea配置tomcat,idea配置web下lib的包,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-05-05
如何在Spring Boot應(yīng)用程序中配置了兩個不同的SOAP Web服務(wù)端點(diǎn)
這篇文章主要介紹了如何在Spring Boot應(yīng)用程序中配置了兩個不同的SOAP Web服務(wù)端點(diǎn),本文通過示例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-08-08
java通過isAccessAllowed方法實(shí)現(xiàn)訪問控制
在Web應(yīng)用開發(fā)中,使用Apache Shiro框架的isAccessAllowed方法可以有效管理用戶的訪問權(quán)限,本文詳細(xì)解析了該方法的實(shí)現(xiàn)過程,包括用戶身份驗證、權(quán)限判斷和安全性分析,下面就一起來了解一下2024-09-09
學(xué)習(xí)Java模擬實(shí)現(xiàn)百度文檔在線瀏覽
這片文章介紹了如何使用Java模擬實(shí)現(xiàn)百度文檔在線瀏覽,文章思路清晰,需要的朋友可以參考下2015-07-07

