Spring boot | tools

看下 Spring boot tools 子項目包含的內容:


Spring-boot-tools.png

重點工具介紹
1拂封、spring-boot-annotation-processor
2善延、spring-boot-maven-plugin
3、spring-boot-loader

1签餐、Spring Boot Annotation Processor

Annotation Processor 是一種利用java 注解 擴展javac 編譯功能的一種方式包颁。
定義一個Processor 铆帽,Processor 可以通過 javac 指定參數 類名的方式獲取,也可以使用服務發(fā)現的方式羹与,javac 會自動掃描類路徑下面 META-INF/services/javax.annotation.processing.Processor文件里面的實現類故硅。spring boot 使用后者的方式,所以每次項目編譯纵搁,都會觸發(fā)Processor 里面的邏輯吃衅。我們可以使用 JavaCompiler 類對書寫的Processor 進行功能測試。詳見 Spring boot

org.springframework.boot.testsupport.compiler.TestCompiler

javac 命令說明:

javac -help
 -processor <class1>[,<class2>,<class3>...] 要運行的注釋處理程序的名稱; 繞過默認的搜索進程
  -processorpath <路徑>        指定查找注釋處理程序的位置

測試代碼:
1腾誉、定義一個BuilderProcessor 自動生成POJO 的builder

   public class Person {
 
    private int age;
 
    private String name;
 
    // getters and setters …
 
}

通過BuilderProcessor 在編譯時候生成如下類徘层。

Person person = new PersonBuilder()
  .setAge(25)
  .setName("John")
  .build();

0、定義一個注解

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.SOURCE)
public @interface BuilderProperty {
}

1利职、BuilderProcessor 實現 抽象類AbstractProcessor

@SupportedAnnotationTypes(
  "com.github.yulechen.BuilderProperty")
@SupportedSourceVersion(SourceVersion.RELEASE_8)
public class BuilderProcessor extends  AbstractProcessor {
 

    @Override
    public boolean process(Set<? extends TypeElement> annotations,
                           RoundEnvironment roundEnv) {
        System.out.println("start process java source files");
        for (TypeElement annotation : annotations) {
            Set<? extends Element> annotatedElements
                    = roundEnv.getElementsAnnotatedWith(annotation);

            Map<Boolean, List<Element>> annotatedMethods = annotatedElements.stream().collect(
                    Collectors.partitioningBy(element ->
                            ((ExecutableType) element.asType()).getParameterTypes().size() == 1
                                    && element.getSimpleName().toString().startsWith("set")));

            List<Element> setters = annotatedMethods.get(true);
            List<Element> otherMethods = annotatedMethods.get(false);

            otherMethods.forEach(element ->
                    processingEnv.getMessager().printMessage(Diagnostic.Kind.ERROR,
                            "@BuilderProperty must be applied to a setXxx method "
                                    + "with a single argument", element));

            if (setters.isEmpty()) {
                continue;
            }

            String className = ((TypeElement) setters.get(0)
                    .getEnclosingElement()).getQualifiedName().toString();

            Map<String, String> setterMap = setters.stream().collect(Collectors.toMap(
                    setter -> setter.getSimpleName().toString(),
                    setter -> ((ExecutableType) setter.asType())
                            .getParameterTypes().get(0).toString()
            ));
            try {
                writeBuilderFile(className,setterMap);
            } catch (IOException e) {
                e.printStackTrace();
            }

        }

        return true;
    }

    private void writeBuilderFile(
            String className, Map<String, String> setterMap)
            throws IOException {

        String packageName = null;
        int lastDot = className.lastIndexOf('.');
        if (lastDot > 0) {
            packageName = className.substring(0, lastDot);
        }

        String simpleClassName = className.substring(lastDot + 1);
        String builderClassName = className + "Builder";
        String builderSimpleClassName = builderClassName
                .substring(lastDot + 1);

        JavaFileObject builderFile = processingEnv.getFiler()
                .createSourceFile(builderClassName);

        try (PrintWriter out = new PrintWriter(builderFile.openWriter())) {

            if (packageName != null) {
                out.print("package ");
                out.print(packageName);
                out.println(";");
                out.println();
            }

            out.print("public class ");
            out.print(builderSimpleClassName);
            out.println(" {");
            out.println();

            out.print("    private ");
            out.print(simpleClassName);
            out.print(" object = new ");
            out.print(simpleClassName);
            out.println("();");
            out.println();

            out.print("    public ");
            out.print(simpleClassName);
            out.println(" build() {");
            out.println("        return object;");
            out.println("    }");
            out.println();

            setterMap.entrySet().forEach(setter -> {
                String methodName = setter.getKey();
                String argumentType = setter.getValue();

                out.print("    public ");
                out.print(builderSimpleClassName);
                out.print(" ");
                out.print(methodName);

                out.print("(");

                out.print(argumentType);
                out.println(" value) {");
                out.print("        object.");
                out.print(methodName);
                out.println("(value);");
                out.println("        return this;");
                out.println("    }");
                out.println();
            });

            out.println("}");
        }
    }
}

2趣效、建一個服務發(fā)現文件 META-INF/services/javax.annotation.processing.Processor ,文件里面為BuilderProcessor 全路徑名稱。

3猪贪、將0跷敬,1,2 編譯成jar 包热押,供其他項目引用西傀。加入jar 包名稱為processor.jar


processor.jar.png

4、另一個項目引用processor.jar

public class Person {
    private int age;
    private String name;

    public int getAge() {
        return age;
    }
    @BuilderProperty
    public void setAge(int age) {
        this.age = age;
    }

    public String getName() {
        return name;
    }
    @BuilderProperty
    public void setName(String name) {
        this.name = name;
    }
}

編譯Person 文件

 javac $XXXPATH/Person.java -cp $XXXLIBPATH/processor.jar

編譯結果楞黄,產生了一個PersonBuilder


BuilderProcessor編譯結果.png

Spring boot 有兩個processor
1池凄、spring-boot-configuration-annotation-processor 項目下面的 ConfigurationMetadataAnnotationProcessor 會掃描屬性相關的注解
ConfigurationProperties,NestedConfigurationProperty鬼廓,DeprecatedConfigurationProperty肿仑。然后生成 META-INF/spring-configuration-metadata.json 文件。

protected ConfigurationMetadata writeMetaData() throws Exception {
        ConfigurationMetadata metadata = this.metadataCollector.getMetadata();
        metadata = mergeAdditionalMetadata(metadata);
        if (!metadata.getItems().isEmpty()) {
            this.metadataStore.writeMetadata(metadata);
            return metadata;
        }
        return null;
    }

2、spring-boot-autoconfigure-annotation-processor 項目下面AutoConfigureAnnotationProcessor 尤慰,它處理的注解有

@ConditionalOnClass
@ConditionalOnBean
@ConditionalOnSingleCandidate
@ConditionalOnWebApplication
@AutoConfigureBefore
@AutoConfigureAfter
@AutoConfigureOrder

編譯處理邏輯:
生成一個"META-INF/spring-autoconfigure-metadata.properties" 文件馏锡。

    private void writeProperties() throws IOException {
        if (!this.properties.isEmpty()) {
            Filer filer = this.processingEnv.getFiler();
            FileObject file = filer.createResource(StandardLocation.CLASS_OUTPUT, "", PROPERTIES_PATH);
            try (Writer writer = new OutputStreamWriter(file.openOutputStream(), StandardCharsets.UTF_8)) {
                for (Map.Entry<String, String> entry : this.properties.entrySet()) {
                    writer.append(entry.getKey());
                    writer.append("=");
                    writer.append(entry.getValue());
                    writer.append(System.lineSeparator());
                }
            }
        }
    }

Spring 專題

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市伟端,隨后出現的幾起案子杯道,更是在濱河造成了極大的恐慌,老刑警劉巖责蝠,帶你破解...
    沈念sama閱讀 217,185評論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件党巾,死亡現場離奇詭異,居然都是意外死亡霜医,警方通過查閱死者的電腦和手機齿拂,發(fā)現死者居然都...
    沈念sama閱讀 92,652評論 3 393
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來肴敛,“玉大人署海,你說我怎么就攤上這事∫侥校” “怎么了砸狞?”我有些...
    開封第一講書人閱讀 163,524評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長镀梭。 經常有香客問我刀森,道長,這世上最難降的妖魔是什么丰辣? 我笑而不...
    開封第一講書人閱讀 58,339評論 1 293
  • 正文 為了忘掉前任撒强,我火速辦了婚禮,結果婚禮上笙什,老公的妹妹穿的比我還像新娘飘哨。我一直安慰自己,他們只是感情好琐凭,可當我...
    茶點故事閱讀 67,387評論 6 391
  • 文/花漫 我一把揭開白布芽隆。 她就那樣靜靜地躺著,像睡著了一般统屈。 火紅的嫁衣襯著肌膚如雪胚吁。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,287評論 1 301
  • 那天愁憔,我揣著相機與錄音腕扶,去河邊找鬼。 笑死吨掌,一個胖子當著我的面吹牛半抱,可吹牛的內容都是我干的脓恕。 我是一名探鬼主播,決...
    沈念sama閱讀 40,130評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼窿侈,長吁一口氣:“原來是場噩夢啊……” “哼炼幔!你這毒婦竟也來了?” 一聲冷哼從身側響起史简,我...
    開封第一講書人閱讀 38,985評論 0 275
  • 序言:老撾萬榮一對情侶失蹤乃秀,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后圆兵,有當地人在樹林里發(fā)現了一具尸體跺讯,經...
    沈念sama閱讀 45,420評論 1 313
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 37,617評論 3 334
  • 正文 我和宋清朗相戀三年殉农,在試婚紗的時候發(fā)現自己被綠了抬吟。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,779評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡统抬,死狀恐怖,靈堂內的尸體忽然破棺而出危队,到底是詐尸還是另有隱情聪建,我是刑警寧澤,帶...
    沈念sama閱讀 35,477評論 5 345
  • 正文 年R本政府宣布茫陆,位于F島的核電站金麸,受9級特大地震影響,放射性物質發(fā)生泄漏簿盅。R本人自食惡果不足惜挥下,卻給世界環(huán)境...
    茶點故事閱讀 41,088評論 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望桨醋。 院中可真熱鬧棚瘟,春花似錦、人聲如沸喜最。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,716評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽瞬内。三九已至迷雪,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間虫蝶,已是汗流浹背章咧。 一陣腳步聲響...
    開封第一講書人閱讀 32,857評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留能真,地道東北人赁严。 一個月前我還...
    沈念sama閱讀 47,876評論 2 370
  • 正文 我出身青樓扰柠,卻偏偏與公主長得像,于是被迫代替她去往敵國和親误澳。 傳聞我的和親對象是個殘疾皇子耻矮,可洞房花燭夜當晚...
    茶點故事閱讀 44,700評論 2 354