亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

簡(jiǎn)單分析JDK中「SPI」的原理

 更新時(shí)間:2025年05月01日 10:32:10   作者:七號(hào)樓  
文章介紹了Java SPI(Service Provider Interface)機(jī)制,包括其概念、入門(mén)案例、原理分析以及實(shí)際應(yīng)用,SPI機(jī)制通過(guò)服務(wù)接口和實(shí)現(xiàn)類的解耦,實(shí)現(xiàn)了服務(wù)的動(dòng)態(tài)加載和擴(kuò)展,文章詳細(xì)解釋了SPI機(jī)制的工作原理,包括迭代、hasNextService和nextService方法的實(shí)現(xiàn)

一、SPI簡(jiǎn)介

1、概念

SPI即service-provider-interface的簡(jiǎn)寫(xiě);

JDK內(nèi)置的服務(wù)提供加載機(jī)制,可以為服務(wù)接口加載實(shí)現(xiàn)類,解耦是其核心思想,也是很多框架和組件的常用手段;

2、入門(mén)案例

2.1 定義接口

就是普通的接口,在SPI的機(jī)制中稱為【service】,即服務(wù);

public interface Animal {
    String animalName () ;
}

2.2 兩個(gè)實(shí)現(xiàn)類

提供兩個(gè)模擬用來(lái)測(cè)試,就是普通的接口實(shí)現(xiàn)類,在SPI的機(jī)制中稱為【service-provider】即服務(wù)提供方;

CatAnimal實(shí)現(xiàn)類;

public class CatAnimal implements Animal {
    @Override
    public String animalName() {
        System.out.println("Cat-Animal:布偶貓");
        return "Ragdoll";
    }
}

DogAnimal實(shí)現(xiàn)類;

public class DogAnimal implements Animal {
    @Override
    public String animalName() {
        System.out.println("Dog-Animal:哈士奇");
        return "husky";
    }
}

2.3 配置文件

文件目錄:在代碼工程中創(chuàng)建META-INF.services文件夾;

文件命名:butte.program.basics.spi.inf.Animal,即全限定接口名稱;

文件內(nèi)容:添加相應(yīng)實(shí)現(xiàn)類的全限定命名;

butte.program.basics.spi.impl.CatAnimal
butte.program.basics.spi.impl.DogAnimal

2.4 測(cè)試代碼

通過(guò)ServiceLoader加載配置文件中指定的服務(wù)實(shí)現(xiàn)類,然后遍歷并調(diào)用Animal接口方法,從而執(zhí)行不同服務(wù)提供方的具體邏輯;

public class SpiAnaly {
    public static void main(String[] args) {
        ServiceLoader<Animal> serviceLoader = ServiceLoader.load(Animal.class);
        Iterator<Animal> animalIterator = serviceLoader.iterator();
        while(animalIterator.hasNext()) {
            Animal animal = animalIterator.next();
            System.out.println("animal-name:" + animal.animalName());
        }
    }
}

結(jié)果輸出

Cat-Animal:布偶貓 \n animal-name:ragdoll
Dog-Animal:哈士奇 \n animal-name:husky

二、原理分析

1、ServiceLoader結(jié)構(gòu)

很顯然,分析SPI機(jī)制的原理,從ServiceLoader源碼中l(wèi)oad方法切入即可,但是需要先從核心類的結(jié)構(gòu)開(kāi)始分析;

public final class ServiceLoader<S> implements Iterable<S> {
    // 配置文件目錄
    private static final String PREFIX = "META-INF/services/";
    // 表示正在加載的服務(wù)的類或接口
    private final Class<S> service;
    // 類加載器用來(lái)定位,加載,實(shí)例化服務(wù)提供方
    private final ClassLoader loader;
    // 創(chuàng)建ServiceLoader時(shí)采用的訪問(wèn)控制上下文
    private final AccessControlContext acc;
    // 按實(shí)例化的順序緩存服務(wù)提供方
    private LinkedHashMap<String,S> providers = new LinkedHashMap<>();
    // 惰性查找迭代器
    private LazyIterator lookupIterator;
    /**
     * service:表示服務(wù)的接口或抽象類
     * loader: 類加載器
     */
    public static <S> ServiceLoader<S> load(Class<S> service) {
        ClassLoader cl = Thread.currentThread().getContextClassLoader();
        return ServiceLoader.load(service, cl);
    }
    /**
     * ServiceLoader構(gòu)造方法
     */
    private ServiceLoader(Class<S> svc, ClassLoader cl) {
        loader = (cl == null) ? ClassLoader.getSystemClassLoader() : cl;
        acc = (System.getSecurityManager() != null) ? AccessController.getContext() : null;
        reload();
    }
    public void reload() {
        providers.clear();
        // 實(shí)例化迭代器
        lookupIterator = new LazyIterator(service, loader);
    }
    public static <S> ServiceLoader<S> load(Class<S> service,ClassLoader loader) {
        return new ServiceLoader<>(service, loader);
    }

    private class LazyIterator implements Iterator<S> {
        // 服務(wù)接口
        Class<S> service;
        // 類加載器
        ClassLoader loader;
        // 實(shí)現(xiàn)類URL
        Enumeration<URL> configs = null;
        // 實(shí)現(xiàn)類全名
        Iterator<String> pending = null;
        // 下個(gè)實(shí)現(xiàn)類全名
        String nextName = null;
    }
}

斷點(diǎn)截圖:

2、iterator迭代方法

在ServiceLoader類的迭代器方法中,實(shí)際使用的是LazyIterator內(nèi)部類的方法;

public Iterator<S> iterator() {
    return new Iterator<S>() {
        Iterator<Map.Entry<String,S>> knownProviders = providers.entrySet().iterator();
        public boolean hasNext() {
            if (knownProviders.hasNext())
                return true;
            return lookupIterator.hasNext();
        }
        public S next() {
            if (knownProviders.hasNext())
                return knownProviders.next().getValue();
            return lookupIterator.next();
        }
        public void remove() {
            throw new UnsupportedOperationException();
        }
    };
}

3、hasNextService方法

從上面迭代方法的源碼中可知,最終執(zhí)行的是LazyIterator#hasNextService判斷方法,該方法通過(guò)解析最終會(huì)得到實(shí)現(xiàn)類的全限定名稱;

private class LazyIterator implements Iterator<S> {
    private boolean hasNextService() {
        // 1、拼接名稱
        String fullName = PREFIX + service.getName();
        // 2、加載資源文件
        configs = loader.getResources(fullName);
        // 3、解析文件內(nèi)容
        pending = parse(service, configs.nextElement());
        nextName = pending.next();
        return true;
    }
}

斷點(diǎn)截圖:

4、nextService方法

迭代器的next方法最終執(zhí)行的是LazyIterator#nextService獲取方法,會(huì)基于上面hasNextService方法獲取的實(shí)現(xiàn)類全限定名稱,獲取其Class對(duì)象,進(jìn)而得到實(shí)例化對(duì)象,緩存并返回;

private class LazyIterator implements Iterator<S> {
    private S nextService() {
        // 1、通過(guò)全限定命名獲取Class對(duì)象
        String cn = nextName;
        Class<?> c = Class.forName(cn, false, loader);
        // 2、實(shí)例化對(duì)象
        S p = service.cast(c.newInstance());
        // 3、放入緩存并返回該對(duì)象
        providers.put(cn, p);
        return p;
    }
}

斷點(diǎn)截圖:

三、SPI實(shí)踐

1、Driver驅(qū)動(dòng)接口

在JDK中提供了數(shù)據(jù)庫(kù)驅(qū)動(dòng)接口java.sql.Driver,無(wú)論是MySQL驅(qū)動(dòng)包還是Druid連接池,都提供了該接口的實(shí)現(xiàn)類,通過(guò)SPI機(jī)制可以加載到這些驅(qū)動(dòng)實(shí)現(xiàn)類;

public class DriverManager {
    private static void loadInitialDrivers() {
        AccessController.doPrivileged(new PrivilegedAction<Void>() {
            public Void run() {
                ServiceLoader<Driver> loadedDrivers = ServiceLoader.load(java.sql.Driver.class);
                Iterator<Driver> driversIterator = loadedDrivers.iterator();
            }
        });
    }
}

斷點(diǎn)截圖:

2、Slf4j日志接口

SLF4J是門(mén)面模式的日志組件,提供了標(biāo)準(zhǔn)的日志服務(wù)SLF4JServiceProvider接口,在LogFactory日志工廠類中,負(fù)責(zé)加載具體的日志實(shí)現(xiàn)類,比如常用的Log4j或Logback日志組件;

public final class LoggerFactory { static List<SLF4JServiceProvider> findServiceProviders() { // 服務(wù)加載 ClassLoader classLoaderOfLoggerFactory = org.slf4j.LoggerFactory.class.getClassLoader(); // 重點(diǎn)看該方法:【getServiceLoader()】 ServiceLoader<SLF4JServiceProvider> serviceLoader = getServiceLoader(classLoaderOfLoggerFactory); // 迭代方法 List<SLF4JServiceProvider> providerList = new ArrayList(); Iterator<SLF4JServiceProvider> iterator = serviceLoader.iterator(); while(iterator.hasNext()) { safelyInstantiate(providerList, iterator); } return providerList; } }

斷點(diǎn)截圖:

到此這篇關(guān)于簡(jiǎn)單分析JDK中「SPI」的原理的文章就介紹到這了,更多相關(guān)JDK中「SPI」的原理內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 淺談一下Java中的ReentrantLock

    淺談一下Java中的ReentrantLock

    這篇文章主要介紹了淺談一下Java中的ReentrantLock,這個(gè)類是JUC工具包中對(duì)線程安全問(wèn)題提供的一種解決方案,它主要是用來(lái)給對(duì)象上鎖,保證同一時(shí)間這能有一個(gè)線程在訪問(wèn)當(dāng)前對(duì)象,需要的朋友可以參考下
    2023-09-09
  • Mybatis常見(jiàn)注解有哪些(總結(jié))

    Mybatis常見(jiàn)注解有哪些(總結(jié))

    這篇文章主要介紹了Mybatis常見(jiàn)注解有哪些(總結(jié)),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2021-03-03
  • Java內(nèi)存模型知識(shí)詳解

    Java內(nèi)存模型知識(shí)詳解

    這篇文章主要介紹了Java內(nèi)存模型知識(shí)詳解,文中通過(guò)對(duì)內(nèi)存訪問(wèn)時(shí)的交互關(guān)系圖解介紹的十分詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-07-07
  • spring boot mogodb多條件拼接的解決方法

    spring boot mogodb多條件拼接的解決方法

    這篇文章主要介紹了spring boot mogodb多條件拼接的解決方法,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2018-08-08
  • 淺析Java 并發(fā)編程中的synchronized

    淺析Java 并發(fā)編程中的synchronized

    這篇文章主要介紹了Java 并發(fā)編程中的synchronized的相關(guān)資料,幫助大家更好的理解和學(xué)習(xí)Java并發(fā)編程,感興趣的朋友可以了解下
    2020-12-12
  • WPF實(shí)現(xiàn)自定義一個(gè)自刪除的多功能ListBox

    WPF實(shí)現(xiàn)自定義一個(gè)自刪除的多功能ListBox

    這篇文章主要為大家詳細(xì)介紹了如何利用WPF實(shí)現(xiàn)自定義一個(gè)自刪除的多功能ListBox,文中示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下
    2022-12-12
  • BaseDao封裝JavaWeb的增刪改查的實(shí)現(xiàn)代碼

    BaseDao封裝JavaWeb的增刪改查的實(shí)現(xiàn)代碼

    Basedao 是一種基于數(shù)據(jù)訪問(wèn)對(duì)象(Data Access Object)模式的設(shè)計(jì)方法,它是一個(gè)用于處理數(shù)據(jù)庫(kù)操作的基礎(chǔ)類,負(fù)責(zé)封裝數(shù)據(jù)庫(kù)訪問(wèn)的底層操作,提供通用的數(shù)據(jù)庫(kù)訪問(wèn)方法,本文給大家介紹了BaseDao封裝JavaWeb的增刪改查的實(shí)現(xiàn)代碼,需要的朋友可以參考下
    2024-03-03
  • 線程池調(diào)用kafka發(fā)送消息產(chǎn)生的內(nèi)存泄漏問(wèn)題排查解決

    線程池調(diào)用kafka發(fā)送消息產(chǎn)生的內(nèi)存泄漏問(wèn)題排查解決

    這篇文章主要為大家介紹了線程池調(diào)用kafka發(fā)送消息產(chǎn)生的內(nèi)存泄漏問(wèn)題排查解決,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-08-08
  • MyEclipse安裝JS代碼提示的教程(Spket插件)

    MyEclipse安裝JS代碼提示的教程(Spket插件)

    本篇文章主要介紹了MyEclipse安裝JS代碼提示的教程(Spket插件),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2017-12-12
  • Java多線程 Producer and Consumer設(shè)計(jì)模式

    Java多線程 Producer and Consumer設(shè)計(jì)模式

    這篇文章主要介紹了Java多線程 Producer and Consumer設(shè)計(jì)模式,producer是生產(chǎn)者的意思:指生產(chǎn)數(shù)據(jù)的線程,consumer是消費(fèi)者的意思,指的是使用數(shù)據(jù)的線程,下文圍繞Producer及Consumer展開(kāi)話題,需要的朋友可以參考一下
    2021-10-10

最新評(píng)論