android 反射

本文详细介绍了Java反射机制,包括如何获取类型类、获取类型信息、Class类的使用,以及getDeclaredMethod*()和getMethod*()的区别。通过理论知识结合Android代码示例,阐述反射在Android开发中的应用,帮助开发者提升代码的灵活性和可移植性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一。理论知识

    所谓反射,可以理解为在运行时期获取对象类型信息的操作。传统的编程方法要求程序员在编译阶段决定使用的类型,但是在反射的帮助下,编程人员可以动态获取这些信息,从而编写更加具有可移植性的代码。严格地说,反射并非编程语言的特性,因为在任何一种语言都可以实现反射机制,但是如果编程语言本身支持反射,那么反射的实现就会方便很多。

1,获得类型类

    我们知道在Java中一切都是对象,我们一般所使用的对象都直接或间接继承自Object类。Object类中包含一个方法名叫getClass,利用这个方法就可以获得一个实例的类型类。类型类指的是代表一个类型的类,因为一切皆是对象,类型也不例外,在Java使用类型类来表示一个类型。所有的类型类都是Class类的实例。例如,有如下一段代码:

A a = new A();

if(a.getClass()==A.class)

System.out.println("equal");

else System.out.println("unequal");

结果就是打印出 “equal”。

    可以看到,对象a是A的一个实例,A某一个类,在if语句中使用a.getClass()返回的结果正是A的类型类,在Java中表示一个特定类型的类型类可以用“类型.class”的方式获得,因为a.getClass()获得是A的类型类,也就是A.class,因此上面的代码执行的结果就是打印出 “equal”。特别注意的是,类型类是一一对应的,父类的类型类和子类的类型类是不同的,因此,假设A是B的子类,那么如下的代码将得到 “unequal”的输出:

A a = new A();

if(a.getClass()==B.class)

System.out.println("equal");

else System.out.println("unequal");

因此,如果你知道一个实例,那么你可以通过实例的“getClass()”方法获得该对象的类型类,如果你知道一个类型,那么你可以使用“.class”的方法获得该类型的类型类。

2,获得类型的信息

在获得类型类之后,你就可以调用其中的一些方法获得类型的信息了,主要的方法有:

getName():String:获得该类型的全称名称。

getSuperClass():Class:获得该类型的直接父类,如果该类型没有直接父类,那么返回null。

getInterfaces():Class[]:获得该类型实现的所有接口。

isArray():boolean:判断该类型是否是数组。

isEnum():boolean:判断该类型是否是枚举类型。

isInterface():boolean:判断该类型是否是接口。

isPrimitive():boolean:判断该类型是否是基本类型,即是否是int,boolean,double等等。

isAssignableFrom(Class cls):boolean:判断这个类型是否是类型cls的父(祖先)类或父(祖先)接口。

getComponentType():Class:如果该类型是一个数组,那么返回该数组的组件类型。

此外还可以进行类型转换这类的操作,主要方法有:

asSubclass(Class clazz):Class:将这个类型

3,Class

  首先向大家说明一点,Class本身就是一个类,Class是该类的名称。看下面这个类的定义:   注意到上面的class的首字母是小写,它表示的是一种类类型,但是我们的Class是一个类,相当于上面定义的MyButton类。所以,千万不要把这里的Class做为一个类类型来理解。明白这一点,我们继续。

  Class类是整个Java反射机制的源头,Class类本身表示Java对象的类型,我们可通过一个Object对象的getClass()方法取得一个对象的类型,此函数返回的就是一个Class类。获取Class对象的方法有很多种:

Android反射机制实现与原理 - Nelson - Nelson

  在平时的使用,要注意对这几种方法的灵活运用,尤其是对Class.forName()方法的使用。因为在很多开发中,会直接通过类的名称取得Class类的对象。

4,getDeclaredMethod*()和getMethod*()的区别

getDeclaredMethod*()获取的是类自身声明的所有方法,包含public、protected和private方法。

getMethod*()获取的是类的所有共有方法,这就包括自身的所有public方法,和从基类继承的、从接口实现的所有public方法。

二。代码例子

1,ReflectStorageManager.java

public class ReflectStorageManager {
	private StorageManager storageManager = null;
	private Context context = null;
	// 要反射类的类型类
	private Class classType = null;
	// 调用方法
	private Method doMethod = null;
	public ReflectStorageManager(Context context) {
		// TODO Auto-generated constructor stub
		this.context = context;
		// 实例化,storageManager
		storageManager = (StorageManager) context.getSystemService("storage");
		// storageManager = (StorageManager) context
		// .getSystemService(STORAGE_SERVICE);
		this.classType = storageManager.getClass();

		// 这里也可以这样用伪代码
		// classType = StorageManager.class;//Class.forName();
		// Object object = classType.newInstance();
		// Method method = classType.getDeclaredMethod()...
		// method.invoke(object,)...

	}

	public String getVolumeState(String mountPoint)
			throws NoSuchMethodException, IllegalArgumentException,
			IllegalAccessException, InvocationTargetException {
		// 找到方法,
		doMethod = classType.getDeclaredMethod("getVolumeState",
				new Class[] { String.class });
		// 调用方法,
		mountPoint = (String) doMethod.invoke(storageManager,
				new Object[] { mountPoint });
		return mountPoint;
	}

	public String[] getVolumePaths() throws NoSuchMethodException,
			IllegalArgumentException, IllegalAccessException,
			InvocationTargetException {

		doMethod = classType.getDeclaredMethod("getVolumePaths", null);
		String[] strings = (String[]) doMethod.invoke(storageManager, null);
		return strings;
	}
}

2,MainActivity.java

public class MainActivity extends Activity {

	private Context context = null;
	private String TAG = "MainActivity";

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);
		context = this;

		ReflectStorageManager rsm = new ReflectStorageManager(context);

		String[] strings = null;
		try {
			strings = rsm.getVolumePaths();
		} catch (IllegalArgumentException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (NoSuchMethodException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IllegalAccessException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (InvocationTargetException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}

		for (String str : strings) {
			Log.i(TAG, "path -->" + str);
			try {
				String str1 = rsm.getVolumeState(str);
				Log.i(TAG, "state= -->" + str1);
			} catch (IllegalArgumentException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			} catch (NoSuchMethodException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			} catch (IllegalAccessException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			} catch (InvocationTargetException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}

		}

	}

}

3,仿真打印

shell@android:/ # logcat -s "MainActivity"
--------- beginning of /dev/log/main
--------- beginning of /dev/log/system
I/MainActivity( 3385): path -->/mnt/sdcard
I/MainActivity( 3385): state= -->removed
I/MainActivity( 3385): path -->/mnt/usb/sda1
I/MainActivity( 3385): state= -->mounted
I/MainActivity( 3385): path -->/mnt/usb/sdb1
I/MainActivity( 3385): state= -->mounted


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值