av一区二区在线观看_亚洲男人的天堂网站_日韩亚洲视频_在线成人免费_欧美日韩精品免费观看视频_久草视

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

關(guān)于Spring Bean實(shí)例過(guò)程中使用反射和遞歸處理的Bean屬性填充問(wèn)題

瀏覽:5日期:2023-07-12 15:33:55
目錄一、前言二、目標(biāo)三、設(shè)計(jì)四、實(shí)現(xiàn)五、測(cè)試六、總結(jié)一、前言

超賣(mài)、掉單、冪等,你的程序總是不抗揍!

想想,運(yùn)營(yíng)已經(jīng)對(duì)外宣傳了七八天的活動(dòng),滿心歡喜的等著最后一天頁(yè)面上線對(duì)外了,突然出現(xiàn)了一堆異常、資損、閃退,而用戶(hù)流量稍縱即逝,最后想死的心都有!

就編程開(kāi)發(fā)來(lái)講,丟三落四、亂碼七糟,可能這就是大部分初級(jí)程序員日常開(kāi)發(fā)的真實(shí)寫(xiě)照,在即使有測(cè)試人員驗(yàn)證的情況下,也會(huì)出現(xiàn)帶Bug上線的現(xiàn)象,只不過(guò)是當(dāng)時(shí)沒(méi)有發(fā)現(xiàn)而已!因?yàn)槭侨藢?xiě)代碼,就一定會(huì)有錯(cuò)誤,即使是老碼農(nóng)

就程序Bug來(lái)講,會(huì)包括產(chǎn)品PRD流程上的Bug、運(yùn)營(yíng)配置活動(dòng)時(shí)候的Bug、研發(fā)開(kāi)發(fā)時(shí)功能實(shí)現(xiàn)的Bug、測(cè)試驗(yàn)證時(shí)漏掉流程的Bug、上線過(guò)程中運(yùn)維服務(wù)相關(guān)配置的Bug,而這些其實(shí)都可以通過(guò)制定的流程規(guī)范和一定的研發(fā)經(jīng)驗(yàn)積累,慢慢盡可能減少。

而另外一類(lèi)是溝通留下的Bug,通常情況下業(yè)務(wù)提需求、產(chǎn)品定方案、研發(fā)做實(shí)現(xiàn),最終還要有UI、測(cè)試、運(yùn)營(yíng)、架構(gòu)等等各個(gè)環(huán)節(jié)的人員參與到一個(gè)項(xiàng)目的承接、開(kāi)發(fā)到上線運(yùn)行,而在這一群人需要保持一個(gè)統(tǒng)一的信息傳播其實(shí)是很難的。比如在項(xiàng)目開(kāi)發(fā)中期,運(yùn)營(yíng)給產(chǎn)品說(shuō)了一個(gè)新增的需求,產(chǎn)品覺(jué)得功能也不大,隨即找到對(duì)應(yīng)的前端研發(fā)加個(gè)邏輯,但沒(méi)想到可能也影響到了后端的開(kāi)發(fā)和測(cè)試的用例。最后功能雖然是上線了,可并不在整個(gè)產(chǎn)研測(cè)的需求覆蓋度范圍里,也就隱形的埋下了一個(gè)坑。

所以,如果你想讓你的程序很抗揍,接的住農(nóng)夫三拳,那么你要做的就不只是一個(gè)單純的搬磚碼農(nóng)!

二、目標(biāo)

首先我們回顧下這幾章節(jié)都完成了什么,包括:實(shí)現(xiàn)一個(gè)容器、定義和注冊(cè)Bean、實(shí)例化Bean,按照是否包含構(gòu)造函數(shù)實(shí)現(xiàn)不同的實(shí)例化策略,那么在創(chuàng)建對(duì)象實(shí)例化這我們還缺少什么?其實(shí)還缺少一個(gè)關(guān)于類(lèi)中是否有屬性的問(wèn)題,如果有類(lèi)中包含屬性那么在實(shí)例化的時(shí)候就需要把屬性信息填充上,這樣才是一個(gè)完整的對(duì)象創(chuàng)建。

對(duì)于屬性的填充不只是 int、Long、String,還包括還沒(méi)有實(shí)例化的對(duì)象屬性,都需要在 Bean 創(chuàng)建時(shí)進(jìn)行填充操作。不過(guò)這里我們暫時(shí)不會(huì)考慮 Bean 的循環(huán)依賴(lài),否則會(huì)把整個(gè)功能實(shí)現(xiàn)撐大,這樣新人學(xué)習(xí)時(shí)就把握不住了,待后續(xù)陸續(xù)先把核心功能實(shí)現(xiàn)后,再逐步完善

三、設(shè)計(jì)

鑒于屬性填充是在 Bean 使用 newInstance 或者 Cglib 創(chuàng)建后,開(kāi)始補(bǔ)全屬性信息,那么就可以在類(lèi) AbstractAutowireCapableBeanFactory 的 createBean 方法中添加補(bǔ)全屬性方法。這部分大家在實(shí)習(xí)的過(guò)程中也可以對(duì)照Spring源碼學(xué)習(xí),這里的實(shí)現(xiàn)也是Spring的簡(jiǎn)化版,后續(xù)對(duì)照學(xué)習(xí)會(huì)更加易于理解

關(guān)于Spring Bean實(shí)例過(guò)程中使用反射和遞歸處理的Bean屬性填充問(wèn)題

屬性填充要在類(lèi)實(shí)例化創(chuàng)建之后,也就是需要在 AbstractAutowireCapableBeanFactory 的 createBean 方法中添加 applyPropertyValues 操作。 由于我們需要在創(chuàng)建Bean時(shí)候填充屬性操作,那么就需要在 bean 定義 BeanDefinition 類(lèi)中,添加 PropertyValues 信息。 另外是填充屬性信息還包括了 Bean 的對(duì)象類(lèi)型,也就是需要再定義一個(gè) BeanReference,里面其實(shí)就是一個(gè)簡(jiǎn)單的 Bean 名稱(chēng),在具體的實(shí)例化操作時(shí)進(jìn)行遞歸創(chuàng)建和填充,與 Spring 源碼實(shí)現(xiàn)一樣。Spring 源碼中 BeanReference 是一個(gè)接口四、實(shí)現(xiàn)

1. 工程結(jié)構(gòu)

small-spring-step-04└── src ├── main │ └── java │ └── cn.bugstack.springframework.beans │ ├── factory │ │ ├── factory │ │ │ ├── BeanDefinition.java │ │ │ ├── BeanReference.java │ │ │ └── SingletonBeanRegistry.java │ │ ├── support │ │ │ ├── AbstractAutowireCapableBeanFactory.java │ │ │ ├── AbstractBeanFactory.java │ │ │ ├── BeanDefinitionRegistry.java │ │ │ ├── CglibSubclassingInstantiationStrategy.java │ │ │ ├── DefaultListableBeanFactory.java │ │ │ ├── DefaultSingletonBeanRegistry.java │ │ │ ├── InstantiationStrategy.java │ │ │ └── SimpleInstantiationStrategy.java │ │ └── BeanFactory.java │ ├── BeansException.java │ ├── PropertyValue.java │ └── PropertyValues.java └── test└── java └── cn.bugstack.springframework.test├── bean│ ├── UserDao.java│ └── UserService.java└── ApiTest.java

工程源碼:

《Spring 手?jǐn)]專(zhuān)欄》學(xué)習(xí)源碼介紹專(zhuān)欄地址:https://bugstack.cn/itstack/spring.html

源碼地址:https://github.com/fuzhengwei/small-spring

Spring Bean 容器類(lèi)關(guān)系,如圖 5-2

關(guān)于Spring Bean實(shí)例過(guò)程中使用反射和遞歸處理的Bean屬性填充問(wèn)題

本章節(jié)中需要新增加3個(gè)類(lèi),BeanReference(類(lèi)引用)、PropertyValue(屬性值)、PropertyValues(屬性集合),分別用于類(lèi)和其他類(lèi)型屬性填充操作。 另外改動(dòng)的類(lèi)主要是 AbstractAutowireCapableBeanFactory,在 createBean 中補(bǔ)全屬性填充部分。

2. 定義屬性

cn.bugstack.springframework.beans.PropertyValue

public class PropertyValue { private final String name; private final Object value; public PropertyValue(String name, Object value) {this.name = name;this.value = value; }// ...get/set}

cn.bugstack.springframework.beans.PropertyValues

public class PropertyValues { private final List<PropertyValue> propertyValueList = new ArrayList<>(); public void addPropertyValue(PropertyValue pv) {this.propertyValueList.add(pv); } public PropertyValue[] getPropertyValues() {return this.propertyValueList.toArray(new PropertyValue[0]); } public PropertyValue getPropertyValue(String propertyName) {for (PropertyValue pv : this.propertyValueList) { if (pv.getName().equals(propertyName)) {return pv; }}return null; }}

這兩個(gè)類(lèi)的作用就是創(chuàng)建出一個(gè)用于傳遞類(lèi)中屬性信息的類(lèi),因?yàn)閷傩钥赡軙?huì)有很多,所以還需要定義一個(gè)集合包裝下。

3. Bean定義補(bǔ)全

cn.bugstack.springframework.beans.factory.config.BeanDefinition

public class BeanDefinition { private Class beanClass; private PropertyValues propertyValues; public BeanDefinition(Class beanClass) {this.beanClass = beanClass;this.propertyValues = new PropertyValues(); } public BeanDefinition(Class beanClass, PropertyValues propertyValues) {this.beanClass = beanClass;this.propertyValues = propertyValues != null ? propertyValues : new PropertyValues(); }// ...get/set} 在 Bean 注冊(cè)的過(guò)程中是需要傳遞 Bean 的信息,在幾個(gè)前面章節(jié)的測(cè)試中都有所體現(xiàn) new BeanDefinition(UserService.class, propertyValues); 所以為了把屬性一定交給 Bean 定義,所以這里填充了 PropertyValues 屬性,同時(shí)把兩個(gè)構(gòu)造函數(shù)做了一些簡(jiǎn)單的優(yōu)化,避免后面 for 循環(huán)時(shí)還得判斷屬性填充是否為空。

4. Bean 屬性填充

cn.bugstack.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory

public abstract class AbstractAutowireCapableBeanFactory extends AbstractBeanFactory { private InstantiationStrategy instantiationStrategy = new CglibSubclassingInstantiationStrategy(); @Override protected Object createBean(String beanName, BeanDefinition beanDefinition, Object[] args) throws BeansException {Object bean = null;try { bean = createBeanInstance(beanDefinition, beanName, args); // 給 Bean 填充屬性 applyPropertyValues(beanName, bean, beanDefinition);} catch (Exception e) { throw new BeansException('Instantiation of bean failed', e);}addSingleton(beanName, bean);return bean; } protected Object createBeanInstance(BeanDefinition beanDefinition, String beanName, Object[] args) {Constructor constructorToUse = null;Class<?> beanClass = beanDefinition.getBeanClass();Constructor<?>[] declaredConstructors = beanClass.getDeclaredConstructors();for (Constructor ctor : declaredConstructors) { if (null != args && ctor.getParameterTypes().length == args.length) {constructorToUse = ctor;break; }}return getInstantiationStrategy().instantiate(beanDefinition, beanName, constructorToUse, args); } /** * Bean 屬性填充 */ protected void applyPropertyValues(String beanName, Object bean, BeanDefinition beanDefinition) {try { PropertyValues propertyValues = beanDefinition.getPropertyValues(); for (PropertyValue propertyValue : propertyValues.getPropertyValues()) {String name = propertyValue.getName();Object value = propertyValue.getValue();if (value instanceof BeanReference) { // A 依賴(lài) B,獲取 B 的實(shí)例化 BeanReference beanReference = (BeanReference) value; value = getBean(beanReference.getBeanName());}// 屬性填充BeanUtil.setFieldValue(bean, name, value); }} catch (Exception e) { throw new BeansException('Error setting property values:' + beanName);} } public InstantiationStrategy getInstantiationStrategy() {return instantiationStrategy; } public void setInstantiationStrategy(InstantiationStrategy instantiationStrategy) {this.instantiationStrategy = instantiationStrategy; }} 這個(gè)類(lèi)的內(nèi)容稍微有點(diǎn)長(zhǎng),主要包括三個(gè)方法:createBean、createBeanInstance、applyPropertyValues,這里我們主要關(guān)注 createBean 的方法中調(diào)用的 applyPropertyValues 方法。 在 applyPropertyValues 中,通過(guò)獲取 beanDefinition.getPropertyValues() 循環(huán)進(jìn)行屬性填充操作,如果遇到的是 BeanReference,那么就需要遞歸獲取 Bean 實(shí)例,調(diào)用 getBean 方法。 當(dāng)把依賴(lài)的 Bean 對(duì)象創(chuàng)建完成后,會(huì)遞歸回現(xiàn)在屬性填充中。這里需要注意我們并沒(méi)有去處理循環(huán)依賴(lài)的問(wèn)題,這部分內(nèi)容較大,后續(xù)補(bǔ)充。BeanUtil.setFieldValue(bean, name, value) 是 hutool-all 工具類(lèi)中的方法,你也可以自己實(shí)現(xiàn)五、測(cè)試

1. 事先準(zhǔn)備

cn.bugstack.springframework.test.bean.UserDao

public class UserDao { private static Map<String, String> hashMap = new HashMap<>(); static {hashMap.put('10001', '小傅哥');hashMap.put('10002', '八杯水');hashMap.put('10003', '阿毛'); } public String queryUserName(String uId) {return hashMap.get(uId); }}

cn.bugstack.springframework.test.bean.UserService

public class UserService { private String uId; private UserDao userDao; public void queryUserInfo() {System.out.println('查詢(xún)用戶(hù)信息:' + userDao.queryUserName(uId)); } // ...get/set}

Dao、Service,是我們平常開(kāi)發(fā)經(jīng)常使用的場(chǎng)景。在 UserService 中注入 UserDao,這樣就能體現(xiàn)出Bean屬性的依賴(lài)了。

2. 測(cè)試用例

@Testpublic void test_BeanFactory() { // 1.初始化 BeanFactory DefaultListableBeanFactory beanFactory = new DefaultListableBeanFactory(); // 2. UserDao 注冊(cè) beanFactory.registerBeanDefinition('userDao', new BeanDefinition(UserDao.class)); // 3. UserService 設(shè)置屬性[uId、userDao] PropertyValues propertyValues = new PropertyValues(); propertyValues.addPropertyValue(new PropertyValue('uId', '10001')); propertyValues.addPropertyValue(new PropertyValue('userDao',new BeanReference('userDao'))); // 4. UserService 注入bean BeanDefinition beanDefinition = new BeanDefinition(UserService.class, propertyValues); beanFactory.registerBeanDefinition('userService', beanDefinition);// 5. UserService 獲取bean UserService userService = (UserService) beanFactory.getBean('userService'); userService.queryUserInfo();} 與直接獲取 Bean 對(duì)象不同,這次我們還需要先把 userDao 注入到 Bean 容器中。beanFactory.registerBeanDefinition('userDao', new BeanDefinition(UserDao.class)); 接下來(lái)就是屬性填充的操作了,一種是普通屬性 new PropertyValue('uId', '10001'),另外一種是對(duì)象屬性 new PropertyValue('userDao',new BeanReference('userDao')) 接下來(lái)的操作就簡(jiǎn)單了,只不過(guò)是正常獲取 userService 對(duì)象,調(diào)用方法即可。

3. 測(cè)試結(jié)果

查詢(xún)用戶(hù)信息:小傅哥Process finished with exit code 0 從測(cè)試結(jié)果看我們的屬性填充已經(jīng)起作用了,因?yàn)橹挥袑傩蕴畛浜螅拍苷{(diào)用到Dao方法,如:userDao.queryUserName(uId) 那么我們?cè)诳纯碊ebug調(diào)試的情況下,有沒(méi)有進(jìn)入到實(shí)現(xiàn)的 Bean 屬性填充中,如下:

關(guān)于Spring Bean實(shí)例過(guò)程中使用反射和遞歸處理的Bean屬性填充問(wèn)題

好,就是截圖這里,我們看到已經(jīng)開(kāi)始進(jìn)行屬性填充操作了,當(dāng)發(fā)現(xiàn)屬性是 BeanReference 時(shí),則需要獲取創(chuàng)建 Bean 實(shí)例。

六、總結(jié) 在本章節(jié)中我們把 AbstractAutowireCapableBeanFactory 類(lèi)中的創(chuàng)建對(duì)象功能又做了擴(kuò)充,依賴(lài)于是否有構(gòu)造函數(shù)的實(shí)例化策略完成后,開(kāi)始補(bǔ)充 Bean 屬性信息。當(dāng)遇到 Bean 屬性為 Bean 對(duì)象時(shí),需要遞歸處理。最后在屬性填充時(shí)需要用到反射操作,也可以使用一些工具類(lèi)處理。 每一個(gè)章節(jié)的功能點(diǎn)我們都在循序漸進(jìn)的實(shí)現(xiàn),這樣可以讓新人更好的接受關(guān)于 Spring 中的設(shè)計(jì)思路。尤其是在一些已經(jīng)開(kāi)發(fā)好的類(lèi)上,怎么擴(kuò)充新的功能時(shí)候的設(shè)計(jì)更為重要。學(xué)習(xí)編程有的時(shí)候?qū)W習(xí)思路設(shè)計(jì)要比僅僅是做簡(jiǎn)單實(shí)現(xiàn),更能提升編程思維。 到這一章節(jié)關(guān)于 Bean 的創(chuàng)建操作就開(kāi)發(fā)完成了,接下來(lái)需要整個(gè)框架的基礎(chǔ)上完成資源屬性的加載,就是我們需要去動(dòng) Xml 配置了,讓我們這小框架越來(lái)越像 Spring。另外在框架實(shí)現(xiàn)的過(guò)程中所有的類(lèi)名都會(huì)參考 Spring 源碼,以及相應(yīng)的設(shè)計(jì)實(shí)現(xiàn)步驟也是與 Spring 源碼中對(duì)應(yīng),只不過(guò)會(huì)簡(jiǎn)化一些流程,但你可以拿相同的類(lèi)名,去搜到每一個(gè)功能在 Spring 源碼中的實(shí)現(xiàn)。

以上就是關(guān)于Spring Bean實(shí)例過(guò)程中使用反射和遞歸處理的Bean屬性填充問(wèn)題的詳細(xì)內(nèi)容,更多關(guān)于Spring Bean Bean屬性填充的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: Spring
相關(guān)文章:
主站蜘蛛池模板: 美女在线国产 | 国产欧美日韩一区二区三区在线观看 | 国外成人免费视频 | 日本天堂一区二区 | 国产区在线看 | 欧洲一区二区三区 | 亚洲欧美日韩电影 | 亚洲天堂日韩精品 | 亚洲精品久久久久久久久久久久久 | av一区二区在线观看 | 日韩av手机在线观看 | 亚洲xxxxx| 亚洲小视频在线播放 | 精品国产91乱码一区二区三区 | 国产精品观看 | 亚洲一区日韩 | 玖玖综合网 | 国产极品粉嫩美女呻吟在线看人 | 欧美日韩综合视频 | 亚洲成人免费视频在线观看 | 欧美国产精品一区二区 | 精品一二 | 久久黄色精品视频 | 国产96在线 | 99精品国产一区二区三区 | 亚洲视屏| 亚洲日本欧美日韩高观看 | avav在线看| aⅴ色国产 欧美 | www.99热.com| 日韩精品在线免费观看视频 | 成人精品 | 日韩视频―中文字幕 | 在线一区| 日韩亚洲视频 | 国产精品日韩在线观看 | 天天视频一区二区三区 | 日韩在线一区二区三区 | 亚洲一区二区三区四区五区午夜 | 亚洲va国产日韩欧美精品色婷婷 | 国产精品欧美一区二区 |