91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

java中javac AbstractProcessor有什么用

發布時間:2021-11-09 11:15:12 來源:億速云 閱讀:138 作者:小新 欄目:開發技術

小編給大家分享一下java中javac AbstractProcessor有什么用,希望大家閱讀完這篇文章之后都有所收獲,下面讓我們一起去探討吧!

它可以做什么?

它做的事情當然是生成新類或修改原始的類,比如你遇到這樣的情況下就可以使用:

  • 反射好慢,曾見過一個大廠大量是Gson,由于Gson序列化時大量使用了反射,每一個field,每一個get、set都需要用反射,由此帶來了性能問題。解決方法就是使用它盡量減少反射(替換成JSONObject)

  • 生成代碼,只要是有注解的地方都可以讀取,總之很多(有些android orm框架)

Processor

javax.annotation.processing.Processor 這個接口將提供注解處理,它遵循SPI規約進行拓展,jdk默認就有很多處理器的實現。

AbstractProcessor

注解處理器是最重要的拓展處理類了。

注意:請確認JAVA的環境變量已經配置成功,并且把tools.jar(它源于此包)加入到自己電腦的環境變量中

ProcessingEnvironment是一個注解處理工具的集合
Element是一個接口,表示一個程序元素,它可以是包、類、方法或者一個變量。Element已知的子接口有:
java中javac AbstractProcessor有什么用
PackageElement 表示一個包程序元素。提供對有關包及其成員的信息的訪問。
ExecutableElement 表示某個類或接口的方法、構造方法或初始化程序(靜態或實例),包括注釋類型元素。
TypeElement 表示一個類或接口程序元素。提供對有關類型及其成員的信息的訪問。注意,枚舉類型是一種類,而注解類型是一種接口。
VariableElement 表示一個字段、enum 常量、方法或構造方法參數、局部變量或異常參數。

源碼

重點關注process方法

// 源于javax.annotation.processing;
public abstract class AbstractProcessor implements Processor {
    // 集合中指定支持的注解類型的名稱(這里必須時完整的包名+類名)
    public Set<String> getSupportedAnnotationTypes() {
            SupportedAnnotationTypes sat = this.getClass().getAnnotation(SupportedAnnotationTypes.class);
            if  (sat == null) {
                if (isInitialized())
                    processingEnv.getMessager().printMessage(Diagnostic.Kind.WARNING,
                                                             "No SupportedAnnotationTypes annotation " +
                                                             "found on " + this.getClass().getName() +
                                                             ", returning an empty set.");
                return Collections.emptySet();
            }
            else
                return arrayToSet(sat.value());
        }
    // 指定當前正在使用的Java版本
    public SourceVersion getSupportedSourceVersion() {
        SupportedSourceVersion ssv = this.getClass().getAnnotation(SupportedSourceVersion.class);
        SourceVersion sv = null;
        if (ssv == null) {
            sv = SourceVersion.RELEASE_6;
            if (isInitialized())
                processingEnv.getMessager().printMessage(Diagnostic.Kind.WARNING,
                                                         "No SupportedSourceVersion annotation " +
                                                         "found on " + this.getClass().getName() +
                                                         ", returning " + sv + ".");
        } else
            sv = ssv.value();
        return sv;
    }
    // 初始化處理器
    public synchronized void init(ProcessingEnvironment processingEnv) {
        if (initialized)
            throw new IllegalStateException("Cannot call init more than once.");
        Objects.requireNonNull(processingEnv, "Tool provided null ProcessingEnvironment");
 
        this.processingEnv = processingEnv;
        initialized = true;
    }
    /**
     * 這些注解是否由此 Processor 處理,該方法返回ture表示該注解已經被處理, 后續不會再有其他處理器處理; 返回false表示仍可被其他處理器處理
     */
    public abstract boolean process(Set<? extends TypeElement> annotations,
                                    RoundEnvironment roundEnv);
}

實現一個打印可以API的功能

由于本人是maven環境,以此展開講

<build>
    <plugins>
        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-compiler-plugin</artifactId>
            <configuration>
                <source>1.8</source>
                <target>1.8</target>
                <!--Disable annotation processing for ourselves-->
                <!--<compilerArgument>-proc:none</compilerArgument>-->
            </configuration>
        </plugin>
    </plugins>
</build>

步驟1:實現一個注解處理器

@Retention(RetentionPolicy.SOURCE)
@Target({ElementType.TYPE, ElementType.METHOD})
public @interface ApiAnnotation {
    String author() default "alex.chen";
    String date();
    int version() default 1;
}
@SupportedAnnotationTypes({"com.kxtx.annotation.ApiAnnotation"})
@SupportedSourceVersion(SourceVersion.RELEASE_8)
//@AutoService(Processor.class)
public class MyProcessor extends AbstractProcessor {
    //類名的前綴、后綴
    public static final String SUFFIX = "AutoGenerate";
    public static final String PREFIX = "My_";
    @Override
    public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment env) {
        Messager messager = processingEnv.getMessager();
        for (TypeElement typeElement : annotations) {
            for (Element e : env.getElementsAnnotatedWith(typeElement)) {
                //打印
                messager.printMessage(Diagnostic.Kind.WARNING, "Printing:" + e.toString());
                messager.printMessage(Diagnostic.Kind.WARNING, "Printing:" + e.getSimpleName());
                messager.printMessage(Diagnostic.Kind.WARNING, "Printing:" + e.getEnclosedElements().toString());
 
                //獲取注解
                ApiAnnotation annotation = e.getAnnotation(ApiAnnotation.class);
                //獲取元素名并將其首字母大寫
                String name = e.getSimpleName().toString();
                char c = Character.toUpperCase(name.charAt(0));
                name = String.valueOf(c + name.substring(1));
                //包裹注解元素的元素, 也就是其父元素, 比如注解了成員變量或者成員函數, 其上層就是該類
                Element enclosingElement = e.getEnclosingElement();
                //獲取父元素的全類名,用來生成報名
                String enclosingQualifiedname;
                if (enclosingElement instanceof PackageElement) {
                    enclosingQualifiedname = ((PackageElement) enclosingElement).getQualifiedName().toString();
                } else {
                    enclosingQualifiedname = ((TypeElement) enclosingElement).getQualifiedName().toString();
                }
                try {
                    //生成包名
                    String generatePackageName = enclosingQualifiedname.substring(0, enclosingQualifiedname.lastIndexOf("."));
                    // 生成的類名
                    String genarateClassName = PREFIX + enclosingElement.getSimpleName() + SUFFIX;
                    //創建Java 文件
                    JavaFileObject f = processingEnv.getFiler().createSourceFile(genarateClassName);
                    // 在控制臺輸出文件路徑
                    messager.printMessage(Diagnostic.Kind.WARNING, "Printing: " + f.toUri());
                    Writer w = f.openWriter();
                    try {
                        PrintWriter pw = new PrintWriter(w);
                        pw.println("package " + generatePackageName + ";");
                        pw.println("\npublic class " + genarateClassName + " { ");
                        pw.println("\n    /** 打印值 */");
                        pw.println("    public static void print" + name + "() {");
                        pw.println("        // 注解的父元素: " + enclosingElement.toString());
                        pw.println("        System.out.println(\"代碼生成的路徑: " + f.toUri() + "\");");
                        pw.println("        System.out.println(\"注解的元素: " + e.toString() + "\");");
                        pw.println("        System.out.println(\"注解的版本: " + annotation.version() + "\");");
                        pw.println("        System.out.println(\"注解的作者: " + annotation.author() + "\");");
                        pw.println("        System.out.println(\"注解的日期: " + annotation.date() + "\");");
 
                        pw.println("    }");
                        pw.println("}");
                        pw.flush();
                    } finally {
                        w.close();
                    }
                } catch (IOException e1) {
                    processingEnv.getMessager().printMessage(Diagnostic.Kind.ERROR,
                            e1.toString());
                }
            }
        }
        return true;
    }
}

步驟2:配置一個spi,在resources目錄新建META-INF/services/javax.annotation.processing.Processor,內容為MyProcessor類全名。

步驟3:在另一個項目中使用@ApiAnnotation就會發現生成了一個新My_feignAutoGenerate.class文件:

public class My_feignAutoGenerate {
    public My_feignAutoGenerate() {
    }
    public static void printStartUp() {
        System.out.println("代碼生成的路徑: file:/C:/Users/Administrator/Desktop/feign-async-master/target/generated-sources/annotations/My_feignAutoGenerate.java");
        System.out.println("注解的元素: com.github.feign.StartUp");
        System.out.println("注解的版本: 1");
        System.out.println("注解的作者: alex");
        System.out.println("注解的日期: 2019-03-6");
    }
}

到這里基本上已經演示完了。

google的 auto-service

<dependency>
    <groupId>com.google.auto.service</groupId>
    <artifactId>auto-service</artifactId>
    <version>1.0-rc2</version>
</dependency>

這個類庫非常有用,它非常簡單,使用@AutoService(Processor.class)會基于該接口和注解的類上自動幫我們生成META-INF/services下對應spi文件。它實現的原理就是通過注解處理器。

javapoet

有沒有覺得上面pw.println("package " + generatePackageName + ";");這樣的代碼很痛苦啊?

JavaPoet is a Java API for generating .java source files.

package com.example.helloworld; 
public final class HelloWorld {
  public static void main(String[] args) {
    System.out.println("Hello, JavaPoet!");
  }
}
MethodSpec main = MethodSpec.methodBuilder("main")
    .addModifiers(Modifier.PUBLIC, Modifier.STATIC)
    .returns(void.class)
    .addParameter(String[].class, "args")
    .addStatement("$T.out.println($S)", System.class, "Hello, JavaPoet!")
    .build();
 
TypeSpec helloWorld = TypeSpec.classBuilder("HelloWorld")
    .addModifiers(Modifier.PUBLIC, Modifier.FINAL)
    .addMethod(main)
    .build();
 
JavaFile javaFile = JavaFile.builder("com.example.helloworld", helloWorld)
    .build();
 
javaFile.writeTo(System.out);

你喜歡的lombok實現原理是怎樣的呢?

lombok(用來幫助開發人員消除 Java 對象 的冗長),非常好用

java中javac AbstractProcessor有什么用

看完了這篇文章,相信你對“java中javac AbstractProcessor有什么用”有了一定的了解,如果想了解更多相關知識,歡迎關注億速云行業資訊頻道,感謝各位的閱讀!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

九台市| 湘阴县| 德兴市| 内乡县| 平和县| 佛山市| 呼伦贝尔市| 惠水县| 清原| 双城市| 沙坪坝区| 阿拉善盟| 信阳市| 边坝县| 宣化县| 临海市| 黄龙县| 丹凤县| 丹巴县| 通山县| 兰溪市| 杂多县| 嵩明县| 淮安市| 石景山区| 礼泉县| 南昌县| 新河县| 那坡县| 蒙阴县| 山丹县| 宜都市| 阿勒泰市| 宜阳县| 白水县| 虎林市| 珠海市| 志丹县| 莎车县| 阿鲁科尔沁旗| 博客|