Android组件化之路由

项目搭建

1.需求分析

2.功能模块划分

3.基础工具类、框架、接口提取

4.创建路由接口

5.配置gradle完成模块单独测试

if (appb_lib) {
   apply plugin: 'com.android.library'
}else {
   apply plugin: 'com.android.application'
}

Demo地址:https://github.com/zhanpple/MyProject

安装 Octotree浏览器插件查看项目结构

image

baseMoudle路由工具类 RouterTools

    //添加router与activity对应关系
    public void addRouter(String router, Class<? extends Activity> clazz) {
        mRouterMap.put(router, clazz);
    }
    //根据router跳转界面
    public void navigate(Context context, String router) {
        System.out.println(mRouterMap.get(router));
        context.startActivity(new Intent(context, mRouterMap.get(router)));
    }

    //初始化router与activity对应关系数据
    public void init(Context context) {
        //用于asm增强
        loadRouterMap();
        if (isUseAms) {
            return;
        }
        
        //如果没有使用asm 则采用遍历apk中的所有class,找到实现IRouter的类反射调用
        String packageResourcePath = context.getApplicationContext().getPackageResourcePath();
        try {
            DexFile dexFile = new DexFile(packageResourcePath);
            Enumeration<String> entries = dexFile.entries();
            while (entries.hasMoreElements()) {
                String element = entries.nextElement();
                Log.e("RouterTools", "init: " + element);
                if (element.contains("com.example.router")) {
                    Log.e("RouterTools", "init: " + element);
                    Class<?> aClass = Class.forName(element);
                    int modifiers = aClass.getModifiers();
                    if (IRouter.class.isAssignableFrom(aClass) && !Modifier.isInterface(modifiers)) {
                        ((IRouter) aClass.newInstance()).addRouter(mRouterMap);
                    }
                }
            }
        } catch (Exception e) {
            e.printStackTrace();
            Log.e("RouterTools", "init: ", e);
        }

        for (Map.Entry<String, Class<? extends Activity>> stringClassEntry : mRouterMap.entrySet()) {
            Log.e("RouterTools", "stringClassEntry: "
                    + stringClassEntry.getKey() + "--" + stringClassEntry.getValue());

        }
    }

    //用于asm增强函数
    private static void loadRouterMap() {
        isUseAms = false;
        //在此处完成字节码插桩
        //register("com.example.router.appc.MyRouter$$APPC")
        //register("com.example.router.appb.MyRouter$$APPB")
    }

    //用于asm调用,如果使用了ams就不需要遍历apk
    private static void register(String className) {
        isUseAms = true;
        Log.e("RouterTools", "register: " + className);
        try {
            Class<?> aClass = Class.forName(className);
            int modifiers = aClass.getModifiers();
            if (IRouter.class.isAssignableFrom(aClass) && !Modifier.isInterface(modifiers)) {
                ((IRouter) aClass.newInstance()).addRouter(mRouterMap);
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

}

APT自动生成个个模块实现IRouter的类

@AutoService(Processor.class)
@SupportedAnnotationTypes({"com.example.annotation.MyRouter"})
@SupportedSourceVersion(SourceVersion.RELEASE_8)
@SupportedOptions({"MODULE_NAME"})
public class MyRouterProcessor extends AbstractProcessor {

    private Filer filer;

    private Elements elementUtils;

    private Messager printMessage;

    private ArrayList<MyRouterBean> classNames = new ArrayList<>();

    @Override
    public synchronized void init(ProcessingEnvironment processingEnvironment) {
        super.init(processingEnvironment);
        filer = processingEnv.getFiler();
        elementUtils = processingEnv.getElementUtils();
        printMessage = processingEnv.getMessager();
        println("init-----------------------------------------------");
    }

    @Override
    public boolean process(Set<? extends TypeElement> set, RoundEnvironment roundEnvironment) {
        classNames.clear();
        println("process-----------------------------------------------");
        /** 获取对应模块的  MODULE_NAME
        javaCompileOptions {
            annotationProcessorOptions {
                arguments = [MODULE_NAME: project.getName()]
            }
        }
        **/
        String module_name = processingEnv.getOptions().get("MODULE_NAME");
        Set<? extends Element> elements = roundEnvironment.getElementsAnnotatedWith(MyRouter.class);
        //遍历找到带MyRouter注解的Activity
        for (Element element : elements) {
            if (element instanceof TypeElement) {//类注解
                TypeElement typeElement = (TypeElement) element;
                PackageElement packageOf = elementUtils.getPackageOf(typeElement);
                Name qualifiedName = typeElement.getQualifiedName();
                println(packageOf.getQualifiedName().toString() + "--" + qualifiedName);
                MyRouter annotation = typeElement.getAnnotation(MyRouter.class);
                classNames.add(new MyRouterBean(qualifiedName.toString(), annotation.value()));
            }
        }
        println("process---------------------测试--------------------------");
        for (TypeElement typeElement : set) {
            PackageElement packageOf = elementUtils.getPackageOf(typeElement);
            Name qualifiedName = typeElement.getQualifiedName();
            println(packageOf.getQualifiedName().toString() + "--" + qualifiedName);
        }
        
        try {
            writeToFile(module_name);
        } catch (IOException e) {
            e.printStackTrace();
        }
        return true;
    }
    
    //生成Java代码
    private void writeToFile(String module_name) throws IOException {
        if (classNames.isEmpty()) {
            return;
        }
        //ClassName会自动倒入包
        ClassName string = ClassName.get("java.lang",
                "String");
        ClassName hashMap = ClassName.get("java.util",
                "HashMap");
        ClassName activity = ClassName.get("",
                "? extends android.app.Activity");
        ClassName clazz = ClassName.get("java.lang",
                "Class");

        //泛形参数
        ParameterizedTypeName typeName = ParameterizedTypeName.get(clazz, activity);
        ParameterizedTypeName type = ParameterizedTypeName.get(hashMap, string, typeName);
        MethodSpec.Builder addRouter = MethodSpec.methodBuilder("addRouter")
                .addModifiers(Modifier.PUBLIC)
                .addAnnotation(Override.class)
                .addJavadoc("addRouter\r\n@routerMap 路由map")
                .returns(TypeName.VOID)
                .addParameter(
                        type
                        ,"routerMap");
        ClassName routerTools = ClassName.get("com.example.basemoudle",
                "RouterTools");
        for (MyRouterBean className : classNames) {
//            addRouter.addStatement("$T.getInstance().addRouter(\"$L\",$L.class)",routerTools, className.getRouter(), className.getClassName());
            addRouter.addStatement("routerMap.put(\"$L\",$L.class)", className.getRouter(), className.getClassName());
        }

        TypeSpec typeSpec = TypeSpec.classBuilder("MyRouter$$" + module_name.toUpperCase())
                .addSuperinterface( ClassName.get("com.example.basemoudle","IRouter"))
                .addModifiers(Modifier.PUBLIC)
                .addMethod(addRouter.build())
//                .addAnnotation(AnnotationSpec
//                        .builder(MyRouter.class)
//                        .addMember("value","value=$L","2")
//                .build())
                .build();

        JavaFile javaFile = JavaFile.builder("com.example.router." + module_name.toLowerCase(), typeSpec)
                .build();
        javaFile.writeTo(filer);
    }


    private void println(String msg) {
        printMessage.printMessage(Diagnostic.Kind.NOTE,
                "MyProcessor:" + msg + "\r\n");
    }

}

生成的代码如下:

//注解的类
@MyRouter("MainActivity")
class MainActivity : AppCompatActivity() {}


//app模块自动生成的代码
package com.example.router.app;
···
···
public class MyRouter$$APP implements IRouter {
  /**
   * addRouter
   * @routerMap 路由map
   */
  @Override
  public void addRouter(HashMap<String, Class<? extends android.app.Activity>> routerMap) {
    routerMap.put("MainActivity",com.zmp.javaproject.MainActivity.class);
  }
}

app 初始化即可

//初始化
RouterTools.getInstance().init(this)
//使用路由
RouterTools.getInstance().navigate(this,"appb/BMainActivity")

plugin优化

此时的路由框架已经可以使用,但是apk运行期间初始化需要遍历dex,比较耗时

自定义插件

package com.example.plugin;

import com.android.build.gradle.AppExtension;
import com.android.build.gradle.AppPlugin;
import com.example.plugin.tools.Logger;
import com.example.plugin.transform.MyRouterTransform;

import org.gradle.api.Plugin;
import org.gradle.api.Project;
import org.jetbrains.annotations.NotNull;

/**
 * Created at 9:25 2020/9/23
 *
 * @author zmp
 * <p>
 * des:
 */
public class MyPlugin implements Plugin<Project> {

    @Override
    public void apply(@NotNull Project project) {
        Logger.make(project);
        Logger.w("apply:init-----------------");
        boolean hasAppPlugin = project.getPlugins().hasPlugin(AppPlugin.class);
        Logger.w("apply:init-----------------" + hasAppPlugin);
        if (hasAppPlugin) {

            Logger.w("apply:init-----------------" + hasAppPlugin);
            AppExtension android = project.getExtensions().getByType(AppExtension.class);

            //register this plugin 注册MyRouterTransform
            android.registerTransform(new MyRouterTransform(project));
        }
    }

}

自定义MyRouterTransform 完成插桩

/**
 * Created at 9:41 2020/9/23
 *
 * @author zmp
 * <p>
 * des:
 */
public class MyRouterTransform extends MyBaseTransform {

    public MyRouterTransform(Project project) {
        super(project);
    }


    protected void transformJar(File input, File dest) {
        if (ScanUtil.shouldProcessPreDexJar(input.getAbsolutePath())) {
            ScanUtil.scanJar(input, dest);
        } else {
            super.transformJar(input, dest);
        }
    }


    protected void transformSingleFile(String parentPath, File input, File dest) {
        if (!parentPath.endsWith(File.separator)) {
            parentPath += File.separator;
        }
        String replace = input.getAbsolutePath().replace(parentPath, "");
        if (!File.separator.equals("/")) {
            replace = replace.replaceAll("\\\\", "/");
        }
        try {
            if (ScanUtil.shouldProcessClass(replace)) {
                ScanUtil.scanClass(input, dest);
            }
            FileUtils.copyFile(input, dest);
        } catch (Exception e) {
            e.printStackTrace();
        }

    }


    protected void insertInitCodeTo() {
        File containsInitClass = ScanUtil.fileContainsInitClass;
        Logger.e("containsInitClass:" + containsInitClass);
        Logger.e("containsInitClass:" + Arrays.toString(ScanUtil.CLASS_NAMES.toArray()));
        if (containsInitClass != null) {
            insertInitCodeIntoJarFile(containsInitClass);
        }
    }

    /**
     * generate code into jar file
     *
     * @param jarFile the jar file which contains LogisticsCenter.class
     * @return
     */
    private void insertInitCodeIntoJarFile(File jarFile) {
        try {
            File optJar = new File(jarFile.getParent(), jarFile.getName() + ".opt");
            if (optJar.exists()) {
                optJar.delete();
            }
            JarFile file = new JarFile(jarFile);
            Enumeration<JarEntry> enumeration = file.entries();
            JarOutputStream jarOutputStream = new JarOutputStream(new FileOutputStream(optJar));

            while (enumeration.hasMoreElements()) {
                JarEntry jarEntry = (JarEntry) enumeration.nextElement();
                String entryName = jarEntry.getName();
                ZipEntry zipEntry = new ZipEntry(entryName);
                InputStream inputStream = file.getInputStream(jarEntry);
                jarOutputStream.putNextEntry(zipEntry);
                if (ScanUtil.ASM_CLASS_NAME.equals(entryName)) {
                    byte[] bytes = referHackWhenInit(inputStream);
                    jarOutputStream.write(bytes);
                } else {
                    jarOutputStream.write(IOUtils.toByteArray(inputStream));
                }
                inputStream.close();
                jarOutputStream.closeEntry();
            }
            jarOutputStream.close();
            file.close();

            if (jarFile.exists()) {
                jarFile.delete();
            }
            optJar.renameTo(jarFile);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }


    //refer hack class when object init
    private byte[] referHackWhenInit(InputStream inputStream) {
        Logger.e("referHackWhenInit");
        byte[] bytes = null;
        try {
            ClassReader cr = new ClassReader(inputStream);
            ClassWriter cw = new ClassWriter(cr, 0);
            ClassVisitor cv = new MyClassVisitor(Opcodes.ASM5, cw);
            cr.accept(cv, ClassReader.EXPAND_FRAMES);
            bytes = cw.toByteArray();
        } catch (IOException e) {
            e.printStackTrace();
        }
        return bytes;
    }

    class MyClassVisitor extends ClassVisitor {

        MyClassVisitor(int api, ClassVisitor cv) {
            super(api, cv);
        }

        @Override
        public void visit(int version, int access, String name, String signature,
                          String superName, String[] interfaces) {
            super.visit(version, access, name, signature, superName, interfaces);
        }

        @Override
        public MethodVisitor visitMethod(int access, String name, String desc,
                                         String signature, String[] exceptions) {
            MethodVisitor mv = super.visitMethod(access, name, desc, signature, exceptions);
            //generate code into this method
            Logger.e("visitMethod:" + name);
            if (name.equals(ScanUtil.GENERATE_TO_METHOD_NAME)) {
                mv = new RouteMethodVisitor(api, mv, access, name, desc);
            }
            return mv;
        }
    }

    static class RouteMethodVisitor extends AdviceAdapter {

        protected RouteMethodVisitor(int api, MethodVisitor methodVisitor, int access, String name, String descriptor) {
            super(api, methodVisitor, access, name, descriptor);
        }

        @Override
        protected void onMethodEnter() {
            super.onMethodEnter();
        }

        @Override
        protected void onMethodExit(int opcode) {
            super.onMethodExit(opcode);
            //插入字节码
            for (String className : ScanUtil.CLASS_NAMES) {
                Logger.w("className:" + className);
                push(className.replaceAll("/", "."));
                invokeStatic(Type.getType(ScanUtil.ASM_CLASS_NAME_TYPE), new Method("register", "(Ljava/lang/String;)V"));
            }
        }

        @Override
        public void visitMaxs(int maxStack, int maxLocals) {
            super.visitMaxs(maxStack + Integer.MAX_VALUE, maxLocals);
        }
    }

}

发布插件到 mavenLocal

apply plugin: 'java-library'
apply plugin: 'kotlin'

apply plugin: 'maven'
dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])
    implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"
    implementation gradleApi()
    implementation 'com.android.tools.build:gradle:4.0.1'
}

apply plugin: 'maven-publish'

publishing {
    publications {
        mavenJava(MavenPublication) {
            groupId 'com.example.android'
            artifactId  "zhanpple"
            version 1.0
            from components.java
            // more goes in here
        }
    }

    repositories {
        mavenLocal()
    }
}



sourceCompatibility = "1.7"
targetCompatibility = "1.7"

image

使用插件 项目build.gradle中

repositories {
        google()
        jcenter()
        mavenLocal()
    }
    dependencies {
        ...
        ...
        classpath "com.example.android:zhanpple:1.0"
    }

使用插件 app中

apply plugin: 'my.router.plugin'

apply plugin: 'my.router.plugin' 要与插件的gradle-plugins对应

image

编译app 对比字节码

image

参考代码 https://github.com/alibaba/ARouter

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351