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

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

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營(yíng)銷解決方案
注解APT應(yīng)用詳解(手把手教你寫B(tài)utterKnife工具)

一、apt是什么?有什么用,帶著疑惑來(lái)學(xué)習(xí)

  • APT(Annotation Processing Tool)即注解處理器,是一種處理注解的工具,確切的說它是javac的一個(gè)工具,它用來(lái)在編譯時(shí)掃描和處理注解。注解處理器以Java代碼(或者編譯過的字節(jié)碼)作為輸入,生成.java文件作為輸出;
  • 簡(jiǎn)單來(lái)說就是在編譯期,通過注解生成.java文件;
  • 使用APT的優(yōu)點(diǎn)就是方便、簡(jiǎn)單,可以少些很多重復(fù)的代碼;用過Butterknife、Dagger、EventBus等注解框架的同學(xué)就能感受到,利用這些框架可以少些很多代碼,只要寫一些注解就可以了,他們不過是通過注解,幫助生成了一些高效代碼;

二、APT應(yīng)用-仿照ButterKnife寫個(gè)注解

通過APT實(shí)現(xiàn)一個(gè)功能,通過對(duì)View變量的注解,實(shí)現(xiàn)View的綁定

創(chuàng)新互聯(lián)長(zhǎng)期為上1000+客戶提供的網(wǎng)站建設(shè)服務(wù),團(tuán)隊(duì)從業(yè)經(jīng)驗(yàn)10年,關(guān)注不同地域、不同群體,并針對(duì)不同對(duì)象提供差異化的產(chǎn)品和服務(wù);打造開放共贏平臺(tái),與合作伙伴共同營(yíng)造健康的互聯(lián)網(wǎng)生態(tài)環(huán)境。為新華企業(yè)提供專業(yè)的成都做網(wǎng)站、成都網(wǎng)站制作、成都外貿(mào)網(wǎng)站建設(shè),新華網(wǎng)站改版等技術(shù)服務(wù)。擁有十年豐富建站經(jīng)驗(yàn)和眾多成功案例,為您定制開發(fā)。

1、創(chuàng)建幾個(gè)Library來(lái)聲明

 
 
 
 
  1. Android Library:aptlibs 正常的寫Android的lib  
  2. Java or Kotlin Library:aptlib-anno (專門放我們編寫的注解)
  3. Java or Kotlin Library :aptlib-processor (編寫動(dòng)態(tài)生成文件的邏輯)
  4. aptlibs
  5. plugins {
  6.     id 'com.android.library'
  7.     id 'kotlin-android'
  8. }
  9. aptlib-anno 
  10. plugins {
  11.     id 'java-library'
  12. }
  13. aptlib-processor
  14. 是plugins {
  15.     id 'java-library'
  16. }

這個(gè)要記清楚,很多博主估計(jì)自己都沒有寫過apt,分不清楚AndroidLib和javaLib

apt 本來(lái)java 提供的,另外 Android庫(kù)中不允許繼承AbstractProcessor

2 、定義注解-自定義注解

記住要在 aptlib-anno 庫(kù)下面創(chuàng)建

 
 
 
 
  1. @Retention(RetentionPolicy.CLASS)
  2. @Target(ElementType.FIELD)
  3. public @interface BindView {
  4.     int value();
  5. }

定義了運(yùn)行時(shí)注解BindView,其中value()用于獲取對(duì)應(yīng)View的id;

  • @Retention(RetentionPolicy.CLASS):表示編譯時(shí)注解
  • @Target(ElementType.FIELD):表示注解范圍為類成員(構(gòu)造方法、方法、成員變量)
  • @Retention:定義被保留的時(shí)間長(zhǎng)短
  • RetentionPoicy.SOURCE、RetentionPoicy.CLASS、RetentionPoicy.RUNTIME
  • @Target:定義所修飾的對(duì)象范圍
  • TYPE、FIELD、METHOD、PARAMETER、CONSTRUCTOR、LOCAL_VARIABLE等

3、定義注解處理器-動(dòng)態(tài)生成關(guān)聯(lián)文件

aptlib-processor 庫(kù)

首先在本lib下添加依賴

 
 
 
 
  1. dependencies {
  2.     implementation 'com.google.auto.service:auto-service:1.0-rc2' 
  3.     implementation project(':aptlib-anno')
  4. }

創(chuàng)建BindViewProcessor

 
 
 
 
  1. @AutoService(Processor.class)
  2. public class BindViewProcessor extends AbstractProcessor {
  3.     private Messager mMessager;
  4.     private Elements mElementUtils;
  5.     private Map mProxyMap = new HashMap<>();
  6.     @Override
  7.     public synchronized void init(ProcessingEnvironment processingEnv) {
  8.         super.init(processingEnv);
  9.         mMessager = processingEnv.getMessager();
  10.         mElementUtils = processingEnv.getElementUtils();
  11.     }
  12.     @Override
  13.     public Set getSupportedAnnotationTypes() {
  14.         HashSet supportTypes = new LinkedHashSet<>();
  15.         supportTypes.add(BindView.class.getCanonicalName());
  16.         return supportTypes;
  17.     }
  18.     @Override
  19.     public SourceVersion getSupportedSourceVersion() {
  20.         return SourceVersion.latestSupported();
  21.     }
  22.     @Override
  23.     public boolean process(Set set, RoundEnvironment roundEnv) {
  24.            mMessager.printMessage(Diagnostic.Kind.NOTE, "processing...");
  25.         mProxyMap.clear();
  26.         //得到所有的注解
  27.         Set elements = roundEnvironment.getElementsAnnotatedWith(BindView.class);
  28.         for (Element element : elements) {
  29.             VariableElement variableElement = (VariableElement) element;
  30.             TypeElement classElement = (TypeElement) variableElement.getEnclosingElement();
  31.             String fullClassName = classElement.getQualifiedName().toString();
  32.             ClassCreatorProxy proxy = mProxyMap.get(fullClassName);
  33.             if (proxy == null) {
  34.                 proxy = new ClassCreatorProxy(mElementUtils, classElement);
  35.                 mProxyMap.put(fullClassName, proxy);
  36.             }
  37.             BindView bindAnnotation = variableElement.getAnnotation(BindView.class);
  38.             int id = bindAnnotation.value();
  39.             proxy.putElement(id, variableElement);
  40.         }
  41.         //通過遍歷mProxyMap,創(chuàng)建java文件
  42.         for (String key : mProxyMap.keySet()) {
  43.             ClassCreatorProxy proxyInfo = mProxyMap.get(key);
  44.             try {
  45.                 mMessager.printMessage(Diagnostic.Kind.NOTE, " --> create " + proxyInfo.getProxyClassFullName());
  46.                 JavaFileObject jfo = processingEnv.getFiler().createSourceFile(proxyInfo.getProxyClassFullName(), proxyInfo.getTypeElement());
  47.                 Writer writer = jfo.openWriter();
  48.                 writer.write(proxyInfo.generateJavaCode());
  49.                 writer.flush();
  50.                 writer.close();
  51.             } catch (IOException e) {
  52.                 mMessager.printMessage(Diagnostic.Kind.NOTE, " --> create " + proxyInfo.getProxyClassFullName() + "error");
  53.             }
  54.         }
  55.         mMessager.printMessage(Diagnostic.Kind.NOTE, "process finish ...");
  56.         return true;
  57.     }
  58. }
 
 
 
 
  1. public class ClassCreatorProxy {
  2.     private String mBindingClassName;
  3.     private String mPackageName;
  4.     private TypeElement mTypeElement;
  5.     private Map mVariableElementMap = new HashMap<>();
  6.     public ClassCreatorProxy(Elements elementUtils, TypeElement classElement) {
  7.         this.mTypeElement = classElement;
  8.         PackageElement packageElement = elementUtils.getPackageOf(mTypeElement);
  9.         String packageName = packageElement.getQualifiedName().toString();
  10.         String className = mTypeElement.getSimpleName().toString();
  11.         this.mPackageName = packageName;
  12.         this.mBindingClassName = className + "_ViewBinding";
  13.     }
  14.     public void putElement(int id, VariableElement element) {
  15.         mVariableElementMap.put(id, element);
  16.     }
  17.     /**
  18.      * 創(chuàng)建Java代碼
  19.      * @return
  20.      */
  21.     public String generateJavaCode() {
  22.         StringBuilder builder = new StringBuilder();
  23.         builder.append("package ").append(mPackageName).append(";\n\n");
  24.         builder.append("import com.example.gavin.apt_library.*;\n");
  25.         builder.append('\n');
  26.         builder.append("public class ").append(mBindingClassName);
  27.         builder.append(" {\n");
  28.         generateMethods(builder);
  29.         builder.append('\n');
  30.         builder.append("}\n");
  31.         return builder.toString();
  32.     }
  33.     /**
  34.      * 加入Method
  35.      * @param builder
  36.      */
  37.     private void generateMethods(StringBuilder builder) {
  38.         builder.append("public void bind(" + mTypeElement.getQualifiedName() + " host ) {\n");
  39.         for (int id : mVariableElementMap.keySet()) {
  40.             VariableElement element = mVariableElementMap.get(id);
  41.             String name = element.getSimpleName().toString();
  42.             String type = element.asType().toString();
  43.             builder.append("host." + name).append(" = ");
  44.             builder.append("(" + type + ")(((android.app.Activity)host).findViewById( " + id + "));\n");
  45.         }
  46.         builder.append("  }\n");
  47.     }
  48.     public String getProxyClassFullName()
  49.     {
  50.         return mPackageName + "." + mBindingClassName;
  51.     }
  52.     public TypeElement getTypeElement()
  53.     {
  54.         return mTypeElement;
  55.     }
  56. }
  • init:初始化。可以得到ProcessingEnviroment,ProcessingEnviroment提供很多有用的工具類Elements, Types 和 Filer
  • getSupportedAnnotationTypes:指定這個(gè)注解處理器是注冊(cè)給哪個(gè)注解的,這里說明是注解BindView
  • getSupportedSourceVersion:指定使用的Java版本,通常這里返回SourceVersion.latestSupported()
  • process:可以在這里寫掃描、評(píng)估和處理注解的代碼,生成Java文件
  • auto-service 庫(kù):自動(dòng)生成代碼需要借助的庫(kù)

4、寫工具類BindViewTools

在aptlib項(xiàng)目中寫綁定類

 
 
 
 
  1. public class BindViewTools {
  2.     public static void bind(Activity activity) {
  3.         Class clazz = activity.getClass();
  4.         try {
  5.             Class bindViewClass = Class.forName(clazz.getName() + "_ViewBinding");
  6.             Method method = bindViewClass.getMethod("bind", activity.getClass());
  7.             method.invoke(bindViewClass.newInstance(), activity);
  8.         } catch (Exception e) {
  9.             e.printStackTrace();
  10.         }
  11.     }
  12. }

5、主項(xiàng)目app中引入

 
 
 
 
  1. implementation project(path: ':aptlib')
  2.  annotationProcessor project(path: ':aptlib-process')

在MainActivity中,在View的前面加上BindView注解,把id傳入即可

 
 
 
 
  1. public class MainActivity extends AppCompatActivity {
  2.     @BindView(R.id.tv)
  3.     TextView mTextView;
  4.     @BindView(R.id.btn)
  5.     Button mButton;
  6.     @Override
  7.     protected void onCreate(Bundle savedInstanceState) {
  8.         super.onCreate(savedInstanceState);
  9.         setContentView(R.layout.activity_main);
  10.         BindViewTools.bind(this);
  11.         mTextView.setText("bind TextView success");
  12.         mButton.setText("bind Button success");
  13.     }
  14. }

總結(jié)

1、APT技術(shù)其實(shí)就是自定義注解和注解處理器,在編譯期間生成Java文件,類似于IOC控制反轉(zhuǎn),可以方便的進(jìn)行解耦;

2、如果你也可以實(shí)現(xiàn)很多不同的項(xiàng)目,比如路由框架等等,后續(xù)也會(huì)寫一些apt的項(xiàng)目


本文名稱:注解APT應(yīng)用詳解(手把手教你寫B(tài)utterKnife工具)
地址分享:http://m.5511xx.com/article/dhsgsso.html