Android 使用自定义注解代替重复写findViewById代码

效果

每次新建页面控件的findViewById是每个android开发者的痛苦。在这方面已经有很多第三方框架帮我们解放了双手,这次就是利用注解来解决findViewById。

public class ObserverActivity extends AppCompatActivity{
    @ViewInject(R.id.activity_btn)
    Button button;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_observer);
        AnnotateUtils.injectViews(this);
        button.setText("newText");
    }

}

一、实现原理

在Java中,通过反射,我们可以获取每一个类的详细信息,比如有什么属性字段,有什么方法,类名等,我们通过注解和反射配合,使用反射调用类中的属性,然后读取注解的参数来进行属性的赋值。简单的说,就是其实我们还是会调用findViewById这个方法,但是,这个方法可以放到工具类中执行了,我们只需要像上面那样给出参数就行了。这个形式就和注解框架ButterKnife一样

二、创建自定义注解

新建一个注解。

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface ViewInject {
    int value();
}

其中@Target的意思是我们注解的目标,这里是ElementType.FIELD,也就是作用于属性的。
它的类型有以下几种:
- 1.CONSTRUCTOR:用于描述构造器
- 2.FIELD:用于描述字段
- 3.LOCAL_VARIABLE:用于描述局部变量
- 4.METHOD:用于描述方法
- 5.PACKAGE:用于描述包
- 6.PARAMETER:用于描述参数
- 7.TYPE:用于描述类、接口(包括注解类型) 或enum声明

然后@Retention的意思是注解的运行级别
它的类型有以下几种
- 1.SOURCE:在源文件中有效(即源文件保留)
- 2.CLASS:在class文件中有效(即class保留)
- 3.RUNTIME:在运行时有效(即运行时保留)

三、注入注解的工具类

public class AnnotateUtils {
    public static void injectViews(Activity activity) {
        // 获取activity的Class
        Class<? extends Activity> object = activity.getClass();
        // 通过Class获取activity的所有属性
        Field[] fields = object.getDeclaredFields();
        for (Field field : fields) {
            // 获取字段的注解,如果没有ViewInject注解,则返回null
            ViewInject viewInject = field.getAnnotation(ViewInject.class);
            if (viewInject != null) {
                // 获取属性的注解的参数,这就是控件的id
                int viewId = viewInject.value();
                try {
                    // 获取类中的findViewById方法,参数为int
                    Method method = object.getMethod("findViewById", int.class);
                    // 执行该方法,返回一个Object类型的View实例
                    Object resView = method.invoke(activity, viewId);
                    field.setAccessible(true);
                    // 把属性的值设置为该View的实例
                    field.set(activity, resView);
                } catch (NoSuchMethodException e) {
                    e.printStackTrace();
                } catch (IllegalAccessException e) {
                    e.printStackTrace();
                } catch (InvocationTargetException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

主要思路:
获取activity的所有的Filed数组,然后遍历它们,检验有自定义注解ViewInfect的属性。
然后获取注解的参数Id,通过反射调用findViewById来获取指定View给该属性赋值。

这样就可以做到ButterKnife一样的效果了。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,以下是一个简单的使用自定义适配器实现列表的Android代码示例: 1. 首先,我们需要创建一个布局文件,用来定义列表项的显示样式。比如,我们可以创建一个名为list_item.xml的布局文件,定义一个简单的TextView用于显示列表项的内容: ```xml <TextView xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/textView" android:layout_width="match_parent" android:layout_height="wrap_content" android:padding="10dp" android:textSize="16sp" /> ``` 2. 接下来,我们需要创建一个自定义适配器类,用于将数据源中的数据绑定到列表项上。比如,我们可以创建一个名为MyAdapter的适配器类,继承自BaseAdapter类,并实现其中的几个方法: ```java public class MyAdapter extends BaseAdapter { private Context mContext; private String[] mData; public MyAdapter(Context context, String[] data) { mContext = context; mData = data; } @Override public int getCount() { return mData.length; } @Override public Object getItem(int position) { return mData[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(mContext).inflate(R.layout.list_item, parent, false); holder = new ViewHolder(); holder.textView = convertView.findViewById(R.id.textView); convertView.setTag(holder); } else { holder = (ViewHolder) convertView.getTag(); } holder.textView.setText(mData[position]); return convertView; } private static class ViewHolder { TextView textView; } } ``` 以上代码中,我们定义了一个MyAdapter类,并在其中实现了BaseAdapter类中的几个重要方法。其中,getCount()方法用于获取数据源中的数据数量,getItem()方法用于获取数据源中指定位置的数据项,getItemId()方法用于获取指定位置的数据项的ID。最重要的是getView()方法,它用于将数据源中的数据绑定到列表项上,并返回一个View对象。 在getView()方法中,我们首先判断convertView是否为空。如果为空,说明当前列表项还没有被创建过,我们就需要通过LayoutInflater从布局文件中加载一个新的View对象,并将其缓存起来。这样,在下次需要显示同样的列表项时,就可以直接从缓存中取出View对象,而不需要重新创建。如果convertView不为空,说明当前列表项已经被创建过,我们可以直接从缓存中取出它的View对象。 接着,我们通过ViewHolder模式,将列表项中的TextView控件缓存起来,避免在每次getView()方法被调用时都需要通过findViewById()方法来查找它。最后,我们将数据源中当前位置的数据绑定到TextView控件上,并返回这个View对象。 3. 最后,我们在Activity中使用自定义适配器,并将它设置到ListView中: ```java ListView listView = findViewById(R.id.listView); String[] data = {"Item 1", "Item 2", "Item 3", "Item 4", "Item 5"}; MyAdapter adapter = new MyAdapter(this, data); listView.setAdapter(adapter); ``` 以上代码中,我们创建了一个MyAdapter适配器对象,并将其设置到ListView中。此时,ListView就会根据适配器中的数据源,创建了对应数量的列表项,并使用自定义的布局文件来显示每个列表项的内容。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值