搭建ARouter环境
源码地址 ARouter
android {
defaultConfig {
...
javaCompileOptions {
annotationProcessorOptions {
arguments = [AROUTER_MODULE_NAME: project.getName()]
}
}
}
}
dependencies {
api 'com.alibaba:arouter-api:x.x.x'
annotationProcessor 'com.alibaba:arouter-compiler:x.x.x'
...
}
build.gradle 添加引用依赖
public class MyApplication extends Application {
@Override
public void onCreate() {
super.onCreate();
if (BuildConfig.DEBUG) {
ARouter.openDebug();
ARouter.openLog();
}
ARouter.init(this);
}
}
MyApplication 添加初始化
public class MainActivity extends AppCompatActivity {
...
public void click1(View view) {
ARouter.getInstance().build("/app/main2").navigation();
}
}
import com.alibaba.android.arouter.facade.annotation.Route;
@Route(path = "/app/main2")
public class Main2Activity extends AppCompatActivity {
...
}
创建MainActivity,设置一个点击事件通过路由方法跳转Main2Activity
以上是一个简单的实践
源码探索 ARouter.init(this);
首先在刚才 Application 中初始化的时候我们调用了 ARouter.init(this);
public final class ARouter {
...
public static void init(Application application) {
...
hasInit = _ARouter.init(application);
...
}
}
先进入 ARouter.class
final class _ARouter {
...
private volatile static ThreadPoolExecutor executor = DefaultPoolExecutor.getInstance();
...
protected static synchronized boolean init(Application application) {
mContext = application;
LogisticsCenter.init(mContext, executor);
...
return true;
}
再进入下一步 _ARouter.class
- 1 判断是否是debug模式或者首次启动
- 2 从apk文件中读取 ARouter 生成的文件
- 3 将读取到的className map 通过 SharedPreferences 保存下来
- 4 之后App再启动的时候就不用去读文件,直接存缓存的 SP 中读取
- 5 遍历map,将读取到的数据保存到 Warehouse.class 的静态常量 Map 中
public class ARouter$$Group$$app implements IRouteGroup {
@Override
public void loadInto(Map atlas) {
atlas.put("/app/main2", RouteMeta.build(RouteType.ACTIVITY, Main2Activity.class, "/app/main2", "app", null, -1, -2147483648));
}
}
public class ARouter$$Root$$app implements IRouteRoot {
@Override
public void loadInto(Map> routes) {
routes.put("app", ARouter$$Group$$app.class);
}
}
public class ARouter$$Providers$$app implements IProviderGroup {
@Override
public void loadInto(Map providers) {
}
}
public interface IRouteGroup {
public interface IRouteRoot {
public interface IInterceptorGroup {
void loadInto(Map> interceptor);
}
以上是 ARouter 生成的文件,继承的接口都是用于读取数据保存到 Warehouse.class 用的
源码探索 ARouter.getInstance().build(String path)
- 1 PathReplaceService extends IProvider,当你的一个模块有一个类继承了PathReplaceService,并重写了它的两个方法方法,在这里就可以进行path的预处理,动态替换我们最终的目的地。
- 2 进入下一步4
- 3 这个方法校验你的path是否写的正确,格式必须是 /xx/xx ,这个reture 的是 “app”
- 3A 这里将继承了 IProvider 的类进行init,和fragment一起都是绿色通道默认不拦截
- 4 最终生成一个 Postcard 对象
public final class Postcard extends RouteMeta {
// Base
private Uri uri;
private Object tag; // A tag prepare for some thing wrong.
private Bundle mBundle; // Data to transform
private int flags = -1; // Flags of route
private int timeout = 300; // Navigation timeout, TimeUnit.Second
private IProvider provider; // It will be set value, if this postcard was provider.
private boolean greenChannel;
private SerializationService serializationService;
// Animation
private Bundle optionsCompat; // The transition animation of activity
private int enterAnim = -1;
private int exitAnim = -1;
public class RouteMeta {
private RouteType type; // Type of route
private Element rawType; // Raw type of route
private Class> destination; // Destination
private String path; // Path of route
private String group; // Group of route
private int priority = -1; // The smaller the number, the higher the priority
private int extra; // Extra data
private Map paramsType; // Param type
private String name;
private Map injectConfig; // Cache inject config.
Postcard 继承 RouteMeta,他们保存了 Intent 的参数
源码探索 ARouter.getInstance().build("/app/main2").navigation();
- 1 Postcard 写入到 LogisticsCenter 保存起来
- 2 Warehouse 中缓存的map数据,通过postcard.getPath()当key,查找出已经缓存的数据
- 3 将对应缓存的数据,赋值到 postcard
- 4 这边是绿色通道,不受拦截的意思,可以使用方法 ARouter.getInstance().build(“xx").greenChannel() 来调用,这部分还有拦截器的操作,那个callback 就是拦截器的监听事件,注册了就会开始走这部分逻辑
- 5 下一步
- 6 开始区分类型,这是打开 Activity 页面的操作,当 context 不是 instanceof Activity时,表示当前不是从APP内部打开的,需要加上 flag Intent.FLAG_ACTIVITY_NEW_TASK,打开一个新的页面
- 7 判断是否主线程,最终由主线程来调用startActivity方法
- 8 requestCode >=0 表示需要下一个页面的反馈然后调用 startActivityForResult,并传递Bundle数据
- 9 页面跳转的动画
- 10 类型的区分,比较熟悉的Fragment通过反射拿到实例,并判断Fragment的版本类型,provider 类型的直接返回接口实例,可以强转成本地的对象,这样就能调用对应的本地方法了
TestService mNavigation = (TestService) ARouter.getInstance().build("/test/service").navigation();
mTextView.setText(mNavigation.getNum());
@Route(path = "/test/service")
public class TestService implements IProvider {
@Override
public void init(Context context) {
Log.d("TestService", "init");
}
public String getNum() {
return String.valueOf(Constant.NUM);
}
}
这是一个 IProvider 的操作的实例效果,获取其他模块的简单数据操作
这部分代码链接
这样下来基本的源码流程已经走完,剩下的一些操作传数据拦截器什么都源码中也都能很直观的看出来
探索Android路由框架-ARouter之深挖源码(二)
谈谈App的统一跳转和ARouter