Android依賴注入之BufferKnife 8.0註解使用
前言:
App專案開發大部分時候還是以UI頁面為主,這時我們需要呼叫大量的findViewById以及setOnClickListener等程式碼,控制元件的少的時候我們還能接受,控制元件多起來有時候就會有一種想砸鍵盤的衝動。所以這個時候我們想著可以藉助註解的方式讓我們從這種繁重的工作中脫離出來,也讓程式碼變得更加簡潔,便於維護,今天主要學習一下只專注View、Resource、Action註解框架BufferKnife 。
BufferKnife 介紹
BufferKnife 是一個專注於Android系統的View、Resource、Action注入框架。
官網:http://jakewharton.github.io/butterknife/
gitHub:https://github.com/JakeWharton/butterknife/
BufferKnife 使用前後對比:
看看沒有使用View註解之前我們是如何做的
1.)使用之前
public class ExampleActivity extends AppCompatActivity { private final static String TAG = ExampleActivity.class.getSimpleName(); String butterKnifeStr; Drawable butterKnifeDrawable; Button butterKnifeBtn; ImageView butterKnifeIv; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_butter_knife); initResource(); initViews(); } private void initViews() { butterKnifeBtn = (Button) findViewById(R.id.btn_butter_knife); butterKnifeBtn.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { Log.e(TAG, "onButterKnifeBtnClick"); } }); butterKnifeIv = (ImageView) findViewById(R.id.iv_butter_knife); butterKnifeBtn.setText(butterKnifeStr); butterKnifeIv.setImageDrawable(butterKnifeDrawable); } private void initResource() { butterKnifeStr = getString(R.string.title_btn_butter_knife); butterKnifeDrawable = getDrawable(R.mipmap.ic_launcher); } }
2.)使用之後
public class ButterKnifeActivity extends AppCompatActivity { private final static String TAG = ButterKnifeActivity.class.getSimpleName(); private Unbinder unbinder; @BindString(R.string.title_btn_butter_knife) String butterKnifeStr; @BindDrawable(R.mipmap.ic_launcher) Drawable butterKnifeDrawable; @BindView(R.id.btn_butter_knife) Button butterKnifeBtn; @BindView(R.id.iv_butter_knife) ImageView butterKnifeIv; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_butter_knife); unbinder = ButterKnife.bind(this); initViews(); } private void initViews() { butterKnifeBtn.setText(butterKnifeStr); butterKnifeIv.setImageDrawable(butterKnifeDrawable); } @OnClick(R.id.btn_butter_knife) public void onButterKnifeBtnClick(View view) { Log.e(TAG, "onButterKnifeBtnClick"); } @Override protected void onDestroy() { super.onDestroy(); unbinder.unbind(); } }
3.)ButterKnife 優勢
通過上面使用前後對比來分析下ButterKnife優勢
- 強大的View繫結和Click事件處理功能,簡化程式碼,提升開發效率
- 方便的處理Adapter裡的ViewHolder繫結問題
- 執行時不會影響APP效率,使用配置方便
- 程式碼清晰,可讀性強
使用前後對比之後有沒有覺得非常的簡單易用。接下來來看下具體怎麼使用的?
BufferKnife 如何使用:
1.)在Project的build.gradle中新增如下配置
buildscript { repositories { mavenCentral() } dependencies { classpath 'com.neenbedankt.gradle.plugins:android-apt:1.8' } }
2.)在Module的build.gradle新增如下配置
apply plugin: 'com.neenbedankt.android-apt' android { ... } dependencies { compile 'com.jakewharton:butterknife:8.1.0' apt 'com.jakewharton:butterknife-compiler:8.1.0' }
3.)注入和重置注入
Activity
class ExampleActivity extends Activity { @BindView(R.id.title) TextView title; @BindView(R.id.subtitle) TextView subtitle; @BindView(R.id.footer) TextView footer; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.simple_activity); ButterKnife.bind(this); // TODO Use fields... } }
Fragment:由於不同的檢視生命週期,所以需要在onCreateView bind,在onDestroyView unbind
public class FancyFragment extends Fragment { @BindView(R.id.button1) Button button1; @BindView(R.id.button2) Button button2; private Unbinder unbinder; @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { View view = inflater.inflate(R.layout.fancy_fragment, container, false); unbinder = ButterKnife.bind(this, view); // TODO Use fields... return view; } @Override public void onDestroyView() { super.onDestroyView(); unbinder.unbind(); } }
ViewHolder
public class MyAdapter extends BaseAdapter { @Override public View getView(int position, View view, ViewGroup parent) { ViewHolder holder; if (view != null) { holder = (ViewHolder) view.getTag(); } else { view = inflater.inflate(R.layout.whatever, parent, false); holder = new ViewHolder(view); view.setTag(holder); } holder.name.setText("John Doe"); // etc... return view; } static class ViewHolder { @BindView(R.id.title) TextView name; @BindView(R.id.job_title) TextView jobTitle; public ViewHolder(View view) { ButterKnife.bind(this, view); } } }
4.)view注入 @BindView,@BindViews
@BindView(R.id.btn_butter_knife) Button butterKnifeBtn; @BindViews({R.id.tv_butter_knife1,R.id.tv_butter_knife2,R.id.tv_butter_knife3}) List<TextView> textViews;
5.)Resource注入
@BindString(R.string.title_btn_butter_knife) String butterKnifeStr;//string註解使用 @BindDrawable(R.mipmap.ic_launcher) Drawable butterKnifeDrawable;//Drawable註解使用 @BindBitmap(R.mipmap.ic_launcher) Bitmap butterKnifeBitmap;;//Bitmap註解使用 @BindArray(R.array.day_of_week) String weeks[];//陣列 @BindColor(R.color.colorPrimary) int colorPrimary;//color註解使用 @BindDimen(R.dimen.activity_horizontal_margin) Float spacer;
6.)單事件注入
一個控制元件指定一個事件回撥
/** * 帶引數 */ @OnClick(R.id.btn_butter_knife) public void onButterKnifeBtnClick() { } /** * 帶引數 */ @OnClick(R.id.btn_butter_knife) public void onButterKnifeBtnClick(View view) { Log.e(TAG, "onButterKnifeBtnClick"); } /** * 帶引數 * @param button */ @OnClick(R.id.btn_butter_knife) public void onButterKnifeBtnClick(Button button) { Log.e(TAG, "onButterKnifeBtnClick"); }
也可以多個控制元件指定一個事件回撥
/** * 兩個不同的button都相應onButterKnifeBtnClick事件回撥 * * @param button */ @OnClick({R.id.btn_butter_knife, R.id.btn_butter_knife1}) public void onButterKnifeBtnClick(Button button) { Log.e(TAG, "onButterKnifeBtnClick"); }
自定義的控制元件不通過ID也可以繫結到自己的事件
public class FancyButton extends Button { @OnClick public void onClick() { // TODO do something! } }
7.)多事件回撥
有一些View的listener是有多個回撥方法的,比如EditText新增addTextChangedListener
editText.addTextChangedListener(new TextWatcher() { @Override public void beforeTextChanged(CharSequence s, int start, int count, int after) { } @Override public void onTextChanged(CharSequence s, int start, int before, int count) { } @Override public void afterTextChanged(Editable s) { } });
可以使用註解方式改成如下
@OnTextChanged(value = R.id.nameEditText, callback = OnTextChanged.Callback.BEFORE_TEXT_CHANGED) void beforeTextChanged(CharSequence s, int start, int count, int after) { } @OnTextChanged(value = R.id.nameEditText, callback = OnTextChanged.Callback.TEXT_CHANGED) void onTextChanged(CharSequence s, int start, int before, int count) { } @OnTextChanged(value = R.id.nameEditText, callback = OnTextChanged.Callback.AFTER_TEXT_CHANGED) void afterTextChanged(Editable s) { }
8.)選擇性注入
預設情況下,@Bind
和listener的注入都是必須的,如果target view沒有被發現,則會報錯. 為了抑制這種行為,可以用@Optional註解來標記field和方法,讓注入變成選擇性的,如果targetView存在,則注入, 不存在,則什麼事情都不做.或者使用 Android’s “support-annotations” library.中的@Nullable來修飾
@Nullable @BindView(R.id.might_not_be_there) TextView mightNotBeThere; @Optional @OnClick(R.id.maybe_missing) void onMaybeMissingClicked() { // TODO ... }
9.)ButterKnife.apply()函式
可以通過ButterKnifeapply()函式對view集合元素或者單個view的Action, Setter和Property進行修改
ButterKnife.apply(nameViews, DISABLE); ButterKnife.apply(nameViews, ENABLED, false); static final ButterKnife.Action<View> DISABLE = new ButterKnife.Action<View>() { @Override public void apply(View view, int index) { view.setEnabled(false); } }; static final ButterKnife.Setter<View, Boolean> ENABLED = new ButterKnife.Setter<View, Boolean>() { @Override public void set(View view, Boolean value, int index) { view.setEnabled(value); } }; ButterKnife.apply(nameViews, View.ALPHA, 0.0f);
10.)ButterKnife.findById()
ButterKnife 也提供了findById函式,通過findById()可以獲取Activity、Dialog、View中的view,並且是泛型型別不需要強轉
View view = LayoutInflater.from(context).inflate(R.layout.thing, null); TextView firstName = ButterKnife.findById(view, R.id.first_name); TextView lastName = ButterKnife.findById(view, R.id.last_name); ImageView photo = ButterKnife.findById(view, R.id.photo);
BufferKnife 自動生成外掛安裝:
在AndroidStudio->File->Settings->Plugins->搜尋Zelezny下載新增就行 ,可以快速生成對應元件的例項物件,不用手動寫。使用時,在要匯入註解的Activity 或 Fragment 或 ViewHolder的layout資原始碼上,右鍵——>Generate——Generate ButterKnife Injections,然後就出現如圖的選擇框。
外掛gitHub地址:https://github.com/avast/android-butterknife-zelezny
上面給了一個使用流程圖,不過流程圖不會針對最新的8.0.1版本的,但是都是差不多的
相關文章
- 聊聊依賴注入註解@Resource和@Autowired依賴注入
- 使用google wire解決依賴注入Go依賴注入
- 依賴注入?依賴注入是如何實現解耦的?依賴注入解耦
- 從Hessian RPC 註解方式看Spring依賴注入RPCSpring依賴注入
- 反射,註解,動態代理,依賴注入控制反轉反射依賴注入
- .NET之預設依賴注入依賴注入
- SpringIOC通過註解實現DI(dependence injection,依賴注入)Spring依賴注入
- Laravel 使用依賴注入呼叫方法Laravel依賴注入
- WinForm依賴注入簡單使用ORM依賴注入
- 「轉」Laravel 依賴注入原理(詳細註釋)Laravel依賴注入
- 一、.Net Core 依賴注入詳解及Autofac使用依賴注入
- angular依賴注入Angular依賴注入
- XUnit 依賴注入依賴注入
- Dependency Injection-依賴注入詳解依賴注入
- ASP.NETCore使用AutoFac依賴注入ASP.NETNetCore依賴注入
- Java開發學習(十二)----基於註解開發依賴注入Java依賴注入
- IceRPC之依賴注入>快樂的RPCRPC依賴注入
- 我看依賴注入依賴注入
- 依賴注入系列教程依賴注入
- Spring 依賴注入 DISpring依賴注入
- 理解 Angular 依賴注入Angular依賴注入
- Abp vNext 依賴注入依賴注入
- Sping-依賴注入依賴注入
- Spring IOC——依賴注入Spring依賴注入
- [譯] 依賴注入?? 哈??依賴注入
- Angular 依賴注入原理Angular依賴注入
- spring 的依賴注入Spring依賴注入
- .Net Core — 依賴注入依賴注入
- SpringDI四種依賴注入方式詳解Spring依賴注入
- Swift中依賴注入的解耦策略Swift依賴注入解耦
- Spring 原始碼分析之 bean 依賴注入原理(注入屬性)Spring原始碼Bean依賴注入
- quarkus依賴注入之八:裝飾器(Decorator)依賴注入
- .NET Core基礎篇之:依賴注入DependencyInjection依賴注入
- 我用段子講.NET之依賴注入(一)依賴注入
- 詳解 Laravel 中的依賴注入和 IoCLaravel依賴注入
- spring 詳細講解(ioc,依賴注入,aop)Spring依賴注入
- 關於依賴注入(typescript)依賴注入TypeScript
- 深入淺出依賴注入依賴注入
- AngularJS 4(六)【依賴注入】AngularJS依賴注入