你所不知道的Java之ClassLoader并行加載

在JDK1.7之前的JDK1.6 ClassLoader筋粗,是這個(gè)樣子的:

protected synchronized Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {
    Class c = findLoadedClass(name);
    if (c == null) {
        try {
            if (parent != null) {
                c = parent.loadClass(name, false);
            } else {
                c = findBootstrapClassOrNull(name);
            }
        } catch (ClassNotFoundException e) {
        }
        if (c == null) {
            c = findClass(name);
        }
    }
    if (resolve) {
        resolveClass(c);
    }
    return c;
}

可以看到loadClass方法是synchronized,所以如果調(diào)用了java.lang.ClassLoader的loadClass方法窗悯,則會(huì)是對(duì)象級(jí)同步調(diào)用。

PS:請(qǐng)問方法的修飾符偷拔,可以被子類繼承么蒋院?

為了提高效率,在JDK1.7后莲绰,ClassLoader進(jìn)行了優(yōu)化欺旧,可以并行加載。

如果想要ClassLoader支持并行的能力蛤签,必須滿足以下條件:

  1. 目前沒有創(chuàng)建該ClassLoader的實(shí)例辞友。
  2. 所有該ClassLoader的父類都支持并行加載。

PS:一旦聲明為并行加載顷啼,則不能回退(即改回串行加載)踏枣。

總的來說,就是在該ClassLoader和該ClasssLoader的父類中钙蒙,聲明以下靜態(tài)代碼塊(需要注意靜態(tài)代碼塊的順序茵瀑,不懂得請(qǐng)留言)。

static {
    ClassLoader.registerAsParallelCapable();
}
private static class ParallelLoaders {
    private ParallelLoaders() {
    }

    private static final Set<Class<? extends ClassLoader>> loaderTypes =
            Collections.newSetFromMap(new WeakHashMap<>());

    static {
        synchronized (loaderTypes) {
            loaderTypes.add(ClassLoader.class);
        }
    }
    static boolean register(Class<? extends ClassLoader> c) {
        synchronized (loaderTypes) {
            if (loaderTypes.contains(c.getSuperclass())) {
                loaderTypes.add(c);
                return true;
            } else {
                return false;
            }
        }
    }
    static boolean isRegistered(Class<? extends ClassLoader> c) {
        synchronized (loaderTypes) {
            return loaderTypes.contains(c);
        }
    }
}

ClassLoader部分代碼截裙帷:


private final ConcurrentHashMap<String, Object> parallelLockMap;

protected static boolean registerAsParallelCapable() {
    Class<? extends ClassLoader> callerClass =
        Reflection.getCallerClass().asSubclass(ClassLoader.class);
    return ParallelLoaders.register(callerClass);
}
protected Object getClassLoadingLock(String className) {
    Object lock = this;
    if (parallelLockMap != null) {
        Object newLock = new Object();
        lock = parallelLockMap.putIfAbsent(className, newLock);
        if (lock == null) {
            lock = newLock;
        }
    }
    return lock;
}

protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {
    synchronized (getClassLoadingLock(name)) {
        Class<?> c = findLoadedClass(name);
        if (c == null) {
            long t0 = System.nanoTime();
            try {
                if (parent != null) {
                    c = parent.loadClass(name, false);
                } else {
                    c = findBootstrapClassOrNull(name);
                }
            } catch (ClassNotFoundException e) {
            }

            if (c == null) {
                long t1 = System.nanoTime();
                c = findClass(name);
                sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
                sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
                sun.misc.PerfCounter.getFindClasses().increment();
            }
        }
        if (resolve) {
            resolveClass(c);
        }
        return c;
    }
}

可以看到马昨,如果注冊(cè)了ClassLoader為并行加載,則loadClass的時(shí)候扛施,鎖的粒度是className鸿捧,否則鎖的粒度是ClassLoader實(shí)例本身this

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末疙渣,一起剝皮案震驚了整個(gè)濱河市匙奴,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌妄荔,老刑警劉巖泼菌,帶你破解...
    沈念sama閱讀 217,277評(píng)論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異啦租,居然都是意外死亡哗伯,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評(píng)論 3 393
  • 文/潘曉璐 我一進(jìn)店門篷角,熙熙樓的掌柜王于貴愁眉苦臉地迎上來焊刹,“玉大人,你說我怎么就攤上這事∨翱椋” “怎么了俩滥?”我有些...
    開封第一講書人閱讀 163,624評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)非凌。 經(jīng)常有香客問我举农,道長(zhǎng),這世上最難降的妖魔是什么敞嗡? 我笑而不...
    開封第一講書人閱讀 58,356評(píng)論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮航背,結(jié)果婚禮上喉悴,老公的妹妹穿的比我還像新娘。我一直安慰自己玖媚,他們只是感情好箕肃,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,402評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著今魔,像睡著了一般勺像。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上错森,一...
    開封第一講書人閱讀 51,292評(píng)論 1 301
  • 那天吟宦,我揣著相機(jī)與錄音,去河邊找鬼涩维。 笑死殃姓,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的瓦阐。 我是一名探鬼主播蜗侈,決...
    沈念sama閱讀 40,135評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼睡蟋!你這毒婦竟也來了踏幻?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,992評(píng)論 0 275
  • 序言:老撾萬榮一對(duì)情侶失蹤戳杀,失蹤者是張志新(化名)和其女友劉穎该面,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體豺瘤,經(jīng)...
    沈念sama閱讀 45,429評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,636評(píng)論 3 334
  • 正文 我和宋清朗相戀三年坐求,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了蚕泽。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,785評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖荒吏,靈堂內(nèi)的尸體忽然破棺而出敛惊,到底是詐尸還是另有隱情,我是刑警寧澤绰更,帶...
    沈念sama閱讀 35,492評(píng)論 5 345
  • 正文 年R本政府宣布瞧挤,位于F島的核電站,受9級(jí)特大地震影響儡湾,放射性物質(zhì)發(fā)生泄漏特恬。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,092評(píng)論 3 328
  • 文/蒙蒙 一徐钠、第九天 我趴在偏房一處隱蔽的房頂上張望癌刽。 院中可真熱鬧,春花似錦尝丐、人聲如沸显拜。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)远荠。三九已至,卻和暖如春呢簸,著一層夾襖步出監(jiān)牢的瞬間矮台,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評(píng)論 1 269
  • 我被黑心中介騙來泰國(guó)打工根时, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留瘦赫,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,891評(píng)論 2 370
  • 正文 我出身青樓蛤迎,卻偏偏與公主長(zhǎng)得像确虱,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子替裆,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,713評(píng)論 2 354

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