Android-使用@AutoService实现spi

什么是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

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

  • 2
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java中的autoservice注解是一种用于服务发现的机制,它能够自动将服务的实现类注册到指定的配置文件中,一般是META-INF/services目录下的文件。 然而,的确有一些情况下autoservice注解无法生成META-INF文件。这可能是由于以下几种原因导致的: 1. 编译问题:可能是由于编译配置的问题,导致编译器无法正确处理autoservice注解。这种情况下,我们可以尝试检查编译器的配置,或者使用其他IDE或编译工具进行尝试。 2. 库或框架限制:有些库或框架可能不支持autoservice注解,或者在使用autoservice注解时会有特殊的要求。我们可以查看相关库或框架的文档,了解其对autoservice注解的支持情况,或者尝试其他方式实现服务发现。 3. 配置文件缺失或错误:autoservice注解生成META-INF文件的前提是META-INF目录存在,并且配置文件的名称和路径正确。如果自动生成的META-INF文件不存在,或者文件名或路径有误,就无法实现服务发现。我们需要检查项目的文件结构,确认META-INF目录是否存在,并且配置文件的名称和路径是否正确。 总之,虽然autoservice注解通常可以自动将服务实现类注册到META-INF文件中,但在某些情况下可能会遇到无法生成META-INF文件的问题。我们需要仔细检查编译配置、库或框架限制,以及配置文件是否正确等因素,以找出问题所在,并采取相应的解决方案。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值