@Tyhj
        
        2019-04-16T03:11:37.000000Z
        字数 2062
        阅读 1852
    Android
之前有写过两篇关于运行时注解和编译时注解的文章,主要是实现了控件的初始化@ViewById和点击事件@Click的功能;之前使用androidannotations,常用的注解还包括了@Background和@UiThread操作来切换线程;
但是通过这种在编译时生成代码的方式好像实现不了这两个操作,@Click和@ViewById注解虽然一个是用于方法,一个是用于变量,但是其实都是拿到这个控件变量,然后进行初始化和设置点击事件;而线程切换是对一个方法来操作,类似代理,需要在方法里面添加代码;
AspectJ是一个面向切面的框架,它扩展了Java语言。AspectJ定义了AOP语法,它有一个专门的编译器用来生成遵守Java字节编码规范的Class文件。所以想在方法里面添加代码可以使用Aspectj来实现,它的原理大概就是去修改Class文件;
具体实现还是非常简单的,在Android中有一个可以直接使用的AspectJ框架
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 {}
新建一个类,添加@Aspect注解,使用了@Around注解申明两个方法,注解里面的值是注解的路径,这里使用RxJava来进行线程的切换
@Aspectpublic class ThreadsAspect {@Around("execution(@com.dhht.annotation.Background void *(..))")public void doBackground(final ProceedingJoinPoint joinPoint) {Observable.create(new ObservableOnSubscribe<Object>() {@Overridepublic 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>() {@Overridepublic void subscribe(ObservableEmitter<Object> emitter) throws Exception {try {joinPoint.proceed();} catch (Throwable throwable) {throwable.printStackTrace();}}}).subscribeOn(AndroidSchedulers.mainThread()).subscribe();}}
然后具体的使用
@UiThreadvoid toast(String msg) {Toast.makeText(this, msg, Toast.LENGTH_SHORT).show();}@Backgroundvoid backgroud() {Log.e("backgroud", Thread.currentThread().getName());toast("主线程");}
整体过程非常简单,比实现注解处理器生成代码简单多了;AspectJ是一个面向切面的框架,它可以做的事情还有很多,这只是小小的栗子
