首页 > 代码库 > Butterknife案例和原理

Butterknife案例和原理

最近使用了一个注入类型的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();
    }
}


SimpleAdapter.java 

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相关知识。在此不多做讲解。



Butterknife案例和原理