博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
ARouter-万能路由协议
阅读量:6327 次
发布时间:2019-06-22

本文共 8073 字,大约阅读时间需要 26 分钟。

Github源码地址:

一、功能介绍

  1. 支持直接解析标准URL进行跳转,并自动解析参数注入
  2. 支持多模块工程使用
  3. 支持添加多个拦截器,自定义拦截顺序
  4. 支持依赖注入,可单独作为依赖注入框架使用
  5. 支持InstantRun
  6. 支持MultiDex(Google方案)
  7. 映射关系按组分类、多级管理,按需初始化
  8. 支持用户指定全局降级与局部降级策略
  9. 页面、拦截器、服务等组件均自动注册到框架

二、典型应用

  1. 从外部URL映射到内部页面,以及参数传递与解析
  2. 跨模块页面跳转,模块间解耦
  3. 拦截跳转过程,处理登陆、埋点等逻辑
  4. 跨模块API调用,通过控制反转来做组件解耦

三、基础功能

  1. 添加依赖和配置

    android {    defaultConfig {        ...        javaCompileOptions {            annotationProcessorOptions {                arguments = [ moduleName : project.getName() ]            }        }    }}dependencies {    compile 'com.alibaba:arouter-api:x.x.x'    annotationProcessor 'com.alibaba:arouter-compiler:x.x.x'    ...}// 旧版本gradle插件(< 2.2),可以使用apt插件,配置方法见文末'其他#4'

     

  2. 添加注解

    // 在支持路由的页面上添加注解(必选)// 这里的路径需要注意的是至少需要有两级,/xx/xx@Route(path = "/test/activity")public class YourActivity extend Activity {    ...}

     

  3. 初始化SDK

    if (isDebug()) {    ARouter.openLog();     // 打印日志    ARouter.openDebug();   // 开启调试模式(如果在InstantRun模式下运行,必须开启调试模式!线上版本需要关闭,否则有安全风险)}ARouter.init(mApplication); // 尽可能早,推荐在Application中初始化

     

  4. 发起路由操作

    // 1. 应用内简单的跳转(通过URL跳转在'进阶用法'中)ARouter.getInstance().build("/test/activity").navigation();// 2. 跳转并携带参数ARouter.getInstance().build("/test/1")            .withLong("key1", 666L)            .withString("key3", "888")            .navigation();

     

  5. 添加混淆规则(如果使用了Proguard)

    -keep public class com.alibaba.android.arouter.routes.**{*;}-keep class * implements com.alibaba.android.arouter.facade.template.ISyringe{*;}

     

四、进阶用法

  1. 通过URL跳转

    // 新建一个Activity用于监听Schame事件,之后直接把url传递给ARouter即可public class SchameFilterActivity extends Activity {    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        Uri uri = getIntent().getData();        ARouter.getInstance().build(uri).navigation();        finish();    }}// AndroidManifest.xml

     

  2. 解析URL中的参数

    // 为每一个参数声明一个字段,并使用 @Autowired 标注@Route(path = "/test/activity")public class Test1Activity extends Activity {    @Autowired    public String name;    @Autowired    private int age;    @Autowired(name = "girl") // 通过name来映射URL中的不同参数    private boolean boy;    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        ARouter.getInstance().inject(this);        // ARouter会自动对字段进行赋值,无需主动获取        Log.d("param", name + age + boy);    }}

     

  3. 声明拦截器(拦截跳转过程,面向切面编程)

    // 比较经典的应用就是在跳转过程中处理登陆事件,这样就不需要在目标页重复做登陆检查// 拦截器会在跳转之间执行,多个拦截器会按优先级顺序依次执行@Interceptor(priority = 8, name = "测试用拦截器")public class TestInterceptor implements IInterceptor {    @Override    public void process(Postcard postcard, InterceptorCallback callback) {        ...        callback.onContinue(postcard);  // 处理完成,交还控制权        // callback.onInterrupt(new RuntimeException("我觉得有点异常"));      // 觉得有问题,中断路由流程        // 以上两种至少需要调用其中一种,否则不会继续路由    }    @Override    public void init(Context context) {        // 拦截器的初始化,会在sdk初始化的时候调用该方法,仅会调用一次    }}

     

  4. 处理跳转结果

    // 使用两个参数的navigation方法,可以获取单次跳转的结果ARouter.getInstance().build("/test/1").navigation(this, new NavigationCallback() {    @Override    public void onFound(Postcard postcard) {          ...    }    @Override    public void onLost(Postcard postcard) {        ...    }});

     

  5. 自定义全局降级策略

    // 实现DegradeService接口,并加上一个Path内容任意的注解即可    @Route(path = "/xxx/xxx")    public class DegradeServiceImpl implements DegradeService {      @Override      public void onLost(Context context, Postcard postcard) {            // do something.      }      @Override      public void init(Context context) {      }    }

     

  6. 为目标页面声明更多信息

    // 我们经常需要在目标页面中配置一些属性,比方说"是否需要登陆"之类的// 可以通过 Route 注解中的 extras 属性进行扩展,这个属性是一个 int值,换句话说,单个int有4字节,也就是32位,可以配置32个开关// 剩下的可以自行发挥,通过字节操作可以标识32个开关,通过开关标记目标页面的一些属性,在拦截器中可以拿到这个标记进行业务逻辑判断@Route(path = "/test/activity", extras = Consts.XXXX)

     

  7. 通过依赖注入解耦:服务管理(一) 暴露服务

    // 声明接口,其他组件通过接口来调用服务public interface HelloService extends IProvider {    String sayHello(String name);}// 实现接口@Route(path = "/service/hello", name = "测试服务")public class HelloServiceImpl implements HelloService {    @Override    public String sayHello(String name) {        return "hello, " + name;    }    @Override    public void init(Context context) {    }}

     

  8. 通过依赖注入解耦:服务管理(二) 发现服务

    public class Test {    @Autowired    HelloService helloService;    @Autowired(name = "/service/hello")    HelloService helloService2;    HelloService helloService3;    HelloService helloService4;    public Test() {        ARouter.getInstance().inject(this);    }    public void testService() {         // 1. (推荐)使用依赖注入的方式发现服务,通过注解标注字段,即可使用,无需主动获取         // Autowired注解中标注name之后,将会使用byName的方式注入对应的字段,不设置name属性,会默认使用byType的方式发现服务(当同一接口有多个实现的时候,必须使用byName的方式发现服务)        helloService.sayHello("Vergil");        helloService2.sayHello("Vergil");        // 2. 使用依赖查找的方式发现服务,主动去发现服务并使用,下面两种方式分别是byName和byType        helloService3 = ARouter.getInstance().navigation(HelloService.class);        helloService4 = (HelloService) ARouter.getInstance().build("/service/hello").navigation();        helloService3.sayHello("Vergil");        helloService4.sayHello("Vergil");    }}

     

五、更多功能

  1. 初始化中的其他设置

    ARouter.openLog();      // 开启日志ARouter.openDebug();    // 使用InstantRun的时候,需要打开该开关,上线之后关闭,否则有安全风险ARouter.printStackTrace(); // 打印日志的时候打印线程堆栈

     

  2. 详细的API说明

    // 构建标准的路由请求ARouter.getInstance().build("/home/main").navigation();// 构建标准的路由请求,并指定分组ARouter.getInstance().build("/home/main", "ap").navigation();// 构建标准的路由请求,通过Uri直接解析Uri uri;ARouter.getInstance().build(uri).navigation();// 构建标准的路由请求,startActivityForResult// navigation的第一个参数必须是Activity,第二个参数则是RequestCodeARouter.getInstance().build("/home/main", "ap").navigation(this, 5);// 直接传递BundleBundle params = new Bundle();ARouter.getInstance()            .build("/home/main")            .with(params)            .navigation();// 指定FlagARouter.getInstance()            .build("/home/main")            .withFlags();            .navigation();// 觉得接口不够多,可以直接拿出Bundle赋值ARouter.getInstance()            .build("/home/main")            .getExtra();// 使用绿色通道(跳过所有的拦截器)ARouter.getInstance().build("/home/main").greenChannal().navigation();// 使用自己的日志工具打印日志ARouter.setLogger();

     

  3. 获取原始的URI

    String uriStr = getIntent().getStringExtra(ARouter.RAW_URI);

     

  4. 重写跳转URL

    // 实现PathReplaceService接口,并加上一个Path内容任意的注解即可@Route(path = "/xxx/xxx") // 必须标明注解public class PathReplaceServiceImpl implements DegradeService {    /**     * For normal path.     *     * @param path raw path     */    String forString(String path) {        return path;    // 按照一定的规则处理之后返回处理后的结果    }    /**     * For uri type.     *     * @param uri raw uri    */    Uri forUri(Uri uri) {        return url;    // 按照一定的规则处理之后返回处理后的结果    } }

     

六、其他

  1. 路由中的分组概念

    • SDK中针对所有的路径(/test/1 /test/2)进行分组,分组只有在分组中的某一个路径第一次被访问的时候,该分组才会被初始化
    • 可以通过 @Route 注解主动指定分组,否则使用路径中第一段字符串(/*/)作为分组
    • 注意:一旦主动指定分组之后,应用内路由需要使用 ARouter.getInstance().build(path, group) 进行跳转,手动指定分组,否则无法找到

      @Route(path = "/test/1", group = "app")
  2. 拦截器和服务的异同

    • 拦截器和服务所需要实现的接口不同,但是结构类似,都存在 init(Context context) 方法,但是两者的调用时机不同
    • 拦截器因为其特殊性,会被任何一次路由所触发,拦截器会在ARouter初始化的时候异步初始化,如果第一次路由的时候拦截器还没有初始化结束,路由会等待,直到初始化完成。
    • 服务没有该限制,某一服务可能在App整个生命周期中都不会用到,所以服务只有被调用的时候才会触发初始化操作
  3. Jack 编译链的支持

    • 因为不想让用户主动设置一堆乱七八糟的参数,在获取模块名的时候使用javac的api,使用了Jack之后没有了javac,只能让用户稍稍动动手了
    • 现在任何情况下都需要在build.gradle中配置moduleName了。。。。
  4. 旧版本gradle插件的配置方式

    apply plugin: 'com.neenbedankt.android-apt'buildscript {    repositories {        jcenter()    }    dependencies {        classpath 'com.neenbedankt.gradle.plugins:android-apt:1.4'    }}apt {    arguments {        moduleName project.getName();    }}dependencies {    compile 'com.alibaba:arouter-api:x.x.x'    apt 'com.alibaba:arouter-compiler:x.x.x'    ...}

     注意:应用宝使用乐固加固后,ARouter的启动器会失效,所以启动器需要改成:

public static void start(Context context, int communityId, String communityName) {        Intent starter = new Intent(context, CommunityApplyListActivity.class);        starter.putExtra("communityId", communityId);        starter.putExtra("communityName", communityName);        if (!(context instanceof Activity)) {            starter.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);        }        context.startActivity(starter);    }

 

转载地址:http://apgaa.baihongyu.com/

你可能感兴趣的文章
分析iOS Crash文件,使用命令符号化iOS Crash文件
查看>>
Java编译命令整理
查看>>
Java数据结构——链表-单链表
查看>>
mesos
查看>>
Sun Grid Engine (SGE)大型集群作业调度系统
查看>>
信号处理——生成给定分布随机数
查看>>
2014年上半年软件设计师考试之绝密答案--有待大家完好
查看>>
Java动态代理学习【Spring AOP基础之一】
查看>>
在cmd窗口输入命令遇到You must run this command from a command prompt with administrator privilege怎么办?...
查看>>
ElasticSearch入门 第五篇:使用C#查询文档
查看>>
设置数据库状态
查看>>
Android之读取 AndroidManifest.xml 中的数据:版本号、应用名称、自定义K-V数据(meta-data)...
查看>>
获取指定的内容---MXCMS ReadNews标签说明
查看>>
R学习笔记 第五篇:字符串操作
查看>>
在Mac OS下配置PHP开发环境
查看>>
(转)介绍下Nuget在传统Asp.net项目中的使用
查看>>
C# ArcEngine 实现点击要素高亮并弹出其属性
查看>>
初识GO语言——安装Go语言
查看>>
SDK命令行操作
查看>>
基于Bootstrap的DropDownList的JQuery组件的完善版
查看>>