背景介绍
在实际开发过程中,有时会遇到需要批量创建控件、批量读取并处理drawable文件,如果是通过写死xml并挨个从drawable中取资源那就太慢太繁琐了,那么有没有想过,针对结构化的drawable资源或者xml中的控件id,可以通过构建字符串数组的方式实现单行循环操作呢?下面介绍两种方法用于批量读取drawable资源文件和xml中的控件。
解决方案
不管是通过findViewById获取xml中的控件、或者是通过R.drawable获取资源文件,在Android 的设计架构中,均是通过R文件进行id(Integer)进行索引查找的,所以基于此,我们只需要获取到控件或者资源的真实id就可以建立引用。
通过Resources来辅助实现
我们采用android.content.res.Resources来实现,因为在Resources类中有一个getIdentifier函数,本质上是通过反射调用实现获取资源的int型的id数值,从而方便后续建立引用。
Resources res=getResources();
return res.getIdentifier(type,"drawable",getPackageName());
getResources方法是来自于contenxt(也就是Activity类),它直接可以返回一个Resouces对象。而Resouces的getIdentifier方法可以返回R.java中的任何资源id,当然,你必须指定3个参数:字段名,类名,包名。
- 包名
指定全限定名的包名部分
(全限定名的定义如下:
类的全限定名 在常量池中, 一个类型的名字并不是我们在源文件中看到的那样, 也不是我们在源文件中使用的包名加类名的形式。 源文件中的全限定名和class文件中的全限定名不是相同的概念。 源文件中的全新定名是包名加类名, 包名的各个部分之间,包名和类名之间, 使用点号分割。 如Object类, 在源文件中的全限定名是java.lang.Object 。 而class文件中的全限定名是将点号替换成“/” 。 例如, Object类在class文件中的全限定名是 java/lang/Object 。 如果读者之前没有接触过class文件格式, 是class文件格式的初学者, 在这里不必知道全限定名在class文件中是如何使用的, 只需要知道, 源文件中一个类的名字, 在class文件中是用全限定名表述的。)
举个栗子:如果R 的全限定名为 android.R 或者 com.company.R,则包名在这里就是“android”或“com.company”。getPackageName其实是this.getPackageName(),它直接返回本类的包名。
- 类名
类名则是资源所属的类。比如我们知道的,在R.java 类中的几个固定的类:drawable、id、string、layout等,在它们下边又定义了许多资源id。
- 字段名
字段名则是资源id的名字。比如 del 这个资源id定义: public static final int del=0x7f020002;
del就是一个资源id的名字,0x7f020002则是它的16进制值。
通过3个参数,getIdentifier 方法就可以通过比较动态的方式获得资源id,获取到这个int型的id之后,我们就可以建立引用,同时,这个字段名就是我们可以通过String数组进行动态组合啦。
操作实例
如下函数可以封装在工具类中供外部直接使用。
/**
我们通过该函数实现对特定的imageview设置为指定名称的drawable资源
**/
public static void setDrawableWithName(Context context, ImageView imageView, String imageName){
Resources resources = context.getResources();
int id = resources.getIdentifier(imageName,"drawable",context.getPackageName());
imageView.setImageResource(id);
}
/**
我们通过该函数实现获取特定资源(包括xml控件、资源等)的id数值
**/
public static int getResourceId(Context context, String name){
Resources resources = context.getResources();
int id = resources.getIdentifier(name,"id",context.getPackageName());
return id;
}
相应地,我们必然需要格式化我们的drawable资源或者xml控件,这样才能实现简洁化,继续举例:
假设我们有这么一个xml文件:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:orientation="vertical">
<LinearLayout
android:orientation="horizontal"
android:layout_width="wrap_content"
android:layout_height="wrap_content">
<ImageView
android:id="@+id/image_porker_1"
android:layout_width="70dp"
android:layout_height="80dp"
android:src="@drawable/ea" />
<ImageView
android:id="@+id/image_porker_2"
android:layout_width="70dp"
android:layout_height="80dp"
android:layout_marginLeft="-30dp"
android:src="@drawable/fa" />
<ImageView
android:id="@+id/image_porker_3"
android:layout_width="70dp"
android:layout_height="80dp"
android:layout_marginLeft="-30dp"
android:src="@drawable/ga" />
<ImageView
android:id="@+id/image_porker_4"
android:layout_width="70dp"
android:layout_height="80dp"
android:layout_marginLeft="-30dp"
android:src="@drawable/ha" />
<ImageView
android:id="@+id/image_porker_5"
android:layout_width="70dp"
android:layout_height="80dp"
android:layout_marginLeft="-30dp"
android:src="@drawable/e2" />
</LinearLayout>
<TextView
android:layout_width="match_parent"
android:layout_height="5dp" />
</LinearLayout>
我们可以看到,xml中若干个imageView是按照格式化的字符串进行id定义的,所以我们就可以采用简洁的方式进行初始化:
ImageView imageViewPorkers[] = new ImageView[NUMBER];
for(int i = 0; i < NUMBER; i++){
imageViewPorkers[i] = findViewById(getResourceId(mContext,"image_porker_" + (i + 1)));
}
按照如下操作即可完成批量初始化,当然,这个例子并不是特别合适,因为门完全可以通过动态去new想要的ImageView控件,只是为了满足习惯于xml进行分层设计或者布局复杂或者想通过include进行布局复用等的童鞋,这也不失为一种选择。
同样地,针对格式化的drawable资源,同样可以采用如上的方式,就不再赘述。
CSDN同步发布地址