使用注解实现Android线程切换
2019-04-16 本文已影响174人
Tyhj
之前有写过两篇关于运行时注解和编译时注解的文章,主要是实现了控件的初始化@ViewById
和点击事件@Click
的功能;之前使用androidannotations,常用的注解还包括了@Background
和@UiThread
操作来切换线程;
但是通过这种在编译时生成代码的方式好像实现不了这两个操作,@Click
和@ViewById
注解虽然一个是用于方法,一个是用于变量,但是其实都是拿到这个控件变量,然后进行初始化和设置点击事件;而线程切换是对一个方法来操作,类似代理,需要在方法里面添加代码;
AspectJ
AspectJ是一个面向切面的框架,它扩展了Java语言。AspectJ定义了AOP语法,它有一个专门的编译器用来生成遵守Java字节编码规范的Class文件。所以想在方法里面添加代码可以使用Aspectj
来实现,它的原理大概就是去修改Class文件;
具体实现
具体实现还是非常简单的,在Android中有一个可以直接使用的AspectJ框架
集成AspectJX
1.在项目根目录的build.gradle里依赖AspectJX
classpath 'com.hujiang.aspectjx:gradle-android-plugin-aspectjx:2.0.4'
2.在使用的module(app)里面添加依赖
api 'org.aspectj:aspectjrt:1.8.9'
3.在app项目的build.gradle里应用插件
apply plugin: 'android-aspectjx'
更多详细的配置可以看GitHub上的文档
定义注解
@Background
用于切换到子线程
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.CLASS)
public @interface Background {
}
@UiThread
用于切换到UI线程
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.CLASS)
public @interface UiThread {
}
实现AOP
新建一个类,添加@Aspect
注解,使用了@Around注解申明两个方法,注解里面的值是注解的路径,这里使用RxJava来进行线程的切换
@Aspect
public class ThreadsAspect {
@Around("execution(@com.dhht.annotation.Background void *(..))")
public void doBackground(final ProceedingJoinPoint joinPoint) {
Observable.create(new ObservableOnSubscribe<Object>() {
@Override
public void subscribe(ObservableEmitter<Object> emitter) throws Exception {
try {
joinPoint.proceed();
} catch (Throwable throwable) {
throwable.printStackTrace();
}
}
}).subscribeOn(Schedulers.io()).subscribe();
}
@Around("execution(@com.dhht.annotation.UiThread void *(..))")
public void doUiThread(final ProceedingJoinPoint joinPoint) {
Observable.create(new ObservableOnSubscribe<Object>() {
@Override
public void subscribe(ObservableEmitter<Object> emitter) throws Exception {
try {
joinPoint.proceed();
} catch (Throwable throwable) {
throwable.printStackTrace();
}
}
}).subscribeOn(AndroidSchedulers.mainThread()).subscribe();
}
}
然后具体的使用
@UiThread
void toast(String msg) {
Toast.makeText(this, msg, Toast.LENGTH_SHORT).show();
}
@Background
void backgroud() {
Log.e("backgroud", Thread.currentThread().getName());
toast("主线程");
}
整体过程非常简单,比实现注解处理器生成代码简单多了,至于添加线程延迟什么的,改改也很简单;AspectJ是一个面向切面的框架,它可以做的事情还有很多,这只是小小的栗子