最近使用了一个注入类型的Android框架——butterknife,这种类型的框架和一般使用注解方式不同。
https://github.com/JakeWharton/butterknife
上面是butterknife的github地址,本文讲解的就是里面的案例。
由于我是使用Android studio,在app目录下的build.gradle中添加如下依赖,项目中就可以直接使用butterknife,不需要想eclipse,需要引用jar文件。
dependencies {
compile fileTree(dir: 'libs', include: ['*.jar'])
compile 'com.android.support:appcompat-v7:21.0.2'
compile 'com.jakewharton:butterknife:6.0.0'
}
下面的时案例的代码,代码比较少,就贴下。
SimpleActivity.java
package com.example.butterknife;
public class SimpleActivity extends Activity {
@InjectView(R.id.title)
TextView title;
@InjectView(R.id.subtitle)
TextView subtitle;
@InjectView(R.id.hello)
Button hello;
@InjectView(R.id.list_of_things)
ListView listOfThings;
@InjectView(R.id.footer)
TextView footer;
SimpleAdapter adapter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.simple_activity);
ButterKnife.setDebug(true);
ButterKnife.inject(this);
// Contrived code to use the "injected" views.
title.setText("Butter Knife");
subtitle.setText("View \"injection\" for Android.");
footer.setText("by Jake Wharton");
hello.setText("Say Hello");
adapter = new SimpleAdapter(this);
listOfThings.setAdapter(adapter);
}
// hello的点击事件
@OnClick(R.id.hello)
void sayHello() {
Toast.makeText(this, "Hello, views!", LENGTH_SHORT).show();
}
// hello的长按事件
@OnLongClick(R.id.hello)
boolean sayGetOffMe() {
Toast.makeText(this, "Let go of me!", LENGTH_SHORT).show();
return true;
}
// listview的item点击事件
@OnItemClick(R.id.list_of_things)
void onItemClick(int position) {
Toast.makeText(this, "You clicked: " + adapter.getItem(position), LENGTH_SHORT).show();
}
}
package com.example.butterknife;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.TextView;
import butterknife.ButterKnife;
import butterknife.InjectView;
public class SimpleAdapter extends BaseAdapter {
private static final String[] CONTENTS = "The quick brown fox jumps over the lazy dog".split(" ");
private final LayoutInflater inflater;
public SimpleAdapter(Context context) {
inflater = LayoutInflater.from(context);
}
@Override
public int getCount() {
return CONTENTS.length;
}
@Override
public String getItem(int position) {
return CONTENTS[position];
}
@Override
public long getItemId(int position) {
return position;
}
@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.simple_list_item, parent, false);
holder = new ViewHolder(view);
view.setTag(holder);
}
String word = getItem(position);
holder.word.setText("Word: " + word);
holder.length.setText("Length: " + word.length());
holder.position.setText("Position: " + position);
// Note: don't actually do string concatenation like this in an adapter's getView.
return view;
}
static class ViewHolder {
@InjectView(R.id.word)
TextView word;
@InjectView(R.id.length)
TextView length;
@InjectView(R.id.position)
TextView position;
ViewHolder(View view) {
ButterKnife.inject(this, view);
}
}
}
simple_activity.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
android:padding="8dp">
<TextView
android:id="@+id/title"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:gravity="center"
android:textSize="50sp"
/>
<TextView
android:id="@+id/subtitle"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:gravity="center"
android:textSize="20sp"
/>
<Button
android:id="@+id/hello"
android:layout_width="match_parent"
android:layout_height="0dp"
android:layout_weight="1"
android:layout_margin="10dp"
/>
<ListView
android:id="@+id/list_of_things"
android:layout_width="match_parent"
android:layout_height="0dp"
android:layout_weight="1"
android:layout_margin="10dp"
/>
<TextView
android:id="@+id/footer"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:gravity="center"
android:textSize="17sp"
android:textStyle="italic"
/>
</LinearLayout>
实际的运行效果,当时是可以的。
butterknife的这注解,实际上只是帮助我们,少些部分代码,或者使我们的代码更加简洁。
比如
@InjectView(R.id.hello)
Button hello;
// hello的点击事件
@OnClick(R.id.hello)
void sayHello() {
Toast.makeText(this, "Hello, views!", LENGTH_SHORT).show();
}
可以代替下面的代码
Button hello = (Button) findViewById(R.id.hello);
hello.setOnClickListener(new View.OnClickListener() {
public void onClick(View v) {
Toast.makeText(SimpleActivity.this, "Hello, views!", LENGTH_SHORT).show();
}
});
只不过使用Java 注解的方式,可以使代码更加简洁。
==========================================华丽丽的分割线=====================================================
下面简单介绍下原理,我们以InjectView为例,先看下下面的代码。
package butterknife;
import java.lang.annotation.Retention;
import java.lang.annotation.Target;
import static java.lang.annotation.ElementType.FIELD;
import static java.lang.annotation.RetentionPolicy.CLASS;
/**
* Bind a field to the view for the specified ID. The view will automatically be cast to the field
* type.
* <pre><code>
* {@literal @}InjectView(R.id.title) TextView title;
* </code></pre>
*
* @see Optional
*/
@Retention(CLASS) @Target(FIELD)
public @interface InjectView {
/** View ID to which the field will be bound. */
int value();
}
Inject上的Retention表示这个注解会加载到哪个时期,一共有三种,source,class,runtime,分别表示这个注解会保留到源码级别,class级别(就是java编译生成的哪个class文件),运行时界别(代码在运行的时候,都会有的)。
InjectView是class级别说明,injectview在代码编译的时候做了一些手脚,然后代码运行的时候,就可以自动做这些操作。
我在项目目录下的编译生成的文件夹里发现 了一些情况。
在/butterknife-sample/app/build/intermediates/classes/debug/com/example/butterknife这个目录里面,基本上都是class文件,
但是我发现了2个Java文件(是的,是java文件,竟然是java文件,不是说java文件编译过后是class文件,怎么会有class文件呢?)
SimpleActivity$$ViewInjector.java
SimpleAdapter$ViewHolder$$ViewInjector.java
联想到Android项目可以自动生成R.java文件,所以我们有理由相信这写java文件是由于编译自动生成的(至于怎么生成的,后面再表)。
先看下这个文件里面到底是什么内容。
SimpleActivity$$ViewInjector.java
// Generated code from Butter Knife. Do not modify!
package com.example.butterknife;
import android.view.View;
import butterknife.ButterKnife.Finder;
public class SimpleActivity$$ViewInjector {
public static void inject(Finder finder, final com.example.butterknife.SimpleActivity target, Object source) {
View view;
view = finder.findRequiredView(source, 2131230759, "field 'title'");
target.title = (android.widget.TextView) view;
view = finder.findRequiredView(source, 2131230783, "field 'subtitle'");
target.subtitle = (android.widget.TextView) view;
view = finder.findRequiredView(source, 2131230784, "field 'hello', method 'sayHello', and method 'sayGetOffMe'");
target.hello = (android.widget.Button) view;
view.setOnClickListener(
new butterknife.internal.DebouncingOnClickListener() {
@Override
public void doClick(
android.view.View p0
) {
target.sayHello();
}
});
view.setOnLongClickListener(
new android.view.View.OnLongClickListener() {
@Override
public boolean onLongClick(
android.view.View p0
) {
return target.sayGetOffMe();
}
});
view = finder.findRequiredView(source, 2131230785, "field 'listOfThings' and method 'onItemClick'");
target.listOfThings = (android.widget.ListView) view;
((android.widget.AdapterView<?>) view).setOnItemClickListener(
new android.widget.AdapterView.OnItemClickListener() {
@Override
public void onItemClick(
android.widget.AdapterView<?> p0,
android.view.View p1,
int p2,
long p3
) {
target.onItemClick(p2);
}
});
view = finder.findRequiredView(source, 2131230786, "field 'footer'");
target.footer = (android.widget.TextView) view;
}
public static void reset(com.example.butterknife.SimpleActivity target) {
target.title = null;
target.subtitle = null;
target.hello = null;
target.listOfThings = null;
target.footer = null;
}
}
这个自动生成的代码里面,只有2个方法,一个是inject,一个是reset,顾名思义,一个是进行初始化操作,一个是释放操作。
发现原来findviewByid这些代码是自动生成的,不是程序在运行的时候,查找注解,然后动态执行findViewById操作,同理OnClick这些事件的绑定也是一样。
所以我们有理由相信在oncreate中执行了ButterKnife.inject(this);那么会再主动调用
SimpleActivity$$ViewInjecort.inject();
这个方法。
以上大致是butterknife的简单原理,至于编译器怎么生成的这些代码,可以参考Java Apt相关知识。在此不多做讲解。