我所知道的偷懒的方式

前言

在现今的软件开发过程中,软件开发人员将更多的精力投入在了重复的相似劳动中。特别是在如今特别流行的 MVC 架构模式中,软件各个层次的功能更加独立,同时代码的相似度也更加高。所以我们需要寻找一种来减少软件开发人员重复劳动的方法,让程序员将更多的精力放在业务逻辑以及其他更加具有创造力的工作上。

服务端的技术发展了很多年,有很多值得可以借鉴的地方。有的时候,在跟后台沟通的时候,发现他们在设计好数据库表结构的时候,经常可以一键生成常见的功能(增、删、改、查)。后来知道,因为相识程度非常的高。所以他们经常以代码来生成代码。就这样完成了一站式的功能。

更多时候,我在考虑,为何他们不优先考虑封装呢?

  1. 可能会经常发生变动。
  2. 人员因素吧

Android 的一种"偷懒"

Android 中有种模板编程的偷懒方式。但不是今天的主角。今天的主角是Annotation Processor(注解处理器)。 在最近查阅源码的时候。发现很多框架都会使用到Annotation Processor比如说,ButterKnifeDaggerARouter等等。所以理解Annotation Processor(注解处理器)的原理,是一个Android程序员必须具备的技能。

1. Annotation Processor

Annotation Processor直译成中文就是(注解处理器),就是能够对注解进行处理。既然能够对注解进行处理,那么先定义一个简单的注解。

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface BindView {
    int value();
}

2. 使用注解

定义完注解后,我们会想着在什么时候使用它呢? 还记得上述的核心是为了偷懒,那么
Android存在有重复性非常高而且难度极地的代码,就是对控件的获取和处理。

public class MainActivity extends AppCompatActivity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        TextView textView1=(TextView) findViewById(R.id.tvHello);
        textView1.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
            }
        });
    }
}

这样的模板代码不仅很浪费时间,而且代码的美观程度也大大的降低了。那么我们现在尝试着用自己定义的注解去处理。

public class MainActivity extends AppCompatActivity {

    @BindView(R.id.tvHello)
    TextView tvHello;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }

    @OnClick(R.id.tvHello)
    public void changeText2() {
        Toast.makeText(this, "2222", Toast.LENGTH_LONG).show();
    }
}

上述的代码,就会比较清晰。不会有获取控件的过程的代码。声明即使用了。那么仅仅这样打上一个注解,能够自动的获取控件吗?答案是不可能的。我们还缺少一步编写注解处理器(其实就是编写遇到这个注解改怎么处理)。

3. 编写注解处理器
xx01 最终生成的类

在编写注解处理器之前,先尝试着去编写一个类。这个类的职责就是获取控件。

public final class MainActivity$ViewBinding {
  public MainActivity$ViewBinding(MainActivity target, View source) {
    if(target == null)  return;
    if(source == null)  return;
    target.tvHello = (android.widget.TextView)source.findViewById(2131165275);
  }

  public MainActivity$ViewBinding(MainActivity target) {
    this(target, target.getWindow().getDecorView());
  }
}

上面的这个类,就是需要注解处理器帮我们自动生成的模板代码。我们必须知道自己需要什么代码,才能控制注解处理器生成我们想要的代码。那么观察一下最终生成的代码,其中有几个需要注意的:

  1. 需要遍历所有需要处理的注解。在本文中我们只关心BindView
  2. 当我们遍历出所有带有BindView的注解的时候,我们需要为这个字段赋值。而赋值的操作,我们是通过属性直接赋值(默认的访问权限是包访问权限)

target.tvHello = (android.widget.TextView)source.findViewById(2131165275);

所以这个由注解处理器帮我生成的类,需要跟目标类在同级包下。

  1. 这个类的名字是按照一定格式生成的,在本文中它是由目标类+$ViewBinding。

到这里,需要注意的点都讲完了,接下开始来编写注解处理器的核心代码。

xx02 项目目录的划分

在知道了我们最终需要生成什么代码之后,就需要对整个目录进行规划一下,因为这个工具不仅仅是当前这个项目会被使用。很有很多项目将对其引用。你可能会将其发布到JCenter上。

image.png

根据上面的图,我们分别建立3module。其中2个是java libaryandroid libary

image.png

最终的目录结构是这样的

image.png

然后我们将刚才编写的BindView注解放入inject-annotation模块中

xx03 编写注解处理器

在划分话目录结构后,我们可以编写注解处理器的核心代码了,也就意味着,我们需要把它的代码放置在inject-compiler中。

需要在inject-compiler下的build.gradle导入几个库

    implementation 'com.google.auto.service:auto-service:1.0-rc4'
    annotationProcessor 'com.google.auto.service:auto-service:1.0-rc4'
    api 'com.squareup:javapoet:1.10.0'
    implementation project(':inject-annotation')

介绍一下这几个库,

  1. auto-service 是帮助生成META-INF/services/javax.annotation.processing文件中的内容
  2. javapoet 更加面向对象的输出代码
  3. inject-annotation 需要处理的注解。

在添加完这些东西之后,我们就需要创建一个注解处理器了

  @Override
    public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
        HashMap<TypeElement, List<Element>> datas = new HashMap<>();
        for (Element element : roundEnv.getElementsAnnotatedWith(BindView.class)) {
            TypeElement originalType = (TypeElement) element.getEnclosingElement();
            if (datas.get(originalType) == null) {
                List<Element> elements = new ArrayList<>();
                datas.put(originalType, elements);
            }
            List<Element> dd = datas.get(originalType);
            dd.add(element);
        }

        for (Map.Entry<TypeElement, List<Element>> entry : datas.entrySet()) {
            TypeElement key = entry.getKey();
            List<Element> value = entry.getValue();
            createFile(key, value);
        }
        return false;
    }

核心代码大概是上面的,就是遍历类中的注解元素,包装成一个Map数据的数据结构。

image.png

然后将构建后的数据结构,进行处理,按照之间MainActivity$ViewBinding的所预想的规则进行输出。

  private MethodSpec createCustomerConstructor2View(TypeElement originalType, List<Element> elements) {
        ParameterSpec targetParamSpec = ParameterSpec.builder(TypeName.get(originalType.asType()), "target").build();
        MethodSpec.Builder constructor1 = MethodSpec.constructorBuilder()
                .addModifiers(Modifier.PUBLIC)
                .addParameter(targetParamSpec)
                .addParameter(mViewParameterSpec)
                .addStatement("if(target == null)  return")
                .addStatement("if(source == null)  return");

        for (Element element : elements) {
            String variateName = element.getSimpleName().toString();
            String variateType = element.asType().toString();
            int resId = element.getAnnotation(BindView.class).value();
            constructor1.addStatement("target.$L = ($N)source.findViewById($L)", variateName, variateType, resId);
        }
        return constructor1.build();
    }

这样我们就完成了注解处理器的编写,当你编写后重新Rebuild Project后,如果不出意外的话,你可以找到下面这个文件。

image.png
xx03 调试Annotation Processor(注解处理器)

如果你在编写注解处理器可能不是你预想,那么断点调试就变得非常的重要了。那么接下来介绍如何调试注解处理器.

  1. 打开Edit Configurations新建一个Remote Configurations
image.png
  1. Terminal中运行下面命令

./gradlew --no-daemon -Dorg.gradle.debug=true :app:clean :app:compileDebugJavaWithJavac

image.png
  1. 需要断点的地方,下断点。

  2. Debug 刚才添加的Configurations

image.png

5.它就可以正常的断点了

image.png
xx04 调用生成的代码

当我们完成了注解处理器的编写,也正常的生成了MainActivity$ViewBinding文件后,我们就需要调用所生成的类,让它来帮我们完成控件的注入。因为其中会涉及到对Activity的引用,所以这个包是android libary。也就意味着,接下来的代码需要在inject-core模块中编写。

我们在进入一个Activity或者Fragment的时候,能够调用inject方法,而inject的实现如下:

public static void inject(Activity activity) {
        String name = activity.getClass().getSimpleName();
        String packName = activity.getPackageName();
        String fullName = name + "$ViewBinding";
        try {
            Class targetClass = activity.getClassLoader().loadClass(packName + "." + fullName);
            Constructor constructor = targetClass.getConstructor(activity.getClass());
            constructor.newInstance(activity);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

就是通过反射,然后实例化刚才的对象。这样的话,我们就完成了控件的注入了。

4. 发布到JCenter

因为我们项目中存在有jaraar格式的包,所以需要分别上传。在编写上传任务之前,需要先在最外层的build.gradle添加plugin,请注意插件的版本。不然会出现一些奇奇怪怪的问题。

buildscript {
    repositories {
        google()
        jcenter()
        
    }
    dependencies {
        classpath 'com.android.tools.build:gradle:3.5.0'
        classpath 'com.github.dcendents:android-maven-gradle-plugin:2.0'
        classpath 'com.jfrog.bintray.gradle:gradle-bintray-plugin:1.6'

    }
}

allprojects {
    repositories {
        google()
        jcenter()
        
    }
}

task clean(type: Delete) {
    delete rootProject.buildDir
}

  1. 编写jar格式的gradle,在最外层新建jar_publish.gradle文件,然后在:inject-compilerbuild.gradle中 apply。
image.png
  1. 编写aar格式的gradle,在最外层新建aar_publish.gradle文件,然后在:inject-corebuild.gradle中 apply。
image.png

然后依次上传即可。

本文中的代码

  1. https://github.com/BelongsH/APTDemo

相关链接

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

推荐阅读更多精彩内容

  • 前面写了Android 开发:由模块化到组件化(一),很多小伙伴来问怎么没有Demo啊?之所以没有立刻放demo的...
    涅槃1992阅读 8,022评论 4 37
  • 网上有很多 APT 相关教程,最近开始学这个,发现有一些内容已经过时了,在使用过程中也发现了一些坑,总结一下,形成...
    拾识物者阅读 2,797评论 0 17
  • Java 中的注解(Annotation) 是一个很方便的特性在Spring当中得到了大量的应用 , 我们也可以开...
    _秋天阅读 8,898评论 3 22
  • 什么是注解注解分类注解作用分类 元注解 Java内置注解 自定义注解自定义注解实现及使用编译时注解注解处理器注解处...
    Mr槑阅读 1,075评论 0 3
  • 前面走出一位美丽可爱的少女,她就是女主角,她叫白钰,所在的家庭很富裕,但她从不看不起任何人,她的爸爸是一名有名企业...
    木零阅读 506评论 0 0