阿里P7移动互联网架构师进阶视频(每日更新中)免费学习请点击:https://space.bilibili.com/474380680

本篇文章将继续从以下两个内容来介绍组件化框架设计:

  • apt编译时期自动生成代码
  • Android动态加载技术基础之类加载(ClassLoader)

一、apt编译时期自动生成代码

第一步
新建一个android项目。
第二步
新建立一个java的Module。注意是javalib。这个lib用来专门写注解就好。

这个lib里面就先放一个注解,叫TestAnno。

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target; @Retention(RetentionPolicy.CLASS)
@Target(ElementType.TYPE)
public @interface TestAnno {
}

RetentionPolicy.CLASS表示编译时候注解。你需要关系的就是@Target(ElementType.TYPE)这个type是类的注解,可以有方法的,属性的等等。

然后这个javalib的gradle文件要这么写。

apply plugin: 'java'

dependencies {
compile fileTree(dir: 'libs', include: ['*.jar'])
}
sourceCompatibility = "1.7"
targetCompatibility = "1.7"

注解库弄好了,在弄新建一个java lib 叫inject_comiler。这个是就是核心代码了,在编译时候,执行这个个库里面的代码,然后 生成代码。这个工程 三个类。一个是注解注解处理器的核心。一个是定义生成java文件的,方法拼接。还有一个就是常量包名类名了。
先看这个的gradle文件。

apply plugin: 'java'
sourceCompatibility = "1.7"
targetCompatibility = "1.7"
dependencies {
compile fileTree(include: ['*.jar'], dir: 'libs')
compile 'com.google.auto.service:auto-service:1.0-rc2'//谷歌的帮助我们快速实现注解处理器
compile project(':inject_annotation')//自己定义的注解的java lib
compile 'com.squareup:javapoet:1.7.0'//用来生成java文件的,避免字符串拼接的尴尬
}

//这个注解是谷歌提供了,快速实现注解处理器,会帮你生成配置文件啥的 。直接用就好
@AutoService(Processor.class)
public class ActivityInjectProcesser extends AbstractProcessor {
private Filer mFiler; //文件相关的辅助类
private Elements mElementUtils; //元素相关的辅助类 许多元素
private Messager mMessager; //日志相关的辅助类 private Map<String, AnnotatedClass> mAnnotatedClassMap; @Override
public synchronized void init(ProcessingEnvironment processingEnv) {
super.init(processingEnv);
mFiler = processingEnv.getFiler();
mElementUtils = processingEnv.getElementUtils();
mMessager = processingEnv.getMessager();
mAnnotatedClassMap = new TreeMap<>();
} //这个方法是核心方法,在这里处理的你的业务。检测类别参数,生成java文件等
@Override
public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
mAnnotatedClassMap.clear(); try {
processActivityCheck(roundEnv);
} catch (Exception e) {
e.printStackTrace();
error(e.getMessage());
} for (AnnotatedClass annotatedClass : mAnnotatedClassMap.values()) {
try {
annotatedClass.generateActivityFile().writeTo(mFiler);
} catch (Exception e) {
error("Generate file failed, reason: %s", e.getMessage());
}
}
return true;
} private void processActivityCheck(RoundEnvironment roundEnv) throws IllegalArgumentException, ClassNotFoundException {
//check ruleslass forName(String className
for (Element element : roundEnv.getElementsAnnotatedWith((Class<? extends Annotation>) Class.forName(TypeUtil.ANNOTATION_PATH))) {
if (element.getKind() == ElementKind.CLASS) {
getAnnotatedClass(element);
} else
error("ActivityInject only can use in ElementKind.CLASS");
}
} private AnnotatedClass getAnnotatedClass(Element element) {
// tipe . can not use chines so ....
// get TypeElement element is class's --->class TypeElement typeElement = (TypeElement) element
// get TypeElement element is method's ---> TypeElement typeElement = (TypeElement) element.getEnclosingElement();
TypeElement typeElement = (TypeElement) element;
String fullName = typeElement.getQualifiedName().toString();
AnnotatedClass annotatedClass = mAnnotatedClassMap.get(fullName);
if (annotatedClass == null) {
annotatedClass = new AnnotatedClass(typeElement, mElementUtils, mMessager);
mAnnotatedClassMap.put(fullName, annotatedClass);
}
return annotatedClass;
} @Override
public SourceVersion getSupportedSourceVersion() {
return SourceVersion.latestSupported();
} //这个个方法返回你要处理注解的类型
@Override
public Set<String> getSupportedAnnotationTypes() {
Set<String> types = new LinkedHashSet<>();
types.add(TypeUtil.ANNOTATION_PATH);
return types;
} private void error(String msg, Object... args) {
mMessager.printMessage(Diagnostic.Kind.ERROR, String.format(msg, args));
} private void log(String msg, Object... args) {
mMessager.printMessage(Diagnostic.Kind.NOTE, String.format(msg, args));
}
}

然后是生成java文件的辅助类。

public class AnnotatedClass {

    private TypeElement mTypeElement;//activity  //fragmemt
private Elements mElements;
private Messager mMessager;//日志打印 public AnnotatedClass(TypeElement typeElement, Elements elements, Messager messager) {
mTypeElement = typeElement;
mElements = elements;
this.mMessager = messager;
} public JavaFile generateActivityFile() {
// build inject method
MethodSpec.Builder injectMethod = MethodSpec.methodBuilder(TypeUtil.METHOD_NAME)
.addModifiers(Modifier.PUBLIC)
.addParameter(TypeName.get(mTypeElement.asType()), "activity", Modifier.FINAL);
injectMethod.addStatement("android.widget.Toast.makeText" +"(activity, $S,android.widget.Toast.LENGTH_SHORT).show();", "from build");
//generaClass
TypeSpec injectClass = TypeSpec.classBuilder(mTypeElement.getSimpleName() + "$$InjectActivity")
.addModifiers(Modifier.PUBLIC)
.addMethod(injectMethod.build())
.build();
String packgeName = mElements.getPackageOf(mTypeElement).getQualifiedName().toString();
return JavaFile.builder(packgeName, injectClass).build();
} }

这里就生成了一个 类名+$$的类,有一个方法叫inject参数是这个类本身,弹出一个toast。最后一个就是一个字符常量类。

public class TypeUtil {
public static final String METHOD_NAME = "inject";
public static final String ANNOTATION_PATH = "com.example.TestAnno";
}

好了lib工程完毕。然后是主工程引用。
首先是在工程的gradle里面配置下apt。

dependencies {
classpath 'com.android.tools.build:gradle:2.3.2'
classpath 'com.neenbedankt.gradle.plugins:android-apt:1.8'
}

然后在app的gradle里面配置如下

apply plugin: 'com.neenbedankt.android-apt'
……
compile project(':inject_annotation')
apt project(':inject_comiler')

这样就行了。注意是apt 。为什么要新建立javalib。因为javalib不能在引用adnroidlib,。而注解处理器是javalib来完成,app里面,可以这引用这2个,apt如果换成complie 会提示错误,但不会影响啥。配置都好了,就是最后的使用了。

 
 

总结构
在我们MainActivity 上面加上注解,使用下。

@TestAnno
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
InjectActivity.inject(this);//调用build生成的类
}
}

最后一个类就是我们的使用这个在运行时,调用生成build的类

public class InjectActivity {
private static final ArrayMap<String, Object> injectMap = new ArrayMap<>(); public static void inject(AppCompatActivity activity) {
String className = activity.getClass().getName();
try { Object inject = injectMap.get(className); if (inject == null) {
//加载build生成的类
Class<?> aClass = Class.forName(className + "$$InjectActivity");
inject = aClass.newInstance();
injectMap.put(className, inject);
}
//反射执行方法
Method m1 = inject.getClass().getDeclaredMethod("inject", activity.getClass());
m1.invoke(inject, activity);
} catch (Exception e) {
e.printStackTrace();
}
}
}

这个类用了反射的方式查找了指定类执行了指定方法。有一个map来缓存,不用每次都重新反射。
当然你可可以不用反射,在创建build生成类的时候,实现一个接口,这里直接强转为接口,直接调用接口的方法也可以。这里简单一些用的反射
通过一个注解,就自动生成弹出toast的代码看下自动 生成的代码

public class MainActivity$$InjectActivity {
public void inject(final MainActivity activity) {
android.widget.Toast.makeText(activity, "from build",android.widget.Toast.LENGTH_SHORT).show();;
}
}

最后看下运行效果。

 
 

二、Android动态加载技术基础之类加载(ClassLoader)

虚拟机类加载机制

  • 类加载过程是指虚拟机将描述类的数据从Class文件中加载到内存,并对数据进行校验,转化解析和初始化,最终形成可以被虚拟机直接使用的Java类型的过程。
  • 在Java中,类的加载和连接过程都是在程序运行期间完成。虽然会增加运行时的性能开销,但可以提高程序灵活性,这也是Java能够实现动态加载的原因之一。

类加载的过程

虚拟机类加载过程分为加载,验证,准备,解析,初始化,使用,卸载七个阶段。其中验证,准备,解析三个部分成为连接阶段。

加载

一般来说,在遇到了以下四种情况的时候,虚拟机会开始进行类的加载:

  • 使用new关键字实例化对象,读取或者设置一个类的静态变量(被final修饰的除外,已经在编译器被加入常量池),以及调用一个类的静态方法的时候
  • 对类进行反射调用的时候
  • 当初始化一个类时,如果其父类没有被加载,则先对其父类进行加载
  • 当虚拟机启动的时候,用户指定的(包含main方法)的类会被加载
    在类的加载阶段,虚拟机会完成一下三件事情:
  • 通过一个类的全限定名获取定义类的二进制字节流
  • 将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构
  • 在Java堆中生成一个代表这个类的Class对象,作为方法区这些数据的访问入口。

验证

这一阶段是为了确保class文件的字节流包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全。主要包括以下几个过程:

  • 文件格式验证
  • 元数据验证
  • 字节码验证
  • 符号引用验证

准备

准备阶段是正式为类变量分配内存并设置类变量初始值的阶段,这些内存都将在方法区中进行分配。这个阶段中进行内存分配的变量只有被static修饰的变量,并将其设置为默认值,而真正的赋值则在初始化阶段。另外,被final static字段修饰的常量在编译器就已经被赋值。

解析

解析阶段主要是虚拟机将常量池内的符号引用替换为直接引用的过程。

初始化

初始化阶段是执行类构造器<cinit>()方法的过程。
<cinit>()与类的构造方法不同,<cinit>()方法是由编译器自动收集类中的所有类变量的赋值动作和静态语句块中的语句合并而成的。编译器收集的顺序是按语句在源文件中出现的顺序决定的,静态语句块中只能访问定义在它之前的静态变量,定义在它之后的静态变量,只可以赋值,不可以访问。
虚拟机会保证子类的<cinit>()方法执行之前,其父类的<cinit>()方法一定被执行(父类先与子类完成加载过程)

Java中的ClassLoader

类加载阶段中,实现“通过一个类的全限定名获取定义类的二进制字节流”的动作被放在了虚拟机外部去实现,以便应用程序决定如何去加载所需要的类,实现这个动作的代码模块被称为“类加载器”
从Java虚拟机的角度上讲,只存在两种不同的类加载器:

  • Bootstrap ClassLoader:使用C++实现,是虚拟机的一部分。它主要负责加载存放在%JAVAHOME%/lib目录中的,或者被-Xbootclasspath指定的类库到虚拟机内存中,Bootstrap ClassLoader无法被java程序直接引用。
  • 继承自java.lang.ClassLoader的类加载器:
  • Extension ClassLoader:主要负责加载%JAVAHOME%/lib/ext目录中的,或者被java.ext.dirs系统变量指定路径的所有类。
  • Application ClassLoader:也被称为系统类加载器(因为其实getSystemClassLoader的返回对象),主要负责加载用户类路径(ClassPath)下的类库

类的双亲委派模型

这些类加载器之间的关系如下:

 
 

双亲委派模型中,除了顶层的BootstrapClassLoader,其他类加载器都要求有自己的父类加载器,这里的类加载器一般以组合的方式实现。

  • 双亲委派模型的工作过程是:当一个类加载器收到一个类加载请求的时候,他首先不会自己加载这个类,而是将这个请求委派给父类加载器去完成,只有当父类加载器无法完成这个加载请求时,子加载器才会尝试自己去加载。
  • 双亲委派模型的作用:
  • 使得java类随着它的类加载器一起具备了一种带有优先级的层次关系
  • 保证Java环境的稳定性
  • 避免重复加载,如果已经加载过一次Class,就不需要再次加载,而是先从缓存中直接读取。

Android中的ClassLoader

  • 由于Android虚拟机的执行文件是Dex文件,而不是JVM中的Class文件,所以Java中的类加载器是无法加载Dex文件的,因此,Android中存在另外一套ClassLoader。

Android的ClassLoader类型

Android中的ClassLoader根据用途可分为一下几种:

  • BootClassLoader:主要用于加载系统的类,包括java和android系统的类库,和JVM中不同,BootClassLoader是ClassLoader内部类,是由Java实现的,它也是所有系统ClassLoader的父ClassLoader
  • PathClassLoader:用于加载Android系统类和开发编写应用的类,只能加载已经安装应用的 dex 或 apk 文件,也是getSystemClassLoader的返回对象
  • DexClassLoader:可以用于加载任意路径的zip,jar或者apk文件,也是进行安卓动态加载的基础

Android中ClassLoader的继承关系

 
 

ClassLoader的执行过程

  • 从上面的ClasLoader结构图可以看到,ClassLoader的主要逻辑主要集中在ClassLoader和BaseDexClassLoder这两个类中。

ClasLoader

  • ClasLoader是所有ClassLoader的父类,它定义了加载Class的一般行为。
  • 与Java中不同,Android中加载类的过程主要是由loadClass方法实现,而在Java中则是findClass方法。
protected Class<?> loadClass(String className, boolean resolve) throws ClassNotFoundException {
Class<?> clazz = findLoadedClass(className);
if (clazz == null) {
ClassNotFoundException suppressed = null;
try {
clazz = parent.loadClass(className, false);
} catch (ClassNotFoundException e) {
suppressed = e;
}
if (clazz == null) {
try {
clazz = findClass(className);
} catch (ClassNotFoundException e) {
e.addSuppressed(suppressed);
throw e;
}
}
}
return clazz;
}

  • 可以看到,当收到一个加载类请求的时候,ClassLoader会先调用findLoadedClass查询是否本类是否被本加载器加载过(调用JNI方法),如果没有被加载则把请求委托给父加载器,当父加载器无法完成加载行为的时候,才会调用findClass方法尝试自己加载,而ClassLoader中的findClass方法并没有实现,而是交给子类去是实现。
 
 

BaseDexClassLoader

  • 作为ClassLoader的直接子类,BaseDexClassLoader实现加载findClass方法的主要逻辑,而其子类DexClassLoader和PathClassLoader都只是加载路径以及某些行为不同而已
 
 
  • 可以看到findClass方法是又是调用了pathList的findClass方法去加载类,而pathList则是一个DexPathList对象,它的findClass对象是这样实现的:
 
 
  • 其中DexFile是Dex文件在Java中的表现形式,而它的loadClassBinaryName方法则是最后调用了JNI方法去完成在dex文件在加载Class对象。
 
 

DexClassLoader和PathClassLoader

  • DexClassLoader和PathClassLoader都是BaseDexClassLoader的子类,他们的实现也很简单,只是构造方法传入了不同的参数而已:

  • DexClassLoader :

     
     
  • PathClassLoader:

 
 
  • 可以看到,DexClassLoader和PathClassLoader的区别就是,PathClassLoader的第二个参数传为NULL,回到BaseDexClassLoader中可以看到:

     
     
  • 根据注释可以看到optimizedDirectory参数是用来放置DexFile的,那么具体是怎么回事呢,再进入DexPathList

 
 
  • optimizedDirectory被传进了makeDexElements方法
 
 
  • 又被传进了loadDexFile

     
    image
 
 
  • 可以看到,如果optimizedDirectory为NULL,则会以原来的路径创建DexFile,否则会以optimizedDirectory为路径创建DexFile

  • 其实optimizedDirectory是要求一个内部路径的,因为动态加载去加载的可执行文件一定要存放在内部存储。而DexClassLoader可以指定optimizedDirectory,所以它可以加载外部的dex,并且这个dex会被复制到内部路径的optimizedDirectory;而PathClassLoader没有optimizedDirectory,所以它只能加载内部路径的dex,也就是存在于已经安装过的apk里面的。

参考https://www.jianshu.com/p/144df8826d15
https://blog.csdn.net/spinchao/article/details/72972185

阿里P7移动互联网架构师进阶视频(每日更新中)免费学习请点击:https://space.bilibili.com/474380680

组件化框架设计之apt编译时期自动生成代码&动态类加载(二)的更多相关文章

  1. 组件化框架设计之AOP&IOC(四)

    阿里P7移动互联网架构师进阶视频(每日更新中)免费学习请点击:https://space.bilibili.com/474380680 本篇文章将从以下两个方面来介绍组件化框架设计: [AOP(面向切 ...

  2. 组件化框架设计之Java SPI机制(三)

    阿里P7移动互联网架构师进阶视频(每日更新中)免费学习请点击:https://space.bilibili.com/474380680 本篇文章将从深入理解java SPI机制来介绍组件化框架设计: ...

  3. java如何在eclipse编译时自动生成代码

    用eclipse写java代码,自动编译时,如何能够触发一个动作,这个动作是生成本项目的代码,并且编译完成后,自动生成的代码也编译好了, java编辑器中就可以做到对新生成的代码的自动提示? 不生成代 ...

  4. (转)MyBatis框架的学习(七)——MyBatis逆向工程自动生成代码

    http://blog.csdn.net/yerenyuan_pku/article/details/71909325 什么是逆向工程 MyBatis的一个主要的特点就是需要程序员自己编写sql,那么 ...

  5. Android组件化框架设计与实践

    在目前移动互联网时代,每个 APP 就是流量入口,与过去 PC Web 浏览器时代不同的是,APP 的体验与迭代速度影响着用户的粘性,这同时也对从事移动开发人员提出更高要求,进而移动端框架也层出不穷. ...

  6. 组件化框架设计之阿里巴巴开源路由框架——ARouter原理分析(一)

    阿里P7移动互联网架构师进阶视频(每日更新中)免费学习请点击:https://space.bilibili.com/474380680 背景 当项目的业务越来越复杂,业务线越来越多的时候,就需要按照业 ...

  7. 从零开始搭建Android组件化框架

    问题 在已经开发过几个项目的童鞋,如果这时需要重新开发一个新项目,是否需要自己重新搭建框架呢,还是从老项目中拷贝粘贴? 我们是否可以封装一个底层的lib库,这个底层的公共基础库 包括了一些第三方库(如 ...

  8. [Android Pro] 终极组件化框架项目方案详解

    cp from : https://blog.csdn.net/pochenpiji159/article/details/78660844 前言 本文所讲的组件化案例是基于自己开源的组件化框架项目g ...

  9. Android组件化框架项目详解

    简介 什么是组件化? 项目发展到一定阶段时,随着需求的增加以及频繁地变更,项目会越来越大,代码变得越来越臃肿,耦合会越来越多,开发效率也会降低,这个时候我们就需要对旧项目进行重构即模块的拆分,官方的说 ...

随机推荐

  1. project euler-34

    145是个奇怪的数字.由于1!+ 4! + 5! = 1 + 24 + 120 = 145. 请求出能表示成其每位数的阶乘的和的全部数的和. 请注意:由于1! = 1和2! = 2不是和,故它们并不包 ...

  2. saltstack的高级管理

    一.saltstack的状态管理 状态管理官网: https://www.unixhot.com/docs/saltstack/ref/states/all/index.html 1)状态分析 [ro ...

  3. go语言从例子开始之Example34.速率限制

    速率限制(英) 是一个重要的控制服务资源利用和质量的途径.Go 通过 Go 协程.通道和打点器优美的支持了速率限制. Example: package main import "fmt&qu ...

  4. 从__name__=='__main__'说到__builtin__

    一.__name__ 我们在写好代码进行自测的时候一般会先写这样一行代码: # inter_method if __name__ == '__main__': 为什么呢,可能并不是所有人都考虑过,这个 ...

  5. oralce 日期 date 相关操作

    1.当前时间加减一年 加一年 select sysdate,add_month(sysdate,12) from dual; 减一年 select sysdate,add_month(sysdate, ...

  6. java ArrayList的几种方法使用

    package java06; import java.util.ArrayList; /* ArrayList的常用的几个方法: public boolean add(E e) : 向集合汇总添加元 ...

  7. web安全—tomcat禁用WebDAV或者禁止不需要的 HTTP 方法

    现在主流的WEB服务器一般都支持WebDAV,使用WebDAV的方便性,呵呵,就不用多说了吧,用过VS.NET开发ASP.Net应用的朋友就应该 知道,新建/修改WEB项目,其实就是通过WebDAV+ ...

  8. ThinkPHP import 类库导入 include PHP文件

    ThinkPHP 模拟了 Java 的类库导入机制,统一采用 import 方法进行类文件的加载.import 方法是 ThinkPHP 内建的类库和文件导入方法,提供了方便和灵活的文件导入机制,完全 ...

  9. Java二级上机训练

    NCRE上机训练一 import javax.swing.JOptionPane; /** * 并完成两个整数的输入,计算乘积,最后按确定键退出程序. */ public class Java_1 { ...

  10. 每天一个Linux命令:find(20)

    find find命令在目录结构中搜索文件,并执行指定的操作.Linux下find命令提供了相当多的查找条件,功能很强大.由于find具有强大的功能,所以它的选项也很多,其中大部分选项都值得我们花时间 ...