Android技术栈(二)组件化改造(1),安卓面试宝典app

//ARouter注解处理器启动参数
javaCompileOptions {
annotationProcessorOptions {
arguments = [AROUTER_MODULE_NAME: project.getName()]
}
}

}

dependencies {
//省略…

//导入公共依赖
implementation project(‘:common’)
//声明ARouter注解处理器
annotationProcessor ‘com.alibaba:arouter-compiler:1.2.2’
}

我们在navi模块中使用@Route注解将PR.navi.navi映射到具体的Fragment或者Activity

这样:

@Route(path = PR.navi.navi)
public class NaviFragment extends Fragment

或者这样:

@Route(path = PR.navi.navi)
public class NaviActivity extends AppCompatActivity

ARouter这种使用path解耦的方式允许我们在开发的过程中更换PR.navi.navi映射到的FragmentActivity,而在代码修改时把对调用方的影响降低到最小.

但值得注意的是,ARouter对不同类型的处理是不一样的,如果path指向的是Fragment,你需要获取navigation的返回值并手动把它添加到FragmentManager中.(如果不了解Fragment的同学可以看这篇文章 从Activity迁移到Fragment)

Fragment fragment = (Fragment) ARouter.getInstance()
.build(PR.navi.navi)
.navigation();

Activity则不需要,它会立即显示

ARouter.getInstance()
.build(PR.navi.navi)
//还可以设置参数,ARouter会帮你存在Bundle中
.withString(“pathId”,UUID.randomUUID().toString())
//Activity 或 Context
.navigation(this);

navi模块是典型的业务逻辑模块,这里你可导入一些只有这个模块才会使用的专属第三方SDK,比如我在navi模块中使用了高德地图SDK,其他模块只需要我这个模块的地图功能,但它不应该知道我到底使用的是高德还是百度还是腾讯地图,这就提高了封装性,在未来改变此模块的具体实现时,代价也会小得多.

image.png

3.自定义全局拦截器、全局降级策略、全局重定向

ARouter实现了module间的路由操作,同时也实现了拦截器的功能,拦截器是一种AOP(面向切面编程),比较经典的使用场景就是处理页面登录与否的问题.拦截器会在跳转之间执行,多个拦截器会按优先级顺序依次执行.通过实现IInterceptor接口并标注@Interceptor注解,这样一来,这个拦截器就被注册到ARouter当中了.

process方法会传入PostcardInterceptorCallback,Postcard携带此次路由的关键信息,而InterceptorCallback则用于处理此次拦截,调用onContinue则放行,又或者调用onInterrupt抛出自定义异常.

拦截器会在ARouter初始化的时候进行异步(不在主线程)初始化,如果第一次路由发生时,还有拦截器没有初始化完毕,那么ARouter会等待该拦截器初始化完毕才进行路由.

@Interceptor(priority = 8)
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) {
// 拦截器的初始化,会在ARouter初始化的时候调用该方法,仅会调用一次
}
}

当页面未找到时,我们可以定义一种降级策略来让程序继续运行,此时我们需要实现DegradeService接口,并用@Route(必须)标注,然后它会在全局范围内生效,你可以在onLost回调中自定义降级逻辑.

@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) {

}
}

有时候页面我们需要将path其重定向别的path,这时我们可以实现PathReplaceService接口,并用@Route(必须)标注,然后它会在全局范围内生效.所以若没有重定向需求记得返回原path

@Route(path = “/xxx/xxx”)
public class PathReplaceServiceImpl implements PathReplaceService {
String forString(String path) {
return path; // 按照一定的规则处理之后返回处理后的结果
}
Uri forUri(Uri uri) {
return url; // 按照一定的规则处理之后返回处理后的结果
}

@Override
public void init(Context context) {

}
}

以上上三种接口中的init方法,只有拦截器的调用时间是特殊的,其他两种,都是在第一次使用时才会进行初始化.

4.接口下沉->暴露服务

有的时候我们可能需要的不是另外一个模块的页面,而是它提供的服务(MVC中的Model层),这时这时我们需要为自己想要的服务编写一个接口,并让他实现IProvider接口,然后把它放到common模块中, 但是接口的实现依然放在非common的具体的模块中,比如common模块的TTSServiceai模块的TTSServiceImpl.

这种做法被称为接口下沉,其实它并不是严格符合解耦思想的,但是它非常有用,就像你使用了ARouter,但没人规定你就不能用startActivity了一样,框架最终的目的还是为了方便我们编码的,而不是为了给我们添堵,更何况最终结果各模块依然是松散耦合的.

服务的初始化时机也是在第一次使用的时候.我们在common模块中声明TTSService接口:

public interface TTSService extends IProvider
{
void send(String text);

void stop();
}

并在ai模块中实现它并使用@Route注解标注

@Route(path = PR.ai.tts_service)
public class TTSServiceImpl implements TTSService
{
//省略…
}

这样我们就能在其他模块使用该服务了

TTSService ttsService = (TTSService) ARouter.getInstance()
.build(PR.ai.tts_service)
.navigation()

5.ContentProvider->模块内的Application

有些第三方SDK初始化是必须要在ApplicationonCreate中进行初始化的,但是如果我们编写独立于hostmodule时,要怎么初始化它们呢?

ARouter并没有提供官方的解决方案,但是经过我的实践,我们可以通过声明ContentProvider并在模块内AndroidManifest中注册它来实现初始化功能.

//java
public class ModuleLoader extends ContentProvider
{
@Override
public boolean onCreate()
{
Context context = getContext();
//TODO
return true;
}

//…

}

//AndroidManifest

ContentProvider#onCreateApplication#attachBaseContext调用之后Application#onCreate调用之前执行,并且可以通过getContext拿到ApplicationContext.这样就解决了部分第三方SDK初始化的问题.

6.ARouter是如何实现的?

简单概括起来其实也就是两个知识点:

  • 使用APT注解处理器通过注解生成RouteMeta元数据到指定包下
  • 启动时扫描Dex指定包下class,加载并缓存路由表,然后在navigation是对path映射到的不同类型尽可能地抽象出同一套接口

如果还想深入理解ARouter,可能就需要去读源码了.

7.ARouter的缺点

ARouter目前暂时不支持多进程开发,这是我觉得比较遗憾的,希望未来能够支持吧.

8.结语

ARouter的介绍就到此为止了,如果还想了解ARouter的依赖注入功能请移步Github.

【附】相关架构及资料

image

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)
img

如何做好面试突击,规划学习方向?

面试题集可以帮助你查漏补缺,有方向有针对性的学习,为之后进大厂做准备。但是如果你仅仅是看一遍,而不去学习和深究。那么这份面试题对你的帮助会很有限。最终还是要靠资深技术水平说话。

网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。建议先制定学习计划,根据学习计划把知识点关联起来,形成一个系统化的知识体系。

学习方向很容易规划,但是如果只通过碎片化的学习,对自己的提升是很慢的。

我们搜集整理过这几年字节跳动,以及腾讯,阿里,华为,小米等公司的面试题,把面试的要求和技术点梳理成一份大而全的“ Android架构师”面试 Xmind(实际上比预期多花了不少精力),包含知识脉络 + 分支细节。

img

我们在搭建这些技术框架的时候,还整理了系统的高级进阶教程,会比自己碎片化学习效果强太多

知识体系。

学习方向很容易规划,但是如果只通过碎片化的学习,对自己的提升是很慢的。

我们搜集整理过这几年字节跳动,以及腾讯,阿里,华为,小米等公司的面试题,把面试的要求和技术点梳理成一份大而全的“ Android架构师”面试 Xmind(实际上比预期多花了不少精力),包含知识脉络 + 分支细节。

[外链图片转存中…(img-h0fbcdCA-1711921897309)]

我们在搭建这些技术框架的时候,还整理了系统的高级进阶教程,会比自己碎片化学习效果强太多

本文已被CODING开源项目:《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》收录

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值