手把手帶你實(shí)現(xiàn)IOC,看完還不懂來打我减江!

前言

Spring ioc 相信很多人都知道這是Spring框架中一個非常核心的組件染突,IoC(控制反轉(zhuǎn)),對于初學(xué)Spring的人來說辈灼,對其的設(shè)計(jì)思想理解可能非常表面份企,要理解好Ioc的關(guān)鍵是要明確 以下幾點(diǎn),誰控制誰巡莹,控制什么司志,為何是反轉(zhuǎn)(那正轉(zhuǎn)?)降宅,哪些方面反轉(zhuǎn)了 骂远。

誰控制誰,控制什么 還記得你們當(dāng)時學(xué)習(xí)servlet的時候在每一層內(nèi)部通過new關(guān)鍵字進(jìn)行創(chuàng)建對象嗎腰根,要清楚這是程序主動去創(chuàng)建依賴對象; 而當(dāng)你們接觸了Spring之后IoC是有專門一個容器來創(chuàng)建這些對象激才,控制對象。

為何是反轉(zhuǎn)(那正轉(zhuǎn)?)瘸恼,這個簡單理解當(dāng)我們主動控制去獲取依賴對象劣挫,這就是正轉(zhuǎn)? 而反轉(zhuǎn)則是由容器來幫忙創(chuàng)建及注入依賴對象; 為何是反轉(zhuǎn) 钞脂? 因?yàn)橛扇萜鲙臀覀儾檎壹白⑷胍蕾噷ο蟠г疲瑢ο笾皇潜粍拥慕邮芤蕾噷ο螅允欠崔D(zhuǎn); 哪些方面反轉(zhuǎn)了冰啃?依賴對象的獲取被反轉(zhuǎn)了邓夕。

手撕最簡易版IOC容器

可能你還不明白,覺得很抽象阎毅?那么跟著小編手把手帶你來實(shí)現(xiàn)IOC焚刚!

ioc不是一個容器嘛,ok扇调,我們先來定義一個容器矿咕,容器的特性肯定擁有存、取對象嘛~

Spring的對象都是一個個bean狼钮,但是我們不知道bean類型是什么碳柱,那么就泛型體現(xiàn),還有bean什么時候存放進(jìn)容器的鞍疚摺莲镣?spring是在啟動的時候會進(jìn)行初始化掃描,我們就定義一個initAutoWired方法來模擬涎拉。

先定義一個橘松容器接口JsContainer瑞侮。

/**
 * @創(chuàng)建人 : 頭條賬號 "深夜敲代碼"
 * @創(chuàng)建時間 2021/7/13
 */
public interface JsContainer {

    /**
     * 根據(jù)Class獲取Bean
     * @param clazz
     * @return
     */
    <T> T getBean(Class<T> clazz);

    /**
     * 注冊一個Class到容器中
     *
     * @param clazz
     */
    Object registerBean(Class<?> clazz);

    /**
     * 初始化裝配
     */
    void initAutoWired();

我們再寫一個自定義注解標(biāo)識JsAutowired,來為了后面通過反射獲取實(shí)例化bean鼓拧,自定義注解可以指定要注入的類型半火,以及注入的bean名稱。

package com.orangesongjava.ioc;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

/**
 * @創(chuàng)建人 : 頭條賬號 "深夜敲代碼"
 * @創(chuàng)建時間 2021/7/13
 */
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface JsAutowired {

    /**
     * @return 要注入的類類型
     */
    Class<?> value() default Class.class;

    /**
     * @return bean的名稱
     */
    String name() default "";
}

容器接口和注解都有了季俩,接下來我們思考來實(shí)現(xiàn)它钮糖!我們定義一個類 JsSampleContainer 來實(shí)現(xiàn)容器接口JsContainer 。

package com.orangesongjava.ioc;

import java.lang.reflect.Field;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

/**
 * @創(chuàng)建人 : 頭條賬號 "深夜敲代碼"
 * @創(chuàng)建時間 2021/7/13
 */
@SuppressWarnings("unchecked")
public class JsSampleContainer implements JsContainer {
    /**
     * 保存所有bean對象酌住,格式為 com.xxx.xxx.XxxClass : @56x2ya
     */
    private Map<String, Object> beanNameMap;

    /**
     * 存儲bean和name的關(guān)系
     */
    private Map<String, String> beanKeys;

    public JsSampleContainer() {
        this.beanNameMap = new ConcurrentHashMap<>();
        this.beanKeys = new ConcurrentHashMap<>();
    }

    @Override
    public <T> T getBean(Class<T> clazz) {
        String name = clazz.getName();
        Object object = beanNameMap.get(name);
        if(null != object){
            return (T) object;
        }
        return null;
    }

    @Override
    public Object registerBean(Class<?> clazz) {
        String name = clazz.getName();
        beanKeys.put(name, name);
        Object bean = newInstance(clazz);
        beanNameMap.put(name, bean);
        return bean;
    }

    @Override
    public void initAutoWired() {
        beanNameMap.forEach((k,v) -> injection(v));
    }
    /**
     * 注入對象
     * @param object
     */
    public void injection(Object object) {
        // 所有字段
        try {
            Field[] fields = object.getClass().getDeclaredFields();
            for (Field field : fields) {
                // 需要注入的字段
                JsAutowired jsAutowired = field.getAnnotation(JsAutowired.class);
                if (null != jsAutowired) {
                    // 要注入的字段
                    Object autoAutoWiredField = null;
                    String name = jsAutowired.name();
                    // 如果說這里JsAutowired自定義注解沒指定name屬性 則默認(rèn)值是""
                    if(!name.equals("")){
                        // 指定了特定的name
                        String className = beanKeys.get(name);
                        if(null != className && !className.equals("")){
                                            autoAutoWiredField = beanNameMap.get(className);
                        }
                        if (null == autoAutoWiredField) {
                            throw new RuntimeException("Unable to load " + name);
                        }
                    } else {
                                        // JsAutowired注解沒有name屬性
                                        // 判斷注入的類型 是否是類Class類型 默認(rèn)值也是Class
                        if(jsAutowired.value() == Class.class){
                                               autoAutoWiredField = register(field.getType());
                        } else {
                            // 指定裝配的類
                                            autoAutoWiredField = this.getBean(jsAutowired.value());
                            if (null == autoAutoWiredField) {
                                                autoAutoWiredField = register(jsAutowired.value());
                            }
                    }
                    }
                    if (null == autoAutoWiredField) {
                        throw new RuntimeException("Unable to load " + field.getType().getCanonicalName());
                    }
                    boolean accessible = field.isAccessible();
                    field.setAccessible(true);
                    field.set(object, autoAutoWiredField);
                    field.setAccessible(accessible);
                }
            }
        } catch (SecurityException e) {
            e.printStackTrace();
        } catch (IllegalArgumentException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }
    }

    private Object register(Class<?> clazz){
        if(null != clazz){
            return this.registerBean(clazz);
        }
        return null;
    }

    /**
     * 創(chuàng)建一個實(shí)例對象
     * @param clazz class對象
     * @return
     */
    public static Object newInstance(Class<?> clazz) {
        try {
            return clazz.newInstance();
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }
        return null;
    }
}

測試一下

我們現(xiàn)在來測試一下店归,我們先定義一個類OrangeSongJavaService,方便待會測試被調(diào)用赂韵。注意哦娱节,這個類上面沒有任何注解挠蛉。也就是他并沒有被Spring管理起來祭示。

/**
 * @創(chuàng)建人 : 頭條賬號 "深夜敲代碼"
 * @創(chuàng)建時間 2021/7/13
 */
package com.orangesongjava.ioc;

public class OrangeSongJavaService {
    public void writeArticleOnJs(String name){
        System.out.println(name + "在頭條上寫文章!");
    }
}

在定義本地調(diào)用的模擬調(diào)用OrangeSongJavaClient。注意這個類上面也是沒有注解,另外引用的OrangeSongJavaService 被我們的自定義注解JsAutowired標(biāo)識起來了质涛。

/**
 * @創(chuàng)建人 : 頭條賬號 "深夜敲代碼"
 * @創(chuàng)建時間 2021/7/13
 */
package com.orangesongjava.ioc;

public class OrangeSongJavaClient {

    @JsAutowired
    private OrangeSongJavaService orangeSongJavaService;

    public void work() {
        orangeSongJavaService.writeArticleOnJs("深夜敲代碼");
    }
    public OrangeSongJavaService getOrangeSongJavaService() {
        return orangeSongJavaService;
    }
}

現(xiàn)在我們寫個測試main方法測試一下稠歉。

/**
 * @創(chuàng)建人 : 頭條賬號 "深夜敲代碼"
 * @創(chuàng)建時間 2021/7/13
 */
package com.orangesongjava.ioc;

public class IocTest {

    private static JsSampleContainer jsContainer = new JsSampleContainer();

    public static void main(String[] args) {
        // 將類注入容器
        jsContainer.registerBean(OrangeSongJavaClient.class);
        // 初始化注入-掃描引用
        jsContainer.initAutoWired();
        // 容器獲取bean
        OrangeSongJavaClient client jsContainer.getBean(OrangeSongJavaClient.class);
        // 執(zhí)行
        client.work();
    }
}

如果能輸出執(zhí)行邏輯,則表示這段程序沒問題汇陆∨ǎ看到這里,你對ioc理解是不是有點(diǎn)清晰了毡代,有任何疑問阅羹,或者不懂的地方可以下方留言。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末教寂,一起剝皮案震驚了整個濱河市捏鱼,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌酪耕,老刑警劉巖导梆,帶你破解...
    沈念sama閱讀 217,826評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異迂烁,居然都是意外死亡看尼,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,968評論 3 395
  • 文/潘曉璐 我一進(jìn)店門盟步,熙熙樓的掌柜王于貴愁眉苦臉地迎上來藏斩,“玉大人,你說我怎么就攤上這事址芯≡肿拢” “怎么了?”我有些...
    開封第一講書人閱讀 164,234評論 0 354
  • 文/不壞的土叔 我叫張陵谷炸,是天一觀的道長北专。 經(jīng)常有香客問我,道長旬陡,這世上最難降的妖魔是什么拓颓? 我笑而不...
    開封第一講書人閱讀 58,562評論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮描孟,結(jié)果婚禮上驶睦,老公的妹妹穿的比我還像新娘。我一直安慰自己匿醒,他們只是感情好场航,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,611評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著廉羔,像睡著了一般溉痢。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,482評論 1 302
  • 那天孩饼,我揣著相機(jī)與錄音髓削,去河邊找鬼。 笑死镀娶,一個胖子當(dāng)著我的面吹牛立膛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播梯码,決...
    沈念sama閱讀 40,271評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼宝泵,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了轩娶?” 一聲冷哼從身側(cè)響起鲁猩,我...
    開封第一講書人閱讀 39,166評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎罢坝,沒想到半個月后廓握,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,608評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡嘁酿,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,814評論 3 336
  • 正文 我和宋清朗相戀三年隙券,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片闹司。...
    茶點(diǎn)故事閱讀 39,926評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡娱仔,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出游桩,到底是詐尸還是另有隱情牲迫,我是刑警寧澤,帶...
    沈念sama閱讀 35,644評論 5 346
  • 正文 年R本政府宣布借卧,位于F島的核電站盹憎,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏铐刘。R本人自食惡果不足惜陪每,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,249評論 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望镰吵。 院中可真熱鬧檩禾,春花似錦、人聲如沸疤祭。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,866評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽勺馆。三九已至戏售,卻和暖如春啦辐,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背蜈项。 一陣腳步聲響...
    開封第一講書人閱讀 32,991評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留续挟,地道東北人紧卒。 一個月前我還...
    沈念sama閱讀 48,063評論 3 370
  • 正文 我出身青樓,卻偏偏與公主長得像诗祸,于是被迫代替她去往敵國和親跑芳。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,871評論 2 354

推薦閱讀更多精彩內(nèi)容