ARouter讲解2-AutowiredProcessor,安卓开发面试题目

// 方法里的 elements 参数是字段元素,是字段一般都是在类或者接口里面
private void categories(Set<? extends Element> elements) throws IllegalAccessException {
if (CollectionUtils.isNotEmpty(elements)) {
for (Element element : elements) {
// 获取这个字段所在的类或接口
TypeElement enclosingElement = (TypeElement) element.getEnclosingElement();
// 该类或者接口不能被 private 修饰
if (element.getModifiers().contains(Modifier.PRIVATE)) {
throw new IllegalAccessException(“The inject fields CAN NOT BE ‘private’!!! please check field [”

  • element.getSimpleName() + “] in class [” + enclosingElement.getQualifiedName() + “]”);
    }

if (parentAndChild.containsKey(enclosingElement)) { // Has categries
parentAndChild.get(enclosingElement).add(element);
} else {
List childs = new ArrayList<>();
childs.add(element);
parentAndChild.put(enclosingElement, childs);
}
}

logger.info(“categories finished.”);
}
}

image-20210511215138090.png

parentAndChild就是收集类中所有被Autowired 注解的字段。

package com.alibaba.android.arouter.facade.template;

public interface ISyringe {
void inject(Object target);
}

private void generateHelper() throws IOException, IllegalAccessException {

// 先准备一些element
// com.alibaba.android.arouter.facade.template.ISyringe
TypeElement type_ISyringe = elementUtils.getTypeElement(ISYRINGE);
// com.alibaba.android.arouter.facade.service.SerializationService 这个是一个 IProvider ,我们后面讲
TypeElement type_JsonService = elementUtils.getTypeElement(JSON_SERVICE);

// com.alibaba.android.arouter.facade.template.IProvider
TypeMirror iProvider = elementUtils.getTypeElement(Consts.IPROVIDER).asType();

// android.app.Activity
TypeMirror activityTm = elementUtils.getTypeElement(Consts.ACTIVITY).asType();

// android.app.Fragment
TypeMirror fragmentTm = elementUtils.getTypeElement(Consts.FRAGMENT).asType();

// android.support.v4.app.Fragment
TypeMirror fragmentTmV4 = elementUtils.getTypeElement(Consts.FRAGMENT_V4).asType();

// Build input param name.
// 方法的参数类型与参数名
ParameterSpec objectParamSpec = ParameterSpec.builder(TypeName.OBJECT, “target”).build();

if (MapUtils.isNotEmpty(parentAndChild)) {
// 遍历一个每一个类内部带有Autowired 注解的字段
for (Map.Entry<TypeElement, List> entry : parentAndChild.entrySet()) {
// Build method : ‘inject’
// 构建一个方法,public void inject(Object target)
MethodSpec.Builder injectMethodBuilder = MethodSpec.methodBuilder(METHOD_INJECT)
.addAnnotation(Override.class)
.addModifiers(PUBLIC)
.addParameter(objectParamSpec);

TypeElement parent = entry.getKey();
List childs = entry.getValue();

// 该类的全限定名
String qualifiedName = parent.getQualifiedName().toString();
// 类的包名
String packageName = qualifiedName.substring(0, qualifiedName.lastIndexOf(“.”));
// 类的名字,注意 生成薪的类的后面追加了 A R o u t e r ARouter ARouterAutowired
String fileName = parent.getSimpleName() + NAME_OF_AUTOWIRED;

// 构建一个类,实现 ISyringe 接口,访问方式是public
TypeSpec.Builder helper = TypeSpec.classBuilder(fileName)
.addJavadoc(WARNING_TIPS)
.addSuperinterface(ClassName.get(type_ISyringe))
.addModifiers(PUBLIC);

// 添加一个成员变量 serializationService
FieldSpec jsonServiceField = FieldSpec.builder(TypeName.get(type_JsonService.asType()), “serializationService”, Modifier.PRIVATE).build();
helper.addField(jsonServiceField);

// 在inject 方法内部
// serializationService = ARouter.getInstance().navigation(SerializationService.class);
// serializationService 的作用是序列化对象,将对象转换成json进行传递
injectMethodBuilder.addStatement(“serializationService = T . g e t I n s t a n c e ( ) . n a v i g a t i o n ( T.getInstance().navigation( T.getInstance().navigation(T.class)”, ARouterClass, ClassName.get(type_JsonService));
injectMethodBuilder.addStatement(“ T s u b s t i t u t e = ( T substitute = ( Tsubstitute=(T)target”, ClassName.get(parent), ClassName.get(parent));

// Generate method body, start inject.
for (Element element : childs) {
Autowired fieldConfig = element.getAnnotation(Autowired.class);
// 元素的名字
String fieldName = element.getSimpleName().toString();

if (types.isSubtype(element.asType(), iProvider)) { // It’s provider
// 如果 element 是 IProvider 的子类
if (“”.equals(fieldConfig.name())) { // User has not set service path, then use byType.

// Getter
// substitute.{fieldName} = ARouter.getInstance().navigation( {elementClass}.class),{}表示占位
injectMethodBuilder.addStatement(
“substitute.” + fieldName + " = T . g e t I n s t a n c e ( ) . n a v i g a t i o n ( T.getInstance().navigation( T.getInstance().navigation(T.class)“,
ARouterClass,
ClassName.get(element.asType())
);
} else { // use byName
// Getter
// substitute.{fieldName} =({elementClass}) ARouter.getInstance().build({ConfigName}).navigation()
injectMethodBuilder.addStatement(
“substitute.” + fieldName + " = ( T ) T) T)T.getInstance().build($S).navigation()”,
ClassName.get(element.asType()),
ARouterClass,
fieldConfig.name()

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

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

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

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

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

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

总结

现在新技术层出不穷,如果每次出新的技术,我们都深入的研究的话,很容易分散精力。新的技术可能很久之后我们才会在工作中用得上,当学的新技术无法学以致用,很容易被我们遗忘,到最后真的需要使用的时候,又要从头来过(虽然上手会更快)。

我觉得身为技术人,针对新技术应该是持拥抱态度的,入了这一行你就应该知道这是一个活到老学到老的行业,所以面对新技术,不要抵触,拥抱变化就好了。

Flutter 明显是一种全新的技术,而对于这个新技术在发布之初,花一个月的时间学习它,成本确实过高。但是周末花一天时间体验一下它的开发流程,了解一下它的优缺点、能干什么或者不能干什么。这个时间,并不是我们不能接受的。

如果有时间,其实通读一遍 Flutter 的文档,是最全面的一次对 Flutter 的了解过程。但是如果我们只有 8 小时的时间,我希望能关注一些最值得关注的点。

(跨平台开发(Flutter)、java基础与原理,自定义view、NDK、架构设计、性能优化、完整商业项目开发等)

ter 的了解过程。但是如果我们只有 8 小时的时间,我希望能关注一些最值得关注的点。

(跨平台开发(Flutter)、java基础与原理,自定义view、NDK、架构设计、性能优化、完整商业项目开发等)

[外链图片转存中…(img-nH7u6btC-1711743535695)]

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

  • 22
    点赞
  • 27
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值