跟我動(dòng)手搭框架一之IOC容器實(shí)現(xiàn)

本篇文章面對(duì)的是有開(kāi)發(fā)經(jīng)驗(yàn)的Java developer 因?yàn)槲覀儗⒁獙?shí)現(xiàn)的Spring的IOC容器,

前些天由于工作中要開(kāi)發(fā)公司的Callback系統(tǒng),一直在研究Netty及IO模型,對(duì)于Netty這種非阻塞異步框架,非常崇拜,于是萌發(fā)一個(gè)想法,用Netty作為web容器,替換Tomcat研究性能.出于這種初衷,就開(kāi)始為SmileBoot項(xiàng)目開(kāi)始慢慢積累開(kāi)發(fā)知識(shí).本篇屬于小編SmileBoot中的一個(gè)模塊,為什么要起名Smile呢?因?yàn)樾【幨冀K認(rèn)為,我們要帶著好的心態(tài),才能學(xué)更多的東西,其實(shí)小編也是一個(gè)菜鳥(niǎo),之所以要寫(xiě)下來(lái),就是為了記憶和理解更深.因?yàn)槿绻炎约豪斫獾臇|西,能清楚的講給其他人,那么才算是真正的理解.

目錄

  • 1.原理分析及設(shè)計(jì)
  • 2.實(shí)現(xiàn)方案
    • 2.1 拿到掃描范圍
    • 2.2 更具掃描范圍加載范圍內(nèi)所有字節(jié)碼文件
    • 2.3 定義自己的上下文對(duì)象接口及實(shí)現(xiàn)類
  • 3.測(cè)試可用性
  • 4.擴(kuò)展性
  • 5.下篇預(yù)告

1.原理分析及設(shè)計(jì)

Spring的源碼,這里不跟著閱讀,直接去實(shí)現(xiàn),然后剛興趣的童鞋,可以自己在看看,原理是一樣的.

1.加載項(xiàng)目中所有的Class文件到Set集合

2.遍歷Set將標(biāo)記有IOC的組件的Class,獲取到,注冊(cè)到IOC容器,這個(gè)里面的重點(diǎn)是如何將Class里面的組件,注入進(jìn)來(lái).

image

@SmileComponent

在這里@SmileComponent注解是用來(lái)標(biāo)記,需要加入到IOC容器的類

@SmileBean

@SmileBean是用來(lái)標(biāo)記方法中返回值作為Bean,是將要被注冊(cè)到IOC容器的對(duì)象

@InsertBean

@InserBean是標(biāo)記,該字段是一個(gè)Bean,需要從IOC容器中獲取,然后注入到該對(duì)象中

2.實(shí)現(xiàn)方案

1.獲取所有的Class字節(jié)碼,在這中間我們有一個(gè)困難那就是如果知道,開(kāi)發(fā)者的所有字節(jié)碼呢?這個(gè)時(shí)候我們就可以用注解的形式,在啟動(dòng)類上做一個(gè)標(biāo)記,那么我們就能獲取到啟動(dòng)類的字節(jié)碼,從而獲取到將要掃描的跟目錄.

我們看下Spring是如何實(shí)現(xiàn)的吧

@SpringBootApplication
public class OtoSaasApplication  {
    public static void main(String[] args) {
      SpringApplication.run(OtoSaasApplication.class, args);
    }
}

在這段代碼中,有一個(gè)注解@SpringBootApplication ,了解Spring的開(kāi)發(fā)同事,都是知道這個(gè)注解其實(shí)包括了多個(gè)注解的,其中一個(gè)就是@ComponentScan

@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE})
@Documented
@Repeatable(ComponentScans.class)
public @interface ComponentScan {
    @AliasFor("basePackages")
    String[] value() default {}
}

那么我們就可以知道,其實(shí)也main方法所包含的注解,拿到根目錄的.這個(gè)有一個(gè)Spring的特性,那就是如果啟動(dòng)類在最外層的,那么默認(rèn)就是掃描,其子目錄中的Class,如果不是在根目錄,那么要指定掃描的范圍.

2.1拿到掃描范圍

那么我們想,如果用戶不指定,我們?cè)趺茨玫礁夸浤?

好,如果有疑惑的話,那么久帶著疑惑,看下面這段代碼吧!

我們定一個(gè)注解@SmileBootApplication 目錄就是獲取到用戶的根目錄,這里關(guān)于注解不在解釋,如果有不了解實(shí)現(xiàn)注解的可以看小編SpringBoot實(shí)踐中的自定義注解

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface SmileBootApplication {
    String[] basePackages() default {};
}
@SmileBootApplication
public class SmileApplication {
    public static void main(String[] args) {
       SmileApplication.run(SmileApplication.class, args);
    }
}

我們定義一個(gè)方法也就是在run方法中,根據(jù)class,文件,獲取到注解的根目錄

 public static String getBaseRootPackage(Class<?> cls) {
        SmileBootApplication declaredAnnotation = null;
        try {
            declaredAnnotation = cls.getDeclaredAnnotation(SmileBootApplication.class);
        } catch (Exception e) {
            throw new IllegalArgumentException("請(qǐng)?zhí)砑覢SmileBootApplication");
        }
        /**
         * 獲取注解上的掃描目錄
         * 如果沒(méi)有指定,就從當(dāng)前目錄獲取
         */
        String[] strings = declaredAnnotation.basePackages();
        String baseRootPackage = "";
        if (strings.length == 0) {
            baseRootPackage = cls.getPackage().getName();
        }
        return baseRootPackage;
    }

看到這里,我們已經(jīng)拿到了項(xiàng)目的根目錄,或者說(shuō)是將要掃描的范圍了

2.2 獲取指定目錄下的所有字節(jié)碼文件

這個(gè)時(shí)候我們要知道一個(gè)基礎(chǔ)的方法,那就是

/**
 * @param className     完整類路徑
 * @param isInitialized 是否初始化 第2個(gè)boolean參數(shù)表示類是否需要初始化Class.forName(className)默認(rèn)是需要初始化吴攒。一旦初始化,就會(huì)觸發(fā)目標(biāo)對(duì)象的 static塊代碼執(zhí)行,static參數(shù)也也會(huì)被再次初始化
 * @param classLoader   類加載器
 * @return
 */
Class.forName(className, isInitialized, classLoader);

我們要用到一個(gè)工具類ClassUtils,該類中可以將根目錄中所有字節(jié)碼(.java,.jar文件)加載到Set<Class<?>>set中

這個(gè)工具也不是小編寫(xiě)的,是參考了很多博客大拿,發(fā)現(xiàn)都有用到,但是具體出自哪位,就不曉得了,那么也分享給大家

可以參考

GITHUB

2.3 定義自己的上下文對(duì)象接口及實(shí)現(xiàn)類

/**
 * @Package: pig.boot.ioc.context
 * @Description: 上下文
 * @author: liuxin
 * @date: 2017/11/17 下午11:52
 */
public interface ApplicationContext {
    Object getBean(String var1);
    <T> T getBean(String name, Class<T> requiredType);
    <T> T getBean(Class<T> name);
    boolean containsBean(String var1);
    void scan(String basePackRoot);
}

public class SmileApplicationContext implements ApplicationContext {
  /**
     * 掃描所有的類,并裝載
     *
     * @param basePackRoot
     */
    @Override
    public void scan(String basePackRoot) {
        ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
        Set<Class<?>> classesByPackage = null;
        try {
            /**
             * recursively 是否從根目錄,向下查找
             */
            classesByPackage = ClassUtils.getClassesByPackageName(classLoader, basePackRoot, true);
        } catch (IOException e) {
            e.printStackTrace();
        }
        /**
         * 加載到所有的bean
         */
        allBeans.addAll(classesByPackage);
       /**
        *掃描所有的標(biāo)記,加入到容器
        */
        classesByPackage.forEach(this::scanComponent);
        /**
         * 將沒(méi)有注冊(cè)的bean檢查,然后注入
         */
        processEarlyBeans();
    }
}

這個(gè)類的重點(diǎn)方法就是scan掃描所有的標(biāo)記,并從set中拿到每個(gè)字節(jié)碼,傳給scanComponent 方法去解析注冊(cè)

這個(gè)時(shí)候我們可能遇到一種情況,就是BeanA中需要注冊(cè)BeanB但是可能BeanB此時(shí)并沒(méi)有解析到,那么這個(gè)時(shí)候,就要考慮,把暫時(shí)實(shí)例化不了的,放入到delayBeans,等所有能解析的解析之后,在回過(guò)頭加載,我們來(lái)看這個(gè)方法,在看之前我們定義這樣一個(gè)類 BeanDefinition 用處就是講Bean的class和實(shí)例化對(duì)象都保存起來(lái)

/**
 * @Package: pig.boot.ioc.context
 * @Description: bean描述
 * @author: liuxin
 * @date: 2017/11/17 下午11:53
 */
public class BeanDefinition {
    Class<?> clazz;
    Object instance;
    public BeanDefinition(Class<?> clazz, Object instance) {
        this.clazz = clazz;
        this.instance = instance;
    }
}
 /**
     * 掃描所有被標(biāo)記的組件
     */
    public void scanComponent(Class<?> nextCls) {
        SmileComponent declaredAnnotation = nextCls.getDeclaredAnnotation(SmileComponent.class);
        Object beanInstance = null;
        if (declaredAnnotation != null) {
            try {
                beanInstance = nextCls.newInstance();
                String beanName = declaredAnnotation.vlaue();
                if (beanName.isEmpty()) {
                    beanName = nextCls.getSimpleName();
                }
                /**
                 * 保證bean名稱的唯一性
                 */
                Long beanId = beanIds.get();
                //將類名,首字母小寫(xiě),并檢查是否存在,如果存在就后面添加id,這個(gè)id是原子操作,保證唯一.
                beanName= getUniqueBeanNameByClassAndBeanId(nextCls,beanId);
                /**
                 * 實(shí)例化里面的需要注入的字段都獲取到
                 * 如果返回true就可以直接添加到IOC容器
                 * lastChance=true 如果注入失敗就報(bào)錯(cuò),false不報(bào)錯(cuò),因?yàn)榈谝淮?可能所有類沒(méi)有初始化,所以等待延遲加載方法去,加載
                 */
                if (autowireFields(beanInstance, nextCls, false)) {
                    registeredBeans.put(beanName, new BeanDefinition(nextCls, beanInstance));
                } else {
                    /**
                     * 上面那種情況,可能會(huì)出現(xiàn),當(dāng)要注入,但是被注入的未加載到IOC容器中的情況,所以對(duì)于這種,就添加到earlyBeans中,后期注入
                     */
                    delayBeans.put(beanName, new BeanDefinition(nextCls, beanInstance));
                }

                /**
                 * 獲取方法上的bean
                 * 因?yàn)榉椒隙ㄊ怯蟹祷刂?的返回值就是實(shí)例化對(duì)象,所以可以直接,加入到IOC容器
                 */
                createBeansByMethodsOfClass(beanInstance, nextCls);
            } catch (Exception e) {

            }

        }
    }

3.測(cè)試可用性

@SmileBootApplication
public class SmileApplication {
    public static void main(String[] args) {
       SmileApplicationContext run = SmileApplication.run(SmileApplication.class, args);
        System.out.println(run.getBean(BeanB.class).toString());
        System.out.println(run.getBean(BeanA.class).beanB().toString());
      //BeanB{content='hi. iam is beanB'}
      //BeanB{content='hi. iam is beanB'}

    }
}
/**
 * @Package: pig.boot.ioc.context
 * @Description: 獲取參數(shù)
 * @author: liuxin
 * @date: 2017/11/17 下午11:55
 */
@SmileComponent
public class BeanA {
    private String content;
    @InsertBean
    private BeanB beanb;
    public BeanA() {
    }
    public BeanA(String content) {
        this.content = content;
    }
    @SmileBean
    public BeanB beanB() {
        return new BeanB("hi. iam is beanB");
    }

}

4.可擴(kuò)展性

  • 定義上下文對(duì)象接口類,developer,可以定義自己的上下文類

    一個(gè)軟件實(shí)體如類、模塊和函數(shù)應(yīng)該對(duì)擴(kuò)展開(kāi)放崇呵,對(duì)修改關(guān)閉

  • ApplicationContextInitialezer 初始化類執(zhí)行,獲取初始化條件和初始化方法,指定合適的上下文實(shí)現(xiàn)類

? 不要存在多于一個(gè)導(dǎo)致類變更的原因,通俗的說(shuō)陨囊,即一個(gè)類只負(fù)責(zé)一項(xiàng)職責(zé)。

  • 小編最喜歡的方法是抽象,即具有共同特征的方法,用抽象類去實(shí)現(xiàn),具體的方法有繼承類去實(shí)現(xiàn)
  • 接口隔離,即依賴最小的接口,eg.A接口有五個(gè)方法 B此時(shí)要用3個(gè),C要用2個(gè),但是他們不得不全部實(shí)現(xiàn).此時(shí)我們可以把A接口拆分為2個(gè). 當(dāng)D5個(gè)方法的時(shí)候,就繼承2個(gè)接口,就可以

5.下篇預(yù)告

定義@SmileGetMapping,@SmilePostMapping,注解,綁定處理邏輯handler. 放入SmileNettyTaskHandler中,交給Netty處理異步處理

附錄

好的代碼就想一本書(shū),讀的書(shū)越多,思路就越廣,想法就越多

每個(gè)開(kāi)發(fā)人員要把自己當(dāng)做一個(gè)工程師,而不是一個(gè)coding 的碼農(nóng),工程師考慮問(wèn)題要從頂層設(shè)計(jì)考慮,而不是為了單純解決一個(gè)問(wèn)題而code.

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末诱建,一起剝皮案震驚了整個(gè)濱河市稀并,隨后出現(xiàn)的幾起案子仅颇,更是在濱河造成了極大的恐慌,老刑警劉巖碘举,帶你破解...
    沈念sama閱讀 211,376評(píng)論 6 491
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件忘瓦,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡引颈,警方通過(guò)查閱死者的電腦和手機(jī)耕皮,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,126評(píng)論 2 385
  • 文/潘曉璐 我一進(jìn)店門(mén)境蜕,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人凌停,你說(shuō)我怎么就攤上這事汽摹。” “怎么了苦锨?”我有些...
    開(kāi)封第一講書(shū)人閱讀 156,966評(píng)論 0 347
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)趴泌。 經(jīng)常有香客問(wèn)我舟舒,道長(zhǎng),這世上最難降的妖魔是什么嗜憔? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,432評(píng)論 1 283
  • 正文 為了忘掉前任秃励,我火速辦了婚禮,結(jié)果婚禮上吉捶,老公的妹妹穿的比我還像新娘夺鲜。我一直安慰自己,他們只是感情好呐舔,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,519評(píng)論 6 385
  • 文/花漫 我一把揭開(kāi)白布币励。 她就那樣靜靜地躺著,像睡著了一般珊拼。 火紅的嫁衣襯著肌膚如雪食呻。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 49,792評(píng)論 1 290
  • 那天澎现,我揣著相機(jī)與錄音仅胞,去河邊找鬼。 笑死剑辫,一個(gè)胖子當(dāng)著我的面吹牛干旧,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播妹蔽,決...
    沈念sama閱讀 38,933評(píng)論 3 406
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼椎眯,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了讹开?” 一聲冷哼從身側(cè)響起盅视,我...
    開(kāi)封第一講書(shū)人閱讀 37,701評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎旦万,沒(méi)想到半個(gè)月后闹击,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,143評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡成艘,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,488評(píng)論 2 327
  • 正文 我和宋清朗相戀三年赏半,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了贺归。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,626評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡断箫,死狀恐怖拂酣,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情仲义,我是刑警寧澤婶熬,帶...
    沈念sama閱讀 34,292評(píng)論 4 329
  • 正文 年R本政府宣布,位于F島的核電站埃撵,受9級(jí)特大地震影響赵颅,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜暂刘,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,896評(píng)論 3 313
  • 文/蒙蒙 一饺谬、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧谣拣,春花似錦募寨、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,742評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至贵涵,卻和暖如春格郁,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背独悴。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,977評(píng)論 1 265
  • 我被黑心中介騙來(lái)泰國(guó)打工例书, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人刻炒。 一個(gè)月前我還...
    沈念sama閱讀 46,324評(píng)論 2 360
  • 正文 我出身青樓决采,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親坟奥。 傳聞我的和親對(duì)象是個(gè)殘疾皇子树瞭,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,494評(píng)論 2 348

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

  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 171,761評(píng)論 25 707
  • Spring Cloud為開(kāi)發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見(jiàn)模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn)爱谁,斷路器晒喷,智...
    卡卡羅2017閱讀 134,629評(píng)論 18 139
  • 我一直覺(jué)得,科技發(fā)展的有點(diǎn)嚇人访敌。 就好像一百年前大家還是處在走路靠腿凉敲,通訊靠吼的時(shí)代,突然一下子汽車出現(xiàn)了,然后飛...
    比說(shuō)閱讀 275評(píng)論 0 0
  • 01 公寓附近有個(gè)有個(gè)咖啡店果复,關(guān)注很久卻一直沒(méi)走進(jìn)去過(guò)。周三收到公眾號(hào)發(fā)來(lái)的信息渤昌,說(shuō)周末將有一個(gè)英語(yǔ)角活動(dòng)虽抄,軟文內(nèi)...
    英語(yǔ)Mango姐閱讀 5,268評(píng)論 4 14
  • 保養(yǎng)卵巢 魚(yú)膠膠原蛋白能促進(jìn)人體紅細(xì)胞集落形成及細(xì)胞攜氧能力极颓,提高女性卵巢細(xì)胞氧飽和度,加速卵巢細(xì)胞的快速再生群嗤,令...
    鵲之燕小窩閱讀 2,251評(píng)論 0 1