日韩无码专区无码一级三级片|91人人爱网站中日韩无码电影|厨房大战丰满熟妇|AV高清无码在线免费观看|另类AV日韩少妇熟女|中文日本大黄一级黄色片|色情在线视频免费|亚洲成人特黄a片|黄片wwwav色图欧美|欧亚乱色一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時(shí)間:8:30-17:00
你可能遇到了下面的問(wèn)題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營(yíng)銷解決方案
AndroidStudio中使用apt

一、前言

成都創(chuàng)新互聯(lián)公司從2013年創(chuàng)立,先為山東等服務(wù)建站,山東等地企業(yè),進(jìn)行企業(yè)商務(wù)咨詢服務(wù)。為山東企業(yè)網(wǎng)站制作PC+手機(jī)+微官網(wǎng)三網(wǎng)同步一站式服務(wù)解決您的所有建站問(wèn)題。

你還在對(duì)著枯燥的重復(fù)代碼一味復(fù)制粘貼嗎?這樣跟搬磚有何區(qū)別?你是否曾想過(guò):你用代碼編寫(xiě)出一個(gè)自動(dòng)化的APP,但為何代碼本身卻缺少了活力?掌握Android-apt,杜絕重復(fù)代碼,讓你寫(xiě)代碼如寫(xiě)詩(shī)般優(yōu)雅。

二、何為apt?

apt意為:annotation processing tool(注解處理工具),這家伙可神奇了,它能通過(guò)注解,在編譯期自動(dòng)生成特定的Java文件,實(shí)現(xiàn)自動(dòng)編寫(xiě)代碼。

問(wèn):有什么用?憑我自己本事能寫(xiě)出來(lái)的代碼,為什么要自動(dòng)化?

大哥,你這是又想施展你的復(fù)制粘貼大法了嗎?稍安勿躁,細(xì)看完這篇文章,你會(huì)愛(ài)上這家伙的。

鼎鼎大名的ButterKnife、Dagger2這兩個(gè)開(kāi)源庫(kù),相信你一定有聽(tīng)過(guò),你應(yīng)該知道我為什么提到它們了吧。沒(méi)錯(cuò)!這兩個(gè)開(kāi)源庫(kù)都是基于apt的。

三、說(shuō)了這么多,要怎么用啊?別急,我們先搭建環(huán)境(基于gradle插件2.2.0以上版本)

1.在Android Studio中新建一個(gè)Java module,用于存裝注解處理邏輯,名字隨便啦,反正我一般都取名:apt。很重要的事:在app module中添加注解處理依賴:annotationProcessor project(‘:apt’)

(解釋原因:由于android的module中不包含有apt相關(guān)類,因此需要新建一個(gè)java module來(lái)編寫(xiě)apt邏輯。什么?你不信?不信你寫(xiě)個(gè)類繼承AbstractProcessor試試)

2.再次新建一個(gè)module(android、java都可以),用于存裝注解,名字也隨便,反正我這里取名為:anno,并且在app、apt的build.gradle文件下,添加依賴compile project(‘:anno’)

(為什么要新建module去盛裝注解類,而不放到app module或者apt module中去:最主要的原因就是app module與apt module不能直接相互依賴,至于為什么不能直接依賴,我就不細(xì)說(shuō)了,總之一句話:不信你試試看就知道嘍!)

3.在apt的build.gradle里,添加如下依賴。到此,我們的環(huán)境配置工作就告一段落了。

(其中:1.auto-service是用于注解后自動(dòng)在特定路徑下生成配置文件;2.javapoet是用于配合apt便捷生成java文件的工具。相信這樣解釋大家還云里霧里,不要著急,繼續(xù)往下看)

四、環(huán)境搭建好了,接下來(lái)就是秀操作時(shí)間

1.首先,在anno module里新建一個(gè)注解類

 
 
 
 
  1. @Retention(RetentionPolicy.SOURCE)
  2. @Target(ElementType.METHOD)
  3. public @interface Test {
  4.     String value();

2.在apt module里新建一個(gè)注解處理類,繼承于AbstractProcessor

 
 
 
 
  1. public class TestProcessor extends AbstractProcessor{
  2.     @Override
  3.     public boolean process(Set annotations, RoundEnvironment roundEnv) {
  4.         return false;
  5.     }

3.既然說(shuō)apt是要自動(dòng)生成java文件,那我們就需要擬構(gòu)出一個(gè)目標(biāo)類。假設(shè)我們要生成這樣一個(gè)類

 
 
 
 
  1. public class TestClass {
  2.     public static void main(String[] args){
  3.         System.out.println("Hallo world!");
  4.     }

4.操作注解處理類,生成目標(biāo)java文件

 
 
 
 
  1. @AutoService(Processor.class)
  2. @SupportedAnnotationTypes({
  3.         "com.aop.anno.Test"
  4. })
  5. public class TestProcessor extends AbstractProcessor{
  6.     @Override
  7.     public boolean process(Set annotations, RoundEnvironment roundEnv) {
  8.         //生成TestClass類
  9.         TypeSpec.Builder tb = TypeSpec.classBuilder("TestClass")
  10.                 .addModifiers(Modifier.PUBLIC);
  11.         //生成main方法
  12.         MethodSpec.Builder mb = MethodSpec.methodBuilder("main")
  13.                 .addModifiers(Modifier.PUBLIC, Modifier.STATIC)
  14.                 .returns(void.class)
  15.                 .addParameter(String[].class, "args");
  16.         //生成代碼塊,并添加到main方法中
  17.         for(TypeElement e : ElementFilter.typesIn(roundEnv.getElementsAnnotatedWith(Test.class))){
  18.             CodeBlock cb = CodeBlock.builder()
  19.                     .addStatement("$T.out.println(\"$L + $L\")", System.class, 
  20.                     e.getAnnotation(Test.class).value(), e.getSimpleName())
  21.                     .build();
  22.             mb.addCode(cb);
  23.         }
  24.         tb.addMethod(mb.build());
  25.         JavaFile jf = JavaFile.builder("com.example.apt", tb.build()).build();
  26.         //將代碼寫(xiě)入java文件中
  27.         try {
  28.             jf.writeTo(processingEnv.getFiler());
  29.         } catch (IOException e) {
  30.             e.printStackTrace();
  31.         }
  32.         return true;
  33.     }

大致說(shuō)下步驟:

(1)添加@AutoService(Processor.class)注解,這個(gè)注解會(huì)自動(dòng)在指定路徑下生成一個(gè)配置文件:

apt/build/classes/main/META-INF/services/javax.annotation.processing.Processor;

(2)添加@SupportedAnnotationTypes注解,配置這個(gè)類所要處理的注解類型。(傳入String類型參數(shù),格式為:包名+類名);

(3)采用javapoet書(shū)寫(xiě)代碼構(gòu)建邏輯,具體用法去這里看看;

(4)生成代碼塊的主要邏輯是:遍歷所有被@Test注解過(guò)的類,取出注解內(nèi)容及類名打印出來(lái)。

5.在類上添加@Test注解,這里就用MainActivity來(lái)試試

 
 
 
 
  1. @Test("abc")
  2. public class MainActivity extends AppCompatActivity {
  3.     @Override
  4.     protected void onCreate(Bundle savedInstanceState) {
  5.         super.onCreate(savedInstanceState);
  6.         setContentView(R.layout.activity_main);
  7.     }

6.rebuild工程,在app/build/generated/source/apt/debug路徑下找到目標(biāo)java文件。至此,大功告成

TestClass代碼如下:

 
 
 
 
  1. public class TestClass {
  2.   public static void main(String[] args) {
  3.     System.out.println("abc + MainActivity");
  4.   }

五、然而并沒(méi)什么卵用?

確實(shí),到此為止,我們確實(shí)是用了幾十行代碼去生成了一個(gè)5行代碼的TestClass,這種操作來(lái)說(shuō)看起來(lái)可以用4個(gè)字來(lái)形容:閑得蛋疼。

然而,接下來(lái)的操作,會(huì)讓你耳目一新。首先我們新建幾個(gè)測(cè)試類,假如我們新建了這樣4個(gè)測(cè)試類:ActivityA,ActivityB,ActivityC,ActivityD,并且都給他們加上注解@Test。然后rebuild一下,你會(huì)發(fā)現(xiàn),我們的TestClass變了樣:

 
 
 
 
  1. public class TestClass {
  2.   public static void main(String[] args) {
  3.     System.out.println("A + ActivityA");
  4.     System.out.println("B + ActivityB");
  5.     System.out.println("C + ActivityC");
  6.     System.out.println("D + ActivityD");
  7.     System.out.println("abc + MainActivity");
  8.   }
  9. }

恍然大悟!原來(lái),是這么玩的!這時(shí)候,你是否已經(jīng)感覺(jué)到apt的魅力了呢?是的,它能幫你干掉重復(fù)代碼,讓你杜絕掉復(fù)制粘貼。


新聞標(biāo)題:AndroidStudio中使用apt
文章分享:http://m.5511xx.com/article/coigjge.html