ARouter系列3:继续学习(手写一个Arouter框架)

本文介绍了如何在Android项目中使用APT(AnnotationProcessingTool)技术,通过注解自动为不同模块生成处理类,实现ARouter框架中Activity的自动映射,提高代码维护性和扩展性。
摘要由CSDN通过智能技术生成

}

}

}

return aRouterTest;

}

/**

  • 向路由表中添加信息

  • @param key 键

  • @param clazz 值

*/

public void addActivity(String key, Class<? extends Activity> clazz) {

if (key != null && clazz != null && !map.containsKey(key)) {

map.put(key, clazz);

}

}

/**

  • 跳转窗体的方法

  • @param key 键

  • @param bundle 值

*/

public void jumpActivity(String key, Bundle bundle) {

Class<? extends Activity> classActivity = map.get(key);

if (classActivity != null) {

Intent intent = new Intent(context, classActivity);

if (bundle != null) {

intent.putExtras(bundle);

}

context.startActivity(intent);

}

}

}

接下来,我们需要把每个module的activity类信息存储到ARouterTest中,写一个接口类。

2.2.2、IRouterTest.java

public interface IRouterTest {

void putActivity();

}

2.2.3、ARouterConstant.java

public class ARouterConstant {

public static final String AROUTER_PARAM_LOGIN_ACTIVITY = “login/LoginActivity”;

public static final String AROUTER_PARAM_UCMAIN_ACTIVITY = “uc/UCMainActivity”;

}

2.2.4、ActivityUtil.java

public class ActivityUtil implements IRouterTest {

@Override

public void putActivity() {

ARouterTest.getInstance().addActivity(ARouterConstant.AROUTER_PARAM_LOGIN_ACTIVITY, LoginActivity.class);

}

}

但是这种手动添加的方式并不利于后期的维护。因此我们需要APT技术来帮助我们自动生成这些类。

3、APT技术

=======

【Android】APT

APT(Annotation Processing Tool)即注解处理器,是一种处理注解的工具,确切的说它是javac的一个工具,它用来在编译时扫描和处理注解。注解处理器以Java代码(或者编译过的字节码)作为输入,生成**.java文件**作为输出。

简单来说就是在编译期,通过注解生成**.java**文件。

APT技术的核心有两块:注解和注解处理器。

4、手写一个ARouter框架

===============

接下来我们利用APT技术自己手写一个ARouter框架。

结构目录:

这个项目架构,一共四个android module:app、login、usercenter、arouter,其中app依赖另外三个module,arouter被其他三个module依赖。

apt-annotation 和 apt-processor 是java module,这两个module被app、login和usercenter依赖。如图:

implementation project(path: ‘:apt-annotation’)

annotationProcessor project(path: ‘:apt-processor’)

4.1、module:apt-annotation


该module只新建了一个注解类。

// 声明注解的作用域

@Target(ElementType.TYPE)

// 声明注解的生命周期

@Retention(RetentionPolicy.CLASS)

public @interface BindPath {

String path();

}

4.2、module:apt-processor


apply plugin: ‘java-library’

dependencies {

implementation fileTree(dir: ‘libs’, include: [‘*.jar’])

implementation project(path: ‘:apt-annotation’)

annotationProcessor ‘com.google.auto.service:auto-service:1.0-rc4’

compileOnly ‘com.google.auto.service:auto-service:1.0-rc3’

}

sourceCompatibility = “1.7”

targetCompatibility = “1.7”

@AutoService(Processor.class)

public class Test1Processor extends AbstractProcessor {

Filer filer;

@Override

public synchronized void init(ProcessingEnvironment processingEnv) {

super.init(processingEnv);

filer = processingEnv.getFiler();

}

@Override

public SourceVersion getSupportedSourceVersion() {

return processingEnv.getSourceVersion();

}

@Override

public Set getSupportedAnnotationTypes() {

Set types=new HashSet<>();

types.add(BindPath.class.getCanonicalName());

return types;

}

@Override

public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {

// 获取到当前模块中用到了BindPath注解的activity的类对象(类节点) (有几个模块中依赖了apt-processor,该方法就会执行几次)

// 类节点 TypeElement 方法节点 ExecutableElement 方法节点 VariableElement

Set<? extends Element> elementAnnotationWith = roundEnv.getElementsAnnotatedWith(BindPath.class);

Map<String, String> map = new HashMap<>();

// 遍历整个模块中用到了BindPath注解的节点

for (Element element : elementAnnotationWith) {

TypeElement typeElement = (TypeElement) element;

// 获取到activity上面的 BindPath 的注解

BindPath annotation = typeElement.getAnnotation(BindPath.class);

// 获取到注解里面带的值 中间容器 map 的activity所对应的 key

String key = annotation.path();

// 获取到包名和类名

Name activityName = typeElement.getQualifiedName();

map.put(key, activityName + “.class”);

}

// 写文件

if (map.size() > 0) {

Writer writer = null;

// 需要生成的文件名 让类名不重复

String activityName = “ActivityUtil” + System.currentTimeMillis();

try {

// 生成一个Java文件

JavaFileObject sourceFile = filer.createSourceFile(“com.gs.util.” + activityName);

// 从生成的这个文件开始写

writer = sourceFile.openWriter();

StringBuffer stringBuffer = new StringBuffer();

stringBuffer.append(“package com.gs.util;\n”);

stringBuffer.append(“import com.gs.arouter.ARouterTest;\n” +

“import com.gs.arouter.IRouterTest;\n” +

“\n” +

“public class " + activityName + " implements IRouterTest{\n” +

" @Override\n" +

" public void putActivity(){\n");

Iterator iterator = map.keySet().iterator();

while (iterator.hasNext()) {

String key = iterator.next();

String className = map.get(key);

stringBuffer.append(" ARouterTest.getInstance().addActivity(“” + key + “”, " +

className + “);”);

}

stringBuffer.append(“\n }\n}”);

writer.write(stringBuffer.toString());

} catch (IOException e) {

e.printStackTrace();

} finally {

if (writer != null) {

try {

writer.close();

} catch (IOException e) {

e.printStackTrace();

}

}

}

}

return false;

}

}

这个很重要,不要搞错了。

4.3、module:arouter


4.3.1、IRouterTest.java

public interface IRouterTest {

void putActivity();

}

4.3.2、ARouterConstant.java

public class ARouterConstant {

public static final String AROUTER_PARAM_LOGIN_ACTIVITY = “login/LoginActivity”;

public static final String AROUTER_PARAM_UCMAIN_ACTIVITY = “uc/UCMainActivity”;

}

4.3.3、 ARouterTest.java

public class ARouterTest {

public volatile static ARouterTest aRouterTest;

private Context context;

private Map<String, Class<? extends Activity>> map;

private ARouterTest() {

map = new HashMap<>();

}

public static ARouterTest getInstance() {

if (aRouterTest == null) {

synchronized (ARouterTest.class) {

if (aRouterTest == null) {

aRouterTest = new ARouterTest();

}

}

}

return aRouterTest;

}

public void init(Context context) {

this.context = context;

List classNames = getClassName(“com.gs.util”);

for (String className : classNames) {

try {

Class<?> clazz = Class.forName(className);

if (IRouterTest.class.isAssignableFrom(clazz)) {

IRouterTest iRouterTest = (IRouterTest) clazz.newInstance();

iRouterTest.putActivity();

}

} catch (Exception e) {

e.printStackTrace();

}

}

}

public void addActivity(String key, Class<? extends Activity> clazz) {

if (key != null && clazz != null && !map.containsKey(key)) {

map.put(key, clazz);

}

}

public void jumpActivity(String key, Bundle bundle) {

Class<? extends Activity> classActivity = map.get(key);

if (classActivity != null) {

Intent intent = new Intent(context, classActivity);

if (bundle != null) {

intent.putExtras(bundle);

}

intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);

context.startActivity(intent);

}

}

/**

  • 通过包名获取这个包下面的所有类名

  • @param packageName 包名

  • @return a list of class

*/

private List getClassName(String packageName) {

List classList = new ArrayList<>();

String path = null;

try {

path = context.getPackageManager()

.getApplicationInfo(context.getPackageName(), 0)

.sourceDir;

DexFile dexFile = new DexFile(path);

Enumeration entries = dexFile.entries();

while (entries.hasMoreElements()) {

String name = (String) entries.nextElement();

if (name.contains(packageName)) {

classList.add(name);

}

}

} catch (Exception e) {

e.printStackTrace();

}

return classList;

}

}

4.4、使用


4.4.1、module:app

App.java

public class App extends Application {

@Override

public void onCreate() {

super.onCreate();

ARouterTest.getInstance().init(this);

}

}

MainActivity.java

public class MainActivity extends AppCompatActivity {

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

findViewById(R.id.btn1).setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

ARouterTest.getInstance().jumpActivity(ARouterConstant.AROUTER_PARAM_LOGIN_ACTIVITY, null);

}

});

}

}

activity_main

<?xml version="1.0" encoding="utf-8"?>

<LinearLayout

xmlns:android=“http://schemas.android.com/apk/res/android”

xmlns:app=“http://schemas.android.com/apk/res-auto”

xmlns:tools=“http://schemas.android.com/tools”

android:layout_width=“match_parent”

android:layout_height=“match_parent”

android:orientation=“vertical”>

<Button

结语

看到这篇文章的人不知道有多少是和我一样的Android程序员。

35岁,这是我们这个行业普遍的失业高发阶段,这种情况下如果还不提升自己的技能,进阶发展,我想,很可能就是本行业的职业生涯的终点了。

我们要有危机意识,切莫等到一切都成定局时才开始追悔莫及。只要有规划的,有系统地学习,进阶提升自己并不难,给自己多充一点电,你才能走的更远。

千里之行始于足下。这是上小学时,那种一元钱一个的日记本上每一页下面都印刷有的一句话,当时只觉得这句话很短,后来渐渐长大才慢慢明白这句话的真正的含义。

有了学习的想法就赶快行动起来吧,不要被其他的事情牵绊住了前行的脚步。不要等到裁员时才开始担忧,不要等到面试前一晚才开始紧张,不要等到35岁甚至更晚才开始想起来要学习要进阶。

给大家一份系统的Android学习进阶资料,希望这份资料可以给大家提供帮助。

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!
id.com/apk/res/android"

xmlns:app=“http://schemas.android.com/apk/res-auto”

xmlns:tools=“http://schemas.android.com/tools”

android:layout_width=“match_parent”

android:layout_height=“match_parent”

android:orientation=“vertical”>

<Button

结语

看到这篇文章的人不知道有多少是和我一样的Android程序员。

35岁,这是我们这个行业普遍的失业高发阶段,这种情况下如果还不提升自己的技能,进阶发展,我想,很可能就是本行业的职业生涯的终点了。

我们要有危机意识,切莫等到一切都成定局时才开始追悔莫及。只要有规划的,有系统地学习,进阶提升自己并不难,给自己多充一点电,你才能走的更远。

千里之行始于足下。这是上小学时,那种一元钱一个的日记本上每一页下面都印刷有的一句话,当时只觉得这句话很短,后来渐渐长大才慢慢明白这句话的真正的含义。

有了学习的想法就赶快行动起来吧,不要被其他的事情牵绊住了前行的脚步。不要等到裁员时才开始担忧,不要等到面试前一晚才开始紧张,不要等到35岁甚至更晚才开始想起来要学习要进阶。

给大家一份系统的Android学习进阶资料,希望这份资料可以给大家提供帮助。
[外链图片转存中…(img-khI7GfQW-1714932581686)]

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值