在开始实施之前,我们必须制定我们的战略。这将减少命中和试验的次数。
注释处理在处理 Java 注释源代码时提供的东西:
所以,我们有一组注释和一个元素列表。我们的库将生成一个包装类,该类将帮助映射活动的视图和点击监听器。
它将具有以下用法:
activity_main.xml定义了一个TextView
带有 id的tv_content
按钮和两个带有 id 和bt_1
的按钮bt_2
。我们的注释将映射视图和按钮以删除样板,就像 ButterKnife 一样。
public class MainActivity extends AppCompatActivity {
@BindView(R.id.tv_content)
TextView tvContent;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Binding.bind(this);
}
@OnClick(R.id.bt_1)
void bt1Click(View v) {
tvContent.setText("Button 1 Clicked");
}
@OnClick(R.id.bt_2)
void bt2Click(View v) {
tvContent.setText("Button 2 Clicked");
}
}
我们将使用MainActivity定义通过注释处理自动生成名为MainActivity$Binding的包装类。
注意:我们将在其中使用注解的任何 Activity 都将创建一个名称以Binding Java 源代码文件的创建。
处理后将创建以下类。
@Keep
public class MainActivity$Binding {
public MainActivity$Binding(MainActivity activity) {
bindViews(activity);
bindOnClicks(activity);
}
private void bindViews(MainActivity activity) {
activity.tvContent = (TextView)activity.findViewById(2131165322);
}
private void bindOnClicks(final MainActivity activity) {
activity.findViewById(2131165218).setOnClickListener(new View.OnClickListener() {
public void onClick(View view) {
activity.bt1Click(view);
}
});
activity.findViewById(2131165219).setOnClickListener(new View.OnClickListener() {
public void onClick(View view) {
activity.bt2Click(view);
}
});
}
}
现在我们知道我们必须生成什么,让我们分析如何使用我们在处理时掌握的信息来创建它。
@BindView
或@OnClick
来自getRootElements方法提供的元素列表的类 (Type) 元素。对于第 1 步,我们希望提高搜索效率。因此,我们将创建一个具有过滤方法的类ProcessingUtils。
public class ProcessingUtils {
private ProcessingUtils() {
// not to be instantiated in public
}
public static Set<TypeElement> getTypeElementsToProcess(Set<? extends Element> elements,
Set<? extends Element> supportedAnnotations) {
Set<TypeElement> typeElements = new HashSet<>();
for (Element element : elements) {
if (element instanceof TypeElement) {
boolean found = false;
for (Element subElement : element.getEnclosedElements()) {
for (AnnotationMirror mirror : subElement.getAnnotationMirrors()) {
for (Element annotation : supportedAnnotations) {
if (mirror.getAnnotationType().asElement().equals(annotation)) {
typeElements.add((TypeElement) element);
found = true;
break;
}
}
if (found) break;
}
if (found) break;
}
}
}
return typeElements;
}
}
tvContent
、onCreate
、bt1Click
和bt2Click
其他继承的成员。@Override
对于onCreate
,@BindView
对于tvContent
和@OnClick
对于bt1Click
。因此,将MainActivitygetTypeElementsToProcess
过滤为我们需要处理的TypeElement 。
现在,我们将扫描所有过滤后的元素以创建相应的包装类。
elementUtils.getPackageOf(typeElement).getQualifiedName().toString()
在我们的例子中:com.mindorks.annotation.processing.example)typeElement.getSimpleName().toString()
在我们的例子中为 MainActivity)ClassName.get(packageName, typeName)
它将为 MainActivity 创建一个 ClassName)注意:为了便于名称维护和良好的编码习惯,我们将创建一个名为NameStore的类。它将包含我们在定义 Binding 类时需要的所有类、变量和方法名称。
public final class NameStore {
private NameStore() {
// not to be instantiated in public
}
public static String getGeneratedClassName(String clsName) {
return clsName + BindingSuffix.GENERATED_CLASS_SUFFIX;
}
public static class Package {
public static final String ANDROID_VIEW = "android.view";
}
public static class Class {
// Android
public static final String ANDROID_VIEW = "View";
public static final String ANDROID_VIEW_ON_CLICK_LISTENER = "OnClickListener";
}
public static class Method {
// Android
public static final String ANDROID_VIEW_ON_CLICK = "onClick";
// Binder
public static final String BIND_VIEWS = "bindViews";
public static final String BIND_ON_CLICKS = "bindOnClicks";
public static final String BIND = "bind";
}
public static class Variable {
public static final String ANDROID_ACTIVITY = "activity";
public static final String ANDROID_VIEW = "view";
}
}
此外,您会发现在binder-annotations库的 ( internal -> BindingSuffix)类中添加了$Binding后缀。这样做有两个目的。
JavaPoet使定义类结构并在处理时编写它变得非常简单。它创建非常接近手写代码的类。它提供了自动推断导入以及美化代码的工具。
要使用 JavaPoet,我们需要将以下依赖项添加到binder-compiler模块中。
dependencies {
implementation project(':binder-annotations')
implementation 'com.squareup:javapoet:1.11.1'
}
注意:使用JavaFileObject是非常不切实际和麻烦的。所以,我们甚至不会谈论它。
本教程所需的JavaPoet的基本用法(任何提前了解都可以从其<u style="text-decoration: none; border-bottom: 1px solid rgb(68, 68, 68);">GitHub Repo</u>中获得。)
addStatement("$N($N)", "bindViews", "activity")
这将生成代码bindViews(activity))。PlaceHolders : 其余的东西可以参考这个JavaPoet的基本介绍很容易理解。我把休息留给你自己弄清楚。我就是这样学习的。
使用 JavaPoet 编写定义的类模式非常简单。
// write the defines class to a java file
try {
JavaFile.builder(packageName,
classBuilder.build())
.build()
.writeTo(filer);
} catch (IOException e) {
messager.printMessage(Diagnostic.Kind.ERROR, e.toString(), typeElement);
}
它将在文件夹中生成源代码。/app/build/generated/source/apt/debug
在我们的例子中:/app/build/generated/source/apt/debug/com/mindorks/annotation/processing/example/MainActivity$Binding.java
作者:Janishar Ali
链接:Android Annotation Processing Tutorial: Part 3: Generate Java Source Code