黄油刀

来源:互联网 发布:乐语软件 编辑:程序博客网 时间:2024/04/28 17:42

【攻克Android (34)】Butter Knife 黄油刀

    博客分类: 
  • 攻克Android系列
 
本文围绕以下四个部分展开: 

一、注解式框架 
二、Butter Knife 
案例一 
案例二:用 ListView 展示一个列表数据,每个Item里含有一个Button,可以点击。 






一、注解式框架
 

        1. 注解式开发: 

        JDK 1.5后支持注解方式。想用注解式开发,就要自定义注解。 

        @Override就是一种自带的注解。很多框架都采用注解方式来实现。 

        当追求更高的开发效率,用更简洁的代码、更清晰的代码逻辑来进行高效的开发的时候,使用注解式框架开发可简化代码提升开发效率代码的可读性。 



        2. 主流注解式框架: 

        (1)AndroidAnnotations 

        配置麻烦,需要在功能清单文件中注册生成的子类。反射机制会占用资源内存,且耗时。 

        (2)Dagger 

        采用预编译技术,高效。但是对View绑定操作注解不是很方便。 

        (3)Butter Knife 

        它是View 及事件等依赖注入框架。使用方便,配置简单,有强大的View注入绑定和简单的常用方法注解。 


二、Butter Knife 

        它是View 及事件等依赖注入框架。 

        1. 特点: 

        (1)强大的View注入绑定和Click事件的处理。简化代码,提升开发效率。 

        (2)可以方便地处理ListView的Adapter里的ViewHolder绑定问题。 

        (3)运行时不会影响App效率,使用配置方便。 

        (4)代码思路清晰,可读性强。 

        2. 用法: 

        导包: 

        compile 'com.jakewharton:butterknife:6.0.0' 

        在当前 Activity(this)的onCreate中注册(注入黄油刀): 

Java代码  收藏代码
  1. ButterKnife.inject(this);  


        (1)View绑定(绑定控件): 

        Activity声明绑定控件: 

Java代码  收藏代码
  1. @InjectView(R.id.tvTitle)   
  2. TextView tvTitle;  


        黄油刀注入控件,相当于:先声明变量,然后通过 findViewById(R.id.tvCompany) 初始化变量。黄油刀直接帮我们绑定好了控件: 

        InjectView: Bind a field to the view for the specified ID. The view will automatically be cast to the field type. 

Java代码  收藏代码
  1. @InjectView(R.id.tvTitle) // 相当于 通过 findViewById(R.id.tvCompany) 初始化变量。  
  2. TextView tvTitle; // 相当于 声明变量:private TextView tvCompany;  


        注意:不同写为:private TextView tvTitle;  会报错:@InjectView fields must not be private or static. 

        (2)Onclick等事件处理: 

Java代码  收藏代码
  1. @OnClick(R.id.btnHello)  
  2. public void sayHi(){  
  3.     Toast.makeText(this,"你好!",Toast.LENGTH_SHORT).show();  
  4. }  


        (3)ListView的Adapter里的ViewHolder绑定问题。 


案例一 














        1. strings.xml。字符串。 

Xml代码  收藏代码
  1. <resources>  
  2.     <string name="app_name">ButterKnifeDemo</string>  
  3.   
  4.     <string name="hello_world">Hello world!</string>  
  5.     <string name="action_settings">Settings</string>  
  6.   
  7.     <string name="btn_good_night">Good Night!</string>  
  8.     <string name="btn_play">播放</string>  
  9.     <string name="btn_stop">停止</string>  
  10.     <string name="btn_view_holder">ViewHolder</string>  
  11. </resources>  


        2. activity_main.xml。布局 

Xml代码  收藏代码
  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  2.     xmlns:tools="http://schemas.android.com/tools"  
  3.     android:layout_width="match_parent"  
  4.     android:layout_height="match_parent"  
  5.     android:paddingBottom="@dimen/activity_vertical_margin"  
  6.     android:paddingLeft="@dimen/activity_horizontal_margin"  
  7.     android:paddingRight="@dimen/activity_horizontal_margin"  
  8.     android:paddingTop="@dimen/activity_vertical_margin"  
  9.     tools:context=".MainActivity">  
  10.   
  11.     <Button  
  12.         android:id="@+id/btnHello"  
  13.         android:layout_width="wrap_content"  
  14.         android:layout_height="wrap_content"  
  15.         android:text="@string/hello_world"/>  
  16.   
  17.     <Button  
  18.         android:id="@+id/btnGoodNight"  
  19.         android:layout_width="wrap_content"  
  20.         android:layout_height="wrap_content"  
  21.         android:layout_below="@id/btnHello"  
  22.         android:text="@string/btn_good_night"/>  
  23.   
  24.     <Button  
  25.         android:id="@+id/btnPlay"  
  26.         android:layout_width="wrap_content"  
  27.         android:layout_height="wrap_content"  
  28.         android:layout_below="@id/btnGoodNight"  
  29.         android:text="@string/btn_play"/>  
  30.   
  31.     <Button  
  32.         android:id="@+id/btnStop"  
  33.         android:layout_width="wrap_content"  
  34.         android:layout_height="wrap_content"  
  35.         android:layout_below="@id/btnPlay"  
  36.         android:text="@string/btn_stop"/>  
  37.   
  38.     <Button  
  39.         android:id="@+id/btnViewHolder"  
  40.         android:layout_width="wrap_content"  
  41.         android:layout_height="wrap_content"  
  42.         android:layout_below="@id/btnStop"  
  43.         android:text="@string/btn_view_holder"/>  
  44.   
  45. </RelativeLayout>  


        3. MainActivity。View绑定和Onclick事件处理 

Java代码  收藏代码
  1. package com.android.butterknifedemo;  
  2.   
  3. import android.app.Activity;  
  4. import android.os.Bundle;  
  5. import android.view.Menu;  
  6. import android.view.MenuItem;  
  7. import android.view.View;  
  8. import android.widget.Button;  
  9. import android.widget.Toast;  
  10.   
  11. import butterknife.ButterKnife;  
  12. import butterknife.InjectView;  
  13. import butterknife.OnClick;  
  14. import butterknife.Optional;  
  15.   
  16.   
  17. public class MainActivity extends Activity {  
  18.   
  19.     // 2. Actitvity里声明绑定控件  
  20.     // @Optional:加入后,若此id不存在,则程序不会崩溃,而是抛出异常  
  21.     @Optional@InjectView(R.id.btnHello)  
  22.     Button buttonHello;  
  23.     @InjectView(R.id.btnPlay)  
  24.     Button buttonPlay;  
  25.     @InjectView(R.id.btnStop)  
  26.     Button buttonStop;  
  27.   
  28.     @Override  
  29.     protected void onCreate(Bundle savedInstanceState) {  
  30.         super.onCreate(savedInstanceState);  
  31.         setContentView(R.layout.activity_main);  
  32.   
  33.         // 1. onCreate() 方法里注册 ButterKnife  
  34.         ButterKnife.inject(this);  
  35.     }  
  36.   
  37.     // 3. 注入 View 事件,不带参数  
  38.     @OnClick(R.id.btnHello)  
  39.     public void sayHi(){  
  40.         Toast.makeText(this,"你好!",Toast.LENGTH_SHORT).show();  
  41.     }  
  42.   
  43.     // 4. 注入 View 事件,带参数  
  44.     @OnClick(R.id.btnGoodNight)  
  45.     public void sayGoodNight(Button btnGoodNight){  
  46.         btnGoodNight.setText("晚上好!");  
  47.     }  
  48.   
  49.     // 5. 同时注入多个 View 事件  
  50.     @OnClick({R.id.btnPlay,R.id.btnStop})  
  51.     public void playMusic(View view){  
  52.         switch (view.getId()){  
  53.             case R.id.btnPlay:  
  54.                 Toast.makeText(this,"播放音乐",Toast.LENGTH_SHORT).show();  
  55.                 break;  
  56.             case R.id.btnStop:  
  57.                 Toast.makeText(this,"停止播放",Toast.LENGTH_SHORT).show();  
  58.         }  
  59.     }  
  60.   
  61.   
  62.     // ----------------------------------------------------------  
  63.     @Override  
  64.     public boolean onCreateOptionsMenu(Menu menu) {  
  65.         // Inflate the menu; this adds items to the action bar if it is present.  
  66.         getMenuInflater().inflate(R.menu.menu_main, menu);  
  67.         return true;  
  68.     }  
  69.   
  70.     @Override  
  71.     public boolean onOptionsItemSelected(MenuItem item) {  
  72.         // Handle action bar item clicks here. The action bar will  
  73.         // automatically handle clicks on the Home/Up button, so long  
  74.         // as you specify a parent activity in AndroidManifest.xml.  
  75.         int id = item.getItemId();  
  76.   
  77.         //noinspection SimplifiableIfStatement  
  78.         if (id == R.id.action_settings) {  
  79.             return true;  
  80.         }  
  81.   
  82.         return super.onOptionsItemSelected(item);  
  83.     }  
  84. }  


        4. MyAdapter。处理ListView的Adapter里的ViewHolder绑定问题。 

Java代码  收藏代码
  1. package com.android.butterknifedemo;  
  2.   
  3. import android.content.Context;  
  4. import android.view.LayoutInflater;  
  5. import android.view.View;  
  6. import android.view.ViewGroup;  
  7. import android.widget.BaseAdapter;  
  8. import android.widget.Button;  
  9. import android.widget.TextView;  
  10.   
  11. import butterknife.ButterKnife;  
  12. import butterknife.InjectView;  
  13.   
  14. /** 
  15.  * 在 ViewHolder 模式中注入 
  16.  */  
  17. // 1. 继承自 BaseAdapter  
  18. public class MyAdapter extends BaseAdapter{  
  19.     // 4. 定义上下文  
  20.     private Context mContext;  
  21.   
  22.     /** 
  23.      * 5. 构造方法 
  24.      * @param context 
  25.      */  
  26.     public MyAdapter(Context context) {  
  27.         // 上下文通过构造方法传过来  
  28.         mContext = context;  
  29.     }  
  30.   
  31.     @Override  
  32.     public int getCount() {  
  33.         return 0;  
  34.     }  
  35.   
  36.     @Override  
  37.     public Object getItem(int position) {  
  38.         return position;  
  39.     }  
  40.   
  41.     @Override  
  42.     public long getItemId(int position) {  
  43.         return 0;  
  44.     }  
  45.   
  46.     @Override  
  47.     public View getView(int position, View convertView, ViewGroup parent) {  
  48.         // 3. 声明 holder 为空  
  49.         ViewHolder holder = null;  
  50.   
  51.         // 6.  
  52.         if(holder==null){  
  53.             // 重新载入布局  
  54.             convertView = LayoutInflater.from(mContext).inflate(  
  55.                     R.layout.activity_main,parent,false);  
  56.             // 对 holder 进行实例化  
  57.             holder = new ViewHolder(convertView);  
  58.             // 获得按钮控件  
  59.             holder.btnViewHolder = (Button) convertView.findViewById(R.id.btnViewHolder);  
  60.             // 设置标签  
  61.             convertView.setTag(holder);  
  62.         } else {  
  63.             holder = (ViewHolder) convertView.getTag();  
  64.         }  
  65.         // 7.  
  66.         holder.btnViewHolder.setText("更改");  
  67.         return convertView;  
  68.     }  
  69.   
  70.     // 2. 写一个 结构持有者 类  
  71.     static class ViewHolder{  
  72.         @InjectView(R.id.btnViewHolder)  
  73.         Button btnViewHolder;  
  74.   
  75.         public ViewHolder(View view){  
  76.             ButterKnife.inject(this,view);  
  77.         }  
  78.     }  
  79. }  



案例二:用 ListView 展示一个列表数据,每个Item里含有一个Button,可以点击。 










        1. activity_main.xml。布局:一个ListView。 

Xml代码  收藏代码
  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  2.     xmlns:tools="http://schemas.android.com/tools"  
  3.     android:layout_width="match_parent"  
  4.     android:layout_height="match_parent"  
  5.     android:paddingBottom="@dimen/activity_vertical_margin"  
  6.     android:paddingLeft="@dimen/activity_horizontal_margin"  
  7.     android:paddingRight="@dimen/activity_horizontal_margin"  
  8.     android:paddingTop="@dimen/activity_vertical_margin"  
  9.     tools:context=".MainActivity">  
  10.   
  11.     <ListView  
  12.         android:id="@+id/listView"  
  13.         android:layout_width="match_parent"  
  14.         android:layout_height="match_parent"/>  
  15.   
  16. </RelativeLayout>  


        2. item.xml。每一个ListView的项。包括一个文本和一个按钮 

Xml代码  收藏代码
  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  3.     android:layout_width="match_parent"  
  4.     android:layout_height="match_parent"  
  5.     android:descendantFocusability="blocksDescendants"  
  6.     android:orientation="vertical">  
  7.   
  8.     <TextView  
  9.         android:id="@+id/tvText"  
  10.         android:layout_width="wrap_content"  
  11.         android:layout_height="wrap_content"  
  12.         android:text="@string/app_name" />  
  13.   
  14.     <Button  
  15.         android:id="@+id/btn"  
  16.         android:layout_width="wrap_content"  
  17.         android:layout_height="wrap_content"  
  18.         android:layout_toRightOf="@id/tvText" />  
  19.   
  20. </RelativeLayout>  


        注:ListView中出现button等可以拦截点击事件的控件的时候,要加属性: 

Xml代码  收藏代码
  1. android:descendantFocusability="blocksDesendants"  


        3. MyAdapter。自定义适配器。 

Java代码  收藏代码
  1. package com.android.bufferknifedemo;  
  2.   
  3. import android.content.Context;  
  4. import android.view.View;  
  5. import android.view.ViewGroup;  
  6. import android.widget.BaseAdapter;  
  7. import android.widget.Button;  
  8. import android.widget.TextView;  
  9. import android.widget.Toast;  
  10.   
  11. import java.util.ArrayList;  
  12.   
  13. import butterknife.ButterKnife;  
  14. import butterknife.InjectView;  
  15.   
  16. /** 
  17.  * Created by Xiangdong on 2015/7/12. 
  18.  */  
  19. public class MyAdapter extends BaseAdapter {  
  20.     Context mContext;  
  21.     private ArrayList<String> lists;  
  22.   
  23.     public MyAdapter(Context context, ArrayList<String> list) {  
  24.         mContext = context;  
  25.         lists = list;  
  26.     }  
  27.   
  28.     @Override  
  29.     public int getCount() {  
  30.         return lists.size();  
  31.     }  
  32.   
  33.     @Override  
  34.     public Object getItem(int position) {  
  35.         return lists.get(position);  
  36.     }  
  37.   
  38.     @Override  
  39.     public long getItemId(int position) {  
  40.         return position;  
  41.     }  
  42.   
  43.     @Override  
  44.     public View getView(int position, View convertView, ViewGroup parent) {  
  45.         ViewHolder holder = null;  
  46.         if (convertView == null) {  
  47.             convertView = View.inflate(mContext, R.layout.item, null);  
  48.             holder = new ViewHolder(convertView);  
  49.             convertView.setTag(holder);  
  50.         } else {  
  51.             holder = (ViewHolder) convertView.getTag();  
  52.         }  
  53.   
  54.         String s = lists.get(position);  
  55.         holder.tvText.setText(s);  
  56.         holder.btn.setOnClickListener(new View.OnClickListener() {  
  57.             @Override  
  58.             public void onClick(View v) {  
  59.                 Toast.makeText(mContext, "点击", Toast.LENGTH_SHORT).show();  
  60.             }  
  61.         });  
  62.         return convertView;  
  63.     }  
  64.   
  65.     static class ViewHolder {  
  66.         // 绑定控件  
  67.         @InjectView(R.id.tvText)  
  68.         TextView tvText;  
  69.         @InjectView(R.id.btn)  
  70.         Button btn;  
  71.   
  72.         public ViewHolder(View view) {  
  73.             ButterKnife.inject(this, view);  
  74.         }  
  75.     }  
  76. }  


        4. MainActivity。 

Java代码  收藏代码
  1. package com.android.bufferknifedemo;  
  2.   
  3. import android.app.Activity;  
  4. import android.os.Bundle;  
  5. import android.view.Menu;  
  6. import android.view.MenuItem;  
  7. import android.widget.ListView;  
  8. import android.widget.Toast;  
  9.   
  10. import java.util.ArrayList;  
  11.   
  12. import butterknife.ButterKnife;  
  13. import butterknife.InjectView;  
  14. import butterknife.OnItemClick;  
  15.   
  16. public class MainActivity extends Activity {  
  17.     @InjectView(R.id.listView)  
  18.     ListView listView;  
  19.     private MyAdapter adapter;  
  20.     private ArrayList<String> list;  
  21.   
  22.     @Override  
  23.     protected void onCreate(Bundle savedInstanceState) {  
  24.         super.onCreate(savedInstanceState);  
  25.         setContentView(R.layout.activity_main);  
  26.   
  27.         ButterKnife.inject(this);  
  28.   
  29.         list = new ArrayList<>();  
  30.         list.add("测试1");  
  31.         list.add("测试2");  
  32.         list.add("测试3");  
  33.         list.add("测试4");  
  34.         list.add("测试5");  
  35.         list.add("测试6");  
  36.         list.add("测试7");  
  37.         list.add("测试8");  
  38.         list.add("测试9");  
  39.         adapter = new MyAdapter(this, list);  
  40.         listView.setAdapter(adapter);  
  41.     }  
  42.   
  43.     @OnItemClick(R.id.listView)  
  44.     public void onMyItemClick(int position) {  
  45.         Toast.makeText(this"位置:" + position, Toast.LENGTH_SHORT).show();  
  46.     }  
  47.   
  48.     // ------------------------------------------------------------------  
  49.     @Override  
  50.     public boolean onCreateOptionsMenu(Menu menu) {  
  51.         // Inflate the menu; this adds items to the action bar if it is present.  
  52.         getMenuInflater().inflate(R.menu.menu_main, menu);  
  53.         return true;  
  54.     }  
  55.   
  56.     @Override  
  57.     public boolean onOptionsItemSelected(MenuItem item) {  
  58.         // Handle action bar item clicks here. The action bar will  
  59.         // automatically handle clicks on the Home/Up button, so long  
  60.         // as you specify a parent activity in AndroidManifest.xml.  
  61.         int id = item.getItemId();  
  62.   
  63.         //noinspection SimplifiableIfStatement  
  64.         if (id == R.id.action_settings) {  
  65.             return true;  
  66.         }  
  67.   
  68.         return super.onOptionsItemSelected(item);  
  69.     }  
  70. }  
0 0