Android-使用@AutoService实现spi

本文详细介绍了Java SPI(ServiceProviderInterface)的工作原理,一种用于框架扩展和组件替换的API,通过接口编程、策略模式及配置文件实现动态加载。文章通过实例展示了如何定义接口、实现接口、使用ServiceLoader读取实现类,以及@AutoService简化开发流程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

什么是SPI?

Java SPI全称Service Provider Interface,是Java提供的一套用来被第三方实现或者扩展的API,它可以用来启用框架扩展和替换组件。实际上是“基于接口的编程+策略模式+配置文件”组合实现的动态加载机制.

具体解释就是:

  • 定义一个接口文件
  • 写出多个该接口文件的实现
  • 在 src/main/resources/ 下建立 /META-INF/services 目录, 新增一个以接口命名的文件 , 内容是要接口的实现类全路径
  • 使用ServiceLoader类 来获取到这些实现的接口

示例

  • 定义一个接口文件 - Book

    package com.github.jokar.spi_test;
    
    public interface Book {
    
        String name();
    }
    
  • 实现两个接口

    package com.github.jokar.spi_test;
    
    public class Android implements Book {
        @Override
        public String name() {
            return "Android";
        }
    }
    
    
    package com.github.jokar.spi_test;
    
    public class Ios implements Book {
        @Override
        public String name() {
            return "iOS";
        }
    }
    
    
  • resources目录下建立/META-INF/services目录,并建立已com.github.jokar.spi_test.Book为命名的文件,然后把AndroidIos全路径添加进去
    image.png

  • 使用ServiceLoader类读取Book接口实现类

    ServiceLoader<Book> load = ServiceLoader.load(Book.class);
            for (Book book : load) {
                System.out.println(book.name());
            }
    
  • 结果
    image.png


可以看到这个过程跟我们平常获取到接口实现类的方式不一样,平常情况下要获取到实现类可能需要个集合然后把实现类一个个添加进去,而用spi就不用这样操作。这样在我们平常组件化开发中非常便利,提供了解耦化的路径。

但是使用起来却很不方便,最麻烦的就是要到/META-INF/services目录建立文件,不能动态添加。所以我们用到了Google的@AutoService,他可以帮我们在编译的时候动态去生成这些东西,这样开发中就不用做太多其他复杂操作。

@AutoService 引入(Koltin)

apply plugin: 'kotlin-kapt'
kapt 'com.google.auto.service:auto-service:1.0-rc6'
api 'com.google.auto.service:auto-service:1.0-rc6'

@AutoService 举例

创建一个接口文件

interface Book {
    fun name() :String
}

创建两个继承

  • Android-Book

    @AutoService(Book::class)
    class Android :Book {
        override fun name():String {
            return "Android"
        }
    }
    
  • iOS-Book

    @AutoService(Book::class)
    class IOS :Book {
        override fun name(): String {
            return "ios"
        }
    }
    

获取继承类

    private fun getBookList() {
        val bookList = ServiceLoader.load(Book::class.java, javaClass.classLoader).toList()
        bookList.forEach {
            Log.d("MainActivity", it.name())
        }
    }

结果

image.png

可以看到获取到了两个继承类,并且不是空!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值