Butterknife全方位解析
概述
Butterknife是供職于Square公司的JakeWharton大神開發(fā)的開源庫,使用這個庫,在AS中搭配Android ButterKnife Zelezny插件,可以大大提高開發(fā)的效率,從此擺脫繁瑣的findViewById(int id),也不用自己手動@bind(int id) , 直接用插件生成即可。本篇博客將對Butterknife進行深入解析。
項目地址: JakeWharton/butterknife
ButterKnife有以下優(yōu)點:
1、強大的View綁定和Click事件處理功能,簡化代碼,提升開發(fā)效率
2、方便的處理Adapter里的ViewHolder綁定問題
3、運行時不會影響APP效率,使用配置方便
4、代碼清晰,可讀性強
如何導(dǎo)入ButterKnife
在項目的build.grade文件中進行如下配置:
- buildscript {
- repositories {
- jcenter()
- mavenCentral()
- maven {
- url "https://plugins.gradle.org/m2/"
- }
- }
- dependencies {
- classpath 'com.android.tools.build:gradle:2.2.0'
- //這里配置 apt 供butterknife使用
- classpath 'com.neenbedankt.gradle.plugins:android-apt:1.8'
- }
- }
例如:
- buildscript {
- repositories {
- jcenter()
- mavenCentral()
- maven {
- url "https://plugins.gradle.org/m2/"
- }
- }
- dependencies {
- classpath 'com.android.tools.build:gradle:2.2.2'
- classpath 'com.neenbedankt.gradle.plugins:android-apt:1.8'
- }
- }
- allprojects {
- repositories {
- jcenter()
- }
- }
- task clean(type: Delete) {
- delete rootProject.buildDir
- }
在app的build.grade文件中進行如下配置:
- apply plugin: 'com.android.application'
- apply plugin: 'com.neenbedankt.android-apt'
- android{...}
- dependencies {
- //視圖綁定 butterknife
- compile 'com.jakewharton:butterknife:8.4.0'
- apt 'com.jakewharton:butterknife-compiler:8.4.0'
- }
例如:
- apply plugin: 'com.android.application'
- apply plugin: 'android-apt'
- android {
- compileSdkVersion 24
- buildToolsVersion "24.0.3"
- defaultConfig {
- minSdkVersion 14
- targetSdkVersion 24
- versionCode 1
- versionName "1.0"
- }
- buildTypes {
- release {
- minifyEnabled false
- proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
- }
- }
- }
- dependencies {
- compile fileTree(dir: 'libs', include: ['*.jar'])
- compile 'com.jakewharton:butterknife:8.4.0'
- apt 'com.jakewharton:butterknife-compiler:8.4.0'
- }
如何使用ButterKnife
1) 由于每次都要在Activity中的onCreate綁定Activity,所以個人建議寫一個BaseActivity完成綁定,子類繼承即可
注:ButterKnife.bind(this);綁定Activity 必須在setContentView之后:
實現(xiàn)如下(FragmentActivity 實現(xiàn)一樣):
- public abstract class BaseActivity extends Activity {
- public abstract int getContentViewId();
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(getContentViewId());
- ButterKnife.bind(this);
- initAllMembersView(savedInstanceState);
- }
- protected abstract void initAllMembersView(Bundle savedInstanceState);
- @Override
- protected void onDestroy() {
- super.onDestroy();
- ButterKnife.unbind(this);//解除綁定,官方文檔只對fragment做了解綁
- }
- }
2) 綁定fragment
- public abstract class BaseFragment extends Fragment {
- public abstract int getContentViewId();
- protected Context context;
- protected View mRootView;
- @Nullable
- @Override
- public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
- mRootView =inflater.inflate(getContentViewId(),container,false);
- ButterKnife.bind(this,mRootView);//綁定framgent
- this.context = getActivity();
- initAllMembersView(savedInstanceState);
- return mRootView;
- }
- protected abstract void initAllMembersView(Bundle savedInstanceState);
- @Override
- public void onDestroyView() {
- super.onDestroyView();
- ButterKnife.unbind(this);//解綁
- }
- }
3) 控件id 注解: @BindView()
- package com.myl.test;
- import android.support.v7.app.AppCompatActivity;
- import android.os.Bundle;
- import android.widget.Button;
- import butterknife.BindView;
- import butterknife.ButterKnife;
- public class ButterknifeActivity extends AppCompatActivity {
- @BindView( R.id.button1 )
- public Button button1 ;
- // 注意:button 的修飾類型不能是:private 或者 static 。 否則會報錯:錯誤: @BindView fields must not be private or static. (com.myl.test.ButterknifeActivity.button1)
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_butterknife);
- //綁定activity
- ButterKnife.bind( this ) ;
- button1.setText( "I am a button ");
- }
- }
4) 多個控件id 注解: @BindViews()
- package com.myl.test;
- import android.support.v7.app.AppCompatActivity;
- import android.os.Bundle;
- import android.widget.Button;
- import java.util.List;
- import butterknife.BindViews;
- import butterknife.ButterKnife;
- public class Main2Activity extends AppCompatActivity {
- @BindViews({ R.id.button1 , R.id.button2 , R.id.button3 })
- public List<Button> buttonList ;
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_main2);
- ButterKnife.bind(this);
- buttonList.get( 0 ).setText( "hello 1 ");
- buttonList.get( 1 ).setText( "hello 2 ");
- buttonList.get( 2 ).setText( "hello 3 ");
- }
- }
5) @BindString() :綁定string 字符串
- package com.myl.test;
- import android.os.Bundle;
- import android.support.v7.app.AppCompatActivity;
- import android.widget.Button;
- import butterknife.BindString;
- import butterknife.BindView;
- import butterknife.ButterKnife;
- public class ButterknifeActivity extends AppCompatActivity {
- @BindView( R.id.button1 ) //綁定button 控件
- public Button button1 ;
- @BindString( R.string.app_name ) //綁定string 字符串
- String meg;
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_butterknife);
- //綁定activity
- ButterKnife.bind( this ) ;
- button1.setText( meg );
- }
- }
6) @BindArray() : 綁定string里面array數(shù)組
- <resources>
- <string name="app_name">校園助手</string>
- <string-array name="city">
- <item>東莞市</item>
- <item>廣州市</item>
- <item>珠海市</item>
- <item>肇慶市</item>
- <item>深圳市</item>
- </string-array>
- </resources>
- -----------------------------------------------------------------
- package com.myl.test;
- import android.os.Bundle;
- import android.support.v7.app.AppCompatActivity;
- import android.widget.Button;
- import butterknife.BindArray;
- import butterknife.BindView;
- import butterknife.ButterKnife;
- public class ButterknifeActivity extends AppCompatActivity {
- @BindView( R.id.button1 ) //綁定button 控件
- public Button button1 ;
- @BindArray(R.array.city ) //綁定string里面array數(shù)組
- String [] citys ;
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_butterknife);
- //綁定activity
- ButterKnife.bind( this ) ;
- button1.setText( citys[0] );
- }
- }
7) @BindBitmap( ) : 綁定Bitmap 資源
- package com.myl.test;
- import android.graphics.Bitmap;
- import android.os.Bundle;
- import android.support.v7.app.AppCompatActivity;
- import android.widget.ImageView;
- import butterknife.BindBitmap;
- import butterknife.BindView;
- import butterknife.ButterKnife;
- public class ButterknifeActivity extends AppCompatActivity {
- @BindView( R.id.imageView ) //綁定ImageView 控件
- public ImageView imageView ;
- @BindBitmap( R.mipmap.wifi ) //綁定Bitmap 資源
- public Bitmap wifi_bitmap ;
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_butterknife);
- //綁定activity
- ButterKnife.bind( this ) ;
- imageView.setImageBitmap( wifi_bitmap );
- }
- }
8) @BindColor( ) : 綁定一個顏色值
- package com.myl.test;
- import android.os.Bundle;
- import android.support.v7.app.AppCompatActivity;
- import android.widget.Button;
- import butterknife.BindColor;
- import butterknife.BindView;
- import butterknife.ButterKnife;
- public class ButterknifeActivity extends AppCompatActivity {
- @BindView( R.id.button1 ) //綁定一個控件
- public Button button1 ;
- @BindColor( R.color.colorAccent ) int black ; //綁定一個顏色值
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_butterknife);
- //綁定activity
- ButterKnife.bind( this ) ;
- button1.setTextColor( black );
- }
- }
9) Adapter ViewHolder 綁定
- public class TestAdapter extends BaseAdapter {
- private List<String> list;
- private Context context;
- public TestAdapter(Context context, List<String> list) {
- this.list = list;
- this.context = context;
- }
- @Override
- public int getCount() {
- return list==null ? 0 : list.size();
- }
- @Override
- public Object getItem(int position) {
- return list.get(position);
- }
- @Override
- public long getItemId(int position) {
- return position;
- }
- @Override
- public View getView(int position, View convertView, ViewGroup parent) {
- ViewHolder holder;
- if (convertView == null) {
- convertView = LayoutInflater.from(context).inflate(R.layout.layout_list_item, null);
- holder = new ViewHolder(convertView);
- convertView.setTag(holder);
- } else {
- holder = (ViewHolder) convertView.getTag();
- }
- holder.textview.setText("item=====" + position);
- return convertView;
- }
- static class ViewHolder {
- @Bind(R.id.hello_world)
- TextView textview;
- public ViewHolder(View view) {
- ButterKnife.bind(this, view);
- }
- }
- }
10) 點擊事件的綁定:不用聲明view,不用setOnClickLisener()就可以綁定點擊事件
a. 直接綁定一個方法
- @OnClick(R.id.submit)
- public void submit(View view) {
- // TODO submit data to server...
- }
b. 所有監(jiān)聽方法的參數(shù)是可選的
- @OnClick(R.id.submit)
- public void submit() {
- // TODO submit data to server...
- }
c. 定義一個特定類型,它將自動被轉(zhuǎn)換
- @OnClick(R.id.submit)
- public void sayHi(Button button) {
- button.setText("Hello!");
- }
d. 多個view統(tǒng)一處理同一個點擊事件,很方便,避免抽方法重復(fù)調(diào)用的麻煩
- @OnClick(R.id.submit)
- public void sayHi(Button button) {
- button.setText("Hello!");
- }
e. 自定義view可以綁定自己的監(jiān)聽,不指定id
- public class FancyButton extends Button {
- @OnClick
- public void onClick() {
- // TODO do something!
- }
- }
f. 給EditText加addTextChangedListener(即添加多回調(diào)方法的監(jiān)聽的使用方法),利用指定回調(diào),實現(xiàn)想回調(diào)的方法即可,哪個注解不會用點進去看下源碼上的注釋
- @OnTextChanged(value = R.id.mobileEditText, callback = OnTextChanged.Callback.BEFORE_TEXT_CHANGED)
- void beforeTextChanged(CharSequence s, int start, int count, int after) {
- }
- @OnTextChanged(value = R.id.mobileEditText, callback = OnTextChanged.Callback.TEXT_CHANGED)
- void onTextChanged(CharSequence s, int start, int before, int count) {
- }
- @OnTextChanged(value = R.id.mobileEditText, callback = OnTextChanged.Callback.AFTER_TEXT_CHANGED)
- void afterTextChanged(Editable s) {
- }
代碼混淆
- -keep class butterknife.** { *; }
- -dontwarn butterknife.internal.**
- -keep class **$$ViewBinder { *; }
- -keepclasseswithmembernames class * {
- @butterknife.* <fields>;
- }
- -keepclasseswithmembernames class * {
- @butterknife.* <methods>;
- }
Zelezny插件的使用
在AndroidStudio->File->Settings->Plugins->搜索Zelezny下載添加就行 ,可以快速生成對應(yīng)組件的實例對象,不用手動寫。使用時,在要導(dǎo)入注解的Activity 或 Fragment 或 ViewHolder的layout資源代碼上,右鍵——>Generate——Generate ButterKnife Injections,然后就出現(xiàn)如圖的選擇框。
ButterKnife實現(xiàn)原理
對ButterKnife有過了解人 , 注入字段的方式是使用注解@BindView(R.id.tv_account_name),但首先我們需要在Activity聲明注入ButterKnife.bind(Activity activity) 。我們知道,注解分為好幾類, 有在源碼生效的注解,有在類文件生成時生效的注解,有在運行時生效的注解。分別為RetentionPolicy.SOURCE,RetentionPolicy.CLASS,RetentionPolicy.RUNTIME ,其中以RetentionPolicy.RUNTIME最為消耗性能。而ButterKnife使用的則是編譯器時期注入,在使用的時候,需要配置classpath ‘com.neenbedankt.gradle.plugins:android-apt:1.8’ , 這個配置說明,在編譯的時候,進行注解處理。要對注解進行處理,則需要繼承AbstractProcessor , 在boolean process(Set
ButterKnife實現(xiàn)方式
知曉了注解可以在編譯的時候進行處理,那么,我們就可以得到注解的字段屬性與所在類 , 進而生成注入文件,生成一個注入類的內(nèi)部類,再進行字段處理 , 編譯之后就會合并到注入類中,達到植入新代碼段的目的。例如:我們注入@VInjector(R.id.tv_show) TextView tvShow;我們就可以得到tvShow這個變量與R.id.tv_show這個id的值,然后進行模式化處理injectObject.tvShow = injectObject.findViewById(R.id.tv_show); ,再將代碼以內(nèi)部類的心事加入到組件所在的類中 , 完成一次DI(注入) 。
a) 首先創(chuàng)建一個視圖注解
b) 創(chuàng)建一個注解處理器,用來得到注解的屬性與所屬類
c) 解析注解,分離組合Class與屬性
d) 組合Class與屬性,生成新的Java File
APT生成的Java File , 以及模式代碼
使用Javac , 編譯時期生成注入類的子類
項目UML圖
簡要說明:
主要類:
VInjectProcessor —-> 注解處理器 , 需要配置注解處理器
- resources
- - META-INF
- - services
- - javax.annotation.processing.Processor
Processor內(nèi)容:
- com.myl.viewinject.apt.VInjectProcessor # 指定處理器全類名
VInjectHandler —-> 注解處理類 , 主要進行注入類與注解字段進行解析與封裝,將同類的字段使用map集合進行映射。exp: Map
自定義ButterKnife具體實現(xiàn)
因微信字?jǐn)?shù)限制,請點擊左下角原文鏈接查看!~