@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 getSupportedAnnotationTypes() {
Set 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字段修饰的常量在编译器就已经被赋值。
解析
解析阶段主要是虚拟机将常量池内的符号引用替换为直接引用的过程。
初始化
初始化阶段是执行类构造器()方法的过程。
()与类的构造方法不同,()方法是由编译器自动收集类中的所有类变量的赋值动作和静态语句块中的语句合并而成的。编译器收集的顺序是按语句在源文件中出现的顺序决定的,静态语句块中只能访问定义在它之前的静态变量,定义在它之后的静态变量,只可以赋值,不可以访问。
虚拟机会保证子类的()方法执行之前,其父类的()方法一定被执行(父类先与子类完成加载过程)
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
【Android 详细知识点思维脑图(技能树)】
我个人是做Android开发,已经有十来年了,目前在某创业公司任职CTO兼系统架构师。虽然 Android 没有前几年火热了,已经过去了会四大组件就能找到高薪职位的时代了。这只能说明 Android 中级以下的岗位饱和了,现在高级工程师还是比较缺少的,很多高级职位给的薪资真的特别高(钱多也不一定能找到合适的),所以努力让自己成为高级工程师才是最重要的。
这里附上上述的面试题相关的几十套字节跳动,京东,小米,腾讯、头条、阿里、美团等公司19年的面试题。把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节。
由于篇幅有限,这里以图片的形式给大家展示一小部分。
网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。
最后,赠与大家一句话,共勉!
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门,即可获取!
ng)
DexClassLoader和PathClassLoader
【Android 详细知识点思维脑图(技能树)】
[外链图片转存中…(img-XSOo985f-1715355039868)]
我个人是做Android开发,已经有十来年了,目前在某创业公司任职CTO兼系统架构师。虽然 Android 没有前几年火热了,已经过去了会四大组件就能找到高薪职位的时代了。这只能说明 Android 中级以下的岗位饱和了,现在高级工程师还是比较缺少的,很多高级职位给的薪资真的特别高(钱多也不一定能找到合适的),所以努力让自己成为高级工程师才是最重要的。
这里附上上述的面试题相关的几十套字节跳动,京东,小米,腾讯、头条、阿里、美团等公司19年的面试题。把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节。
由于篇幅有限,这里以图片的形式给大家展示一小部分。
[外链图片转存中…(img-QP8ivmDd-1715355039868)]
网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。