Android 服务动态发现 SPA 之 Auto Service

转载请标明出处:http://blog.csdn.net/zhaoyanjun6/article/details/125859181
本文出自【赵彦军的博客】

SPI

SPI(Service Provider Interface)是 Java 提供的一种动态服务发现机制。通过SPI 机制,我们可以直接跨模块查找到想要的接口实现类,从而避免不必要的模块间依赖,降低模块之间的耦合性。

Java 内置的 SPI 机制是通过 ServiceLoader 查找某个接口的所有实现类,并实例化。

每个需要实例化模块下需要以该接口的全限定名(包名+类名)为文件名放到 resources/META-INF/services/ 目录下,然后将他的实现类的全限定名按行依次写到该文件中。

1、创建接口类 StudentInterface

public interface StudentInterface {
    void eat(String name);
}

2、创建接口实现类 StudentImpl

package com.zyj.demo;

import android.util.Log;

public class StudentImpl implements StudentInterface {

    @Override
    public void eat(String name) {
        Log.d("yu--", "" + name);
    }
}

StudentImpl 类的全路径为 : com.zyj.demo.StudentImpl

在这里插入图片描述

3、创建目录

  • main 目录下,创建 resources 目录
  • resources 目录下,创建 META-INF 目录
  • META-INF 目录下,创建 service 目录

service 目录下,创建 com.zyj.demo.StudentInterface 文件

文件的内容为:com.zyj.demo.StudentImpl

在这里插入图片描述
4、ServiceLoader 发现服务

接口可以有多个实现类,所以返回值是一个集合

//发现服务,接口可以有多个实现类,所以返回值是一个集合
ServiceLoader<StudentInterface> serviceLoader = ServiceLoader.load(StudentInterface.class);
        
//遍历服务
for (StudentInterface impl : serviceLoader) {
    impl.eat("zyj");
}

总结

  • 动态服务发现机制,可以很好的解耦,不必直接依赖接口实现类。
  • ServiceLoader.load 每次都会创建一个新对象。生命周期用完即销毁
  • 缺点也很明显,要手动往 resources/META-INF/services/ 写入文件。

有没有一种自动写入的工具,可以解放双手?

有的,下面我们就介绍 auto-service

auto-service

auto-service 是 google 出品的自动发现服务工具 ,是 SPA 的一种方式。

  • SPA :Service Pool for Android

依赖:

 annotationProcessor 'com.google.auto.service:auto-service:1.0'
 implementation 'com.google.auto.service:auto-service-annotations:1.0'

我们只需要在 接口的实现类上,加上 @AutoService 注解,就可以了。

@AutoService(StudentInterface.class)
public class StudentImpl implements StudentInterface {

    @Override
    public void eat(String name) {
        Log.d("yu--", "" + name);
    }
}

发现服务,还是需要 ServiceLoader.load ,这个跟 Java 使用的是一致的。

//发现服务,接口可以有多个实现类,所以返回值是一个集合
ServiceLoader<StudentInterface> serviceLoader = ServiceLoader.load(StudentInterface.class);
        
//遍历服务
for (StudentInterface impl : serviceLoader) {
    impl.eat("zyj");
}

总结

auto-service 的优势是,自动的往 resources/META-INF/services/ 写入文件,解放了双手,666666

我们看一眼这个目录,在 build 目录可以看到

在这里插入图片描述

识别多个服务

由于接口可以有多个实现类,所以我们通过 ServiceLoader.load(StudentInterface.class); 有可能有多个实例,我们如何区分自己需要的哪一个。

impl.getClass().getSimpleName() 可以获取实现类的名字。如下:

    void test() {
        ServiceLoader<StudentInterface> serviceLoader = ServiceLoader.load(StudentInterface.class, StudentInterface.class.getClassLoader());
        for (StudentInterface impl : serviceLoader) {

            if (impl.getClass().getSimpleName().equals("StudentImpl")) {
                //这就是我们的需要的服务
                impl.eat("zyj");
            }
        }
    }

实例作用域问题

即使用了 auto-service ,也不能解决实例作用域问题,如果需要全局使用,就需要自己实现单例了,这里不再展开。

SPA应用实战1 —— 子模块如何获取主模块的BuildConfig信息

多模块开发/组件化开发过程中,主模块(plugin为com.android.application的模块,一般指app模块)可以依赖任何模块,但是子模块无法依赖主模块,如果子模块想拿主模块的内容要怎么办呢? 下面演示如何通过Spa来获取主模块的Context和BuildConfig中的内容。

先在接口层定义一个BuildService

public interface BuildService {
    String buglyId(); // build.gradle中使用buildConfigField定义的buglyId

    boolean debuggable();

    String versionName();

    int versionCode();

    String applicationId();

    String buildType();
}

在app模块中,实现这个service接口并使用 @AutoService 标记

BuildServiceImpl.java

@AutoService(BuildService.class)
public class BuildServiceImpl implements BuildService {
    @Override
    public String buglyId() {
        return BuildConfig.BUGLY_ID;
    }

    @Override
    public boolean debuggable() {
        return BuildConfig.DEBUG;
    }

    @Override
    public String versionName() {
        return BuildConfig.VERSION_NAME;
    }

    @Override
    public int versionCode() {
        return BuildConfig.VERSION_CODE;
    }

    @Override
    public String applicationId() {
        return BuildConfig.APPLICATION_ID;
    }

    @Override
    public String buildType() {
        return BuildConfig.BUILD_TYPE;
    }
}

准备工作已经完成,现在我们在pages模块的BuildInfoActivity中应用它

public class BuildInfoActivity extends AppCompatActivity {

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        BuildInfoPageBinding viewBinding = BuildInfoPageBinding.inflate(LayoutInflater.from(this));
        setContentView(viewBinding.getRoot());

        BuildService buildService = Spa.getService(BuildService.class);
        viewBinding.applicationId.setText("applicationId: " + buildService.applicationId());
        viewBinding.versionName.setText("versionName: " + buildService.versionName());
        viewBinding.versionCode.setText("versionCode: " + buildService.versionCode() + "");
        viewBinding.buildType.setText("buildType: " + buildService.buildType());
        viewBinding.debuggable.setText("debuggable: " + buildService.debuggable());
        viewBinding.buglyId.setText("buglyId:" + buildService.buglyId());
    }
}}

Auto Service 源码

https://github.com/google/auto/tree/master/service

  • 3
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值