java提升(四)——java高级特性--反射

1.定义

JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性;这种动态获取信息以及动态调用对象方法的功能称为java语言的反射机制。

2. 用途

在日常的第三方应用开发过程中,经常会遇到某个类的某个成员变量、方法或是属性是私有的或是只对系统应用开放,这时候就可以利用Java的反射机制通过反射来获取所需的私有成员或是方法。当然,也不是所有的都适合反射,之前就遇到一个案例,通过反射得到的结果与预期不符。阅读源码发现,经过层层调用后在最终返回结果的地方对应用的权限进行了校验,对于没有权限的应用返回值是没有意义的缺省值,否则返回实际值起到保护用户的隐私目的。

3.反射机制的相关类*

与Java反射相关的类如下:

  类名	   用途

Class类	  代表类的实体,在运行的Java应用程序中表示类和接口
Field类	  代表类的成员变量(成员变量也称为类的属性)
Method类	  代表类的方法
Constructor类	   代表类的构造方法

现创建Student类,以下代码演示都以Student为例:

package cn.rjxy.tao18.Test04;

public class Student {

	public int age;
	private String name;

	private Student(int age, String name) {
		super();
		System.out.println("调用了有参的构造方法");
		this.age = age;
		this.name = name;
	}

	public Student() {
		super();
		System.out.println("调用了无参构造方法");
	}

	private Student(String name) {
		this.name = name;
	}

	private void drink(String food) {
		System.out.println(name + "喝" + food);
	}

	public void hello() {
		System.out.println("hello");
	}

	public void eat(String food) {
		System.out.println(name + "吃" + food);
	}

	
	private Integer getAge() {
		return age;
	}

	private void setAge(int age) {
		this.age = age;
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

}


<1>Class类

Class代表类的实体,在运行的Java应用程序中表示类和接口。在这个类中提供了很多有用的方法,这里对他们简单的分类介绍。

<1、获得类相关的方法


方法						用途
asSubclass(Class<U> clazz)	把传递的类的对象转换成代表其子类的对象
Cast	把对象转换成代表类或是接口的对象
getClassLoader()	获得类的加载器
getClasses()	返回一个数组,数组中包含该类中所有公共类和接口类的对象
getDeclaredClasses()	返回一个数组,数组中包含该类中所有类和接口类的对象
forName(String className)	根据类名返回类的对象
getName()	获得类的完整路径名字
newInstance()	创建类的实例
getPackage()	获得类的包
getSimpleName()	获得类的名字
getSuperclass()	获得当前类继承的父类的名字
getInterfaces()	获得当前类实现的类或是接口

代码示例:

//1.获取Class类的对象的3种方式
/**
   注意://此处仅仅是获得Class对象  相当于Student stu;
				Class<?> c = Class.forName(className); 
		相当于Student stu;
		而后面学到的(获取构造方法):
			Object obj = c.newInstance();则是为对象实例化
		即:stu = new Student();
*/
	public static void test1() throws ClassNotFoundException {
		//1通过对象
		Class<? extends Student> stu = new Student().getClass();
		
		//2通过类名获取
		Class<Student> stu2 = Student.class;
		
		System.out.println(stu == stu2);  //返回为true,说明是一个对象
		
		//3forName("包名+类名")   
		//ClassNotFoundException 有异常
		Class<?> stu3 = Class.forName("cn.rjxy.tao18.Test04.Student");
		
		System.out.println(stu3 == stu); //返回true
	}

<2、获得类中属性相关的方法

方法	用途
getField(String name)	获得某个公有的属性对象
getFields()	获得所有公有的属性对象
getDeclaredField(String name)	获得某个属性对象
getDeclaredFields()	获得所有属性对象

代码示例:

//4.获取成员变量
	public static void test4() throws Exception {
		Class<Student> sclass = Student.class;
		//获取全部的成员变量
		Field[] fields = sclass.getDeclaredFields();
		for (Field f : fields) {
			//输出修饰符
			System.out.println(f.getModifiers() + "->" +
					Modifier.toString(f.getModifiers()));
			
			//输出全限定名(包名+类名)
			System.out.println(f.getName());
		}
	}
	//使用变量
	public static void test4_1() throws Exception{
		Class<Student> sclass = Student.class;
		Student stu = sclass.newInstance();
		
		//给name进行赋值
		Field f = sclass.getDeclaredField("name");
		f.setAccessible(true);
		f.set(stu, "ttt");  //相当于  stu.name = name;
		System.out.println(f.get(stu));
		
		//给Age进行赋值
		Field f2 = sclass.getDeclaredField("age");
		f2.set(stu, 23);
		System.out.println(f2.get(stu));
	}

❤️、获得类中注解相关的方法

方法	用途
getAnnotation(Class<A> annotationClass)	返回该类中与参数类型匹配的公有注解对象
getAnnotations()	返回该类所有的公有注解对象
getDeclaredAnnotation(Class<A> annotationClass)	返回该类中与参数类型匹配的所有注解对象
getDeclaredAnnotations()	返回该类所有的注解对象

<4、获得类中构造器相关的方法

方法	用途
getConstructor(Class...<?> parameterTypes)	获得该类中与参数类型匹配的公有构造方法
getConstructors()	获得该类的所有公有构造方法
getDeclaredConstructor(Class...<?> parameterTypes)	获得该类中与参数类型匹配的构造方法
getDeclaredConstructors()	获得该类所有构造方法

代码示例:

	//2.获取构造方法
	@SuppressWarnings("rawtypes")
	public static void test2() {
		Class<Student> stu = Student.class;
		
		//获取全部的构造方法
		Constructor[] constructor = stu.getDeclaredConstructors();
		System.out.println(constructor.length); //结果:2
		for (Constructor constructor2 : constructor) {
			
			//输出修饰符
			System.out.println(constructor2.getModifiers() + "->" +
					Modifier.toString(constructor2.getModifiers()));
			
			//输出全限定名(包名+类名)
			System.out.println(constructor2.getName());
			
			//返回参数列表的class类型的数组
			Class[] pClazz = constructor2.getParameterTypes();
			for (Class c : pClazz) {
				System.out.println(c.getName());
			}
			System.out.println("============================");
		}
	
	}
	//获取无参的构造方法,并创建对象
	public static void test2_1() throws Exception {
		Class<Student> qclass = Student.class;
		/**
		 * 方法一
		 */
		//获取无参的构造方法
		Constructor constructor = qclass.getDeclaredConstructor();
		//根据无参的构造方法创建对象
		Student student = (Student) constructor.newInstance();
		student.setName("hhhh");
		student.eat("qph");
		
		/**
		 * 方法二
		 * 通常无参的构造方法创建对象,不需要获取Constructor对象,
		 * 直接用Class对象直接newInstance()即可
		 */
		Student stu1 = qclass.newInstance();
		student.setName("ttt");
		student.eat("www");
	}
	
	//获取有参的构造方法,并创建对象
	public static void test2_2() throws Exception {
		Class<Student> qclass = Student.class;
		
		Constructor<Student> constructor = 
				qclass.getDeclaredConstructor(
					 	new Class[] {int.class, String.class});
		//设置私有的构造方法的可访问性,默认值为false,即默认不能访问私有的构造方法
		constructor.setAccessible(true);
		Student stu = constructor.newInstance(new Object[] {12,"李四"});
		stu.eat("qq");
	}

<5、获得类中方法相关的方法

方法	用途
getMethod(String name, Class...<?> parameterTypes)	获得该类某个公有的方法
getMethods()	获得该类所有公有的方法
getDeclaredMethod(String name, Class...<?> parameterTypes)	获得该类某个方法
getDeclaredMethods()	获得该类所有方法
//3.获取功能方法
	//getDeclaredMethods()只能获得本类的全部方法,包括私有方法
	//getMethods()获取本类及父类的非私有的方法
	public static void test3() {
		Class<Student> sclass = Student.class;
		Method[] methods = sclass.getDeclaredMethods();
		for (Method method : methods) {
			System.out.println(method.getModifiers());
			System.out.println(method.getName());
			System.out.println(method.getReturnType());
			System.out.println("============");
		}
	}
	//使用无参的方法
	public static void test3_1() throws Exception {
		Class<Student> sclass = Student.class;
		//使用无参构造方法创建对象
		Student stu = sclass.newInstance();
		//获取
		Method methods = sclass.getDeclaredMethod("hello");
		methods.invoke(stu);
	}
	
	//使用有参的方法
	public static void test3_2() throws Exception {
		Class<Student> sclass = Student.class;
		Student stu = sclass.newInstance();
	
		//调用eat方法
		Method methods = sclass.getDeclaredMethod("eat", 
				new Class[] {String.class} );
		methods.invoke(stu, new Object[] {"aaaaaa"});
		
		/**
		 * 练习:调用 drink、setAge()方法
		 */
		//调用drink方法
		Method methodDrink = sclass.getDeclaredMethod("drink", 
				new Class[] {String.class});
		methodDrink.setAccessible(true);
		methodDrink.invoke(stu, new Object[] {"水"});
	}

<6、类中其他重要的方法

方法	用途
isAnnotation()	如果是注解类型则返回true
isAnnotationPresent(Class<? extends Annotation> annotationClass)	如果是指定类型注解类型则返回true
isAnonymousClass()	如果是匿名类则返回true
isArray()	如果是一个数组类则返回true
isEnum()	如果是枚举类则返回true
isInstance(Object obj)	如果obj是该类的实例则返回true
isInterface()	如果是接口类则返回true
isLocalClass()	如果是局部类则返回true
isMemberClass()	如果是内部类则返回true

<2>Field类

Field代表类的成员变量(成员变量也称为类的属性)。

方法			用途
equals(Object obj)	属性与obj相等则返回true
get(Object obj)	获得obj中对应的属性值
set(Object obj, Object value)	设置obj中对应属性值

<3>Method类

Method代表类的方法。


方法		用途
invoke(Object obj, Object... args)	传递object对象及参数调用该对象对应的方法

<4>Constructor类

Constructor代表类的构造方法。

方法	用途
newInstance(Object... initargs)	根据传递的参数创建类的对象

▲▲▲▲示例
为了演示反射的使用,首先构造一个与书籍相关的model——Book.java,然后通过反射方法示例创建对象、反射私有构造方法、反射私有属性、反射私有方法,最后给出两个比较复杂的反射示例——获得当前ZenMode和关机Shutdown。

被反射类Book.java
public class Book{
private final static String TAG = “BookTag”;

private String name;
private String author;

@Override
public String toString() {
    return "Book{" +
            "name='" + name + '\'' +
            ", author='" + author + '\'' +
            '}';
}

public Book() {
}

private Book(String name, String author) {
    this.name = name;
    this.author = author;
}

public String getName() {
    return name;
}

public void setName(String name) {
    this.name = name;
}

public String getAuthor() {
    return author;
}

public void setAuthor(String author) {
    this.author = author;
}

private String declaredMethod(int index) {
    String string = null;
    switch (index) {
        case 0:
            string = "I am declaredMethod 1 !";
            break;
        case 1:
            string = "I am declaredMethod 2 !";
            break;
        default:
            string = "I am declaredMethod 1 !";
    }

    return string;
}

}
反射逻辑封装在ReflectClass.java
public class ReflectClass {
private final static String TAG = “peter.log.ReflectClass”;

// 创建对象
public static void reflectNewInstance() {
    try {
        Class<?> classBook = Class.forName("com.android.peter.reflectdemo.Book");
        Object objectBook = classBook.newInstance();
        Book book = (Book) objectBook;
        book.setName("Android进阶之光");
        book.setAuthor("刘望舒");
        Log.d(TAG,"reflectNewInstance book = " + book.toString());
    } catch (Exception ex) {
        ex.printStackTrace();
    }
}

// 反射私有的构造方法
public static void reflectPrivateConstructor() {
    try {
        Class<?> classBook = Class.forName("com.android.peter.reflectdemo.Book");
        Constructor<?> declaredConstructorBook = classBook.getDeclaredConstructor(String.class,String.class);
        declaredConstructorBook.setAccessible(true);
        Object objectBook = declaredConstructorBook.newInstance("Android开发艺术探索","任玉刚");
        Book book = (Book) objectBook;
        Log.d(TAG,"reflectPrivateConstructor book = " + book.toString());
    } catch (Exception ex) {
        ex.printStackTrace();
    }
}

// 反射私有属性
public static void reflectPrivateField() {
    try {
        Class<?> classBook = Class.forName("com.android.peter.reflectdemo.Book");
        Object objectBook = classBook.newInstance();
        Field fieldTag = classBook.getDeclaredField("TAG");
        fieldTag.setAccessible(true);
        String tag = (String) fieldTag.get(objectBook);
        Log.d(TAG,"reflectPrivateField tag = " + tag);
    } catch (Exception ex) {
        ex.printStackTrace();
    }
}

// 反射私有方法
public static void reflectPrivateMethod() {
    try {
        Class<?> classBook = Class.forName("com.android.peter.reflectdemo.Book");
        Method methodBook = classBook.getDeclaredMethod("declaredMethod",int.class);
        methodBook.setAccessible(true);
        Object objectBook = classBook.newInstance();
        String string = (String) methodBook.invoke(objectBook,0);

        Log.d(TAG,"reflectPrivateMethod string = " + string);
    } catch (Exception ex) {
        ex.printStackTrace();
    }
}

// 获得系统Zenmode值
public static int getZenMode() {
    int zenMode = -1;
    try {
        Class<?> cServiceManager = Class.forName("android.os.ServiceManager");
        Method mGetService = cServiceManager.getMethod("getService", String.class);
        Object oNotificationManagerService = mGetService.invoke(null, Context.NOTIFICATION_SERVICE);
        Class<?> cINotificationManagerStub = Class.forName("android.app.INotificationManager$Stub");
        Method mAsInterface = cINotificationManagerStub.getMethod("asInterface",IBinder.class);
        Object oINotificationManager = mAsInterface.invoke(null,oNotificationManagerService);
        Method mGetZenMode = cINotificationManagerStub.getMethod("getZenMode");
        zenMode = (int) mGetZenMode.invoke(oINotificationManager);
    } catch (Exception ex) {
        ex.printStackTrace();
    }

    return zenMode;
}

// 关闭手机
public static void shutDown() {
    try {
        Class<?> cServiceManager = Class.forName("android.os.ServiceManager");
        Method mGetService = cServiceManager.getMethod("getService",String.class);
        Object oPowerManagerService = mGetService.invoke(null,Context.POWER_SERVICE);
        Class<?> cIPowerManagerStub = Class.forName("android.os.IPowerManager$Stub");
        Method mShutdown = cIPowerManagerStub.getMethod("shutdown",boolean.class,String.class,boolean.class);
        Method mAsInterface = cIPowerManagerStub.getMethod("asInterface",IBinder.class);
        Object oIPowerManager = mAsInterface.invoke(null,oPowerManagerService);
        mShutdown.invoke(oIPowerManager,true,null,true);

    } catch (Exception ex) {
        ex.printStackTrace();
    }
}

public static void shutdownOrReboot(final boolean shutdown, final boolean confirm) {
    try {
        Class<?> ServiceManager = Class.forName("android.os.ServiceManager");
        // 获得ServiceManager的getService方法
        Method getService = ServiceManager.getMethod("getService", java.lang.String.class);
        // 调用getService获取RemoteService
        Object oRemoteService = getService.invoke(null, Context.POWER_SERVICE);
        // 获得IPowerManager.Stub类
        Class<?> cStub = Class.forName("android.os.IPowerManager$Stub");
        // 获得asInterface方法
        Method asInterface = cStub.getMethod("asInterface", android.os.IBinder.class);
        // 调用asInterface方法获取IPowerManager对象
        Object oIPowerManager = asInterface.invoke(null, oRemoteService);
        if (shutdown) {
            // 获得shutdown()方法
            Method shutdownMethod = oIPowerManager.getClass().getMethod(
                    "shutdown", boolean.class, String.class, boolean.class);
            // 调用shutdown()方法
            shutdownMethod.invoke(oIPowerManager, confirm, null, false);
        } else {
            // 获得reboot()方法
            Method rebootMethod = oIPowerManager.getClass().getMethod("reboot",
                    boolean.class, String.class, boolean.class);
            // 调用reboot()方法
            rebootMethod.invoke(oIPowerManager, confirm, null, false);
        }
    } catch (Exception e) {
        e.printStackTrace();
    }
}

}
调用相应反射逻辑方法
try {
// 创建对象
ReflectClass.reflectNewInstance();

        // 反射私有的构造方法
        ReflectClass.reflectPrivateConstructor();

        // 反射私有属性
        ReflectClass.reflectPrivateField();

        // 反射私有方法
        ReflectClass.reflectPrivateMethod();
    } catch (Exception ex) {
        ex.printStackTrace();
    }

    Log.d(TAG," zenmode = " + ReflectClass.getZenMode());

Log输出结果如下:

08-27 15:11:37.999 11987-11987/com.android.peter.reflectdemo D/peter.log.ReflectClass: reflectNewInstance book = Book{name=‘Android进阶之光’, author=‘刘望舒’}
08-27 15:11:38.000 11987-11987/com.android.peter.reflectdemo D/peter.log.ReflectClass: reflectPrivateConstructor book = Book{name=‘Android开发艺术探索’, author=‘任玉刚’}
08-27 15:11:38.000 11987-11987/com.android.peter.reflectdemo D/peter.log.ReflectClass: reflectPrivateField tag = BookTag
08-27 15:11:38.000 11987-11987/com.android.peter.reflectdemo D/peter.log.ReflectClass: reflectPrivateMethod string = I am declaredMethod 1 !
08-27 15:11:38.004 11987-11987/com.android.peter.reflectdemo D/peter.log.ReflectDemo: zenmode = 0
总结
本文列举了反射机制使用过程中常用的、重要的一些类及其方法,更多信息和用法需要近一步的阅读Google提供的相关文档和示例。

在阅读Class类文档时发现一个特点,以通过反射获得Method对象为例,一般会提供四种方法,getMethod(parameterTypes)、getMethods()、getDeclaredMethod(parameterTypes)和getDeclaredMethods()。getMethod(parameterTypes)用来获取某个公有的方法的对象,getMethods()获得该类所有公有的方法,getDeclaredMethod(parameterTypes)获得该类某个方法,getDeclaredMethods()获得该类所有方法。带有Declared修饰的方法可以反射到私有的方法,没有Declared修饰的只能用来反射公有的方法。其他的Annotation、Field、Constructor也是如此。

在ReflectClass类中还提供了两种反射PowerManager.shutdown()的方法,在调用的时候会输出如下log,提示没有相关权限。之前在项目中尝试反射其他方法的时候还遇到过有权限和没权限返回的值不一样的情况。如果源码中明确进行了权限验证,而你的应用又无法获得这个权限的话,建议就不要浪费时间反射了。

W/System.err: java.lang.reflect.InvocationTargetException
W/System.err: at java.lang.reflect.Method.invoke(Native Method)
W/System.err: at .ReflectClass.shutDown(ReflectClass.java:104)
W/System.err: at .MainActivity 1. o n C l i c k ( M a i n A c t i v i t y . j a v a : 25 ) W / S y s t e m . e r r : a t a n d r o i d . v i e w . V i e w . p e r f o r m C l i c k ( V i e w . j a v a : 6259 ) W / S y s t e m . e r r : a t a n d r o i d . v i e w . V i e w 1.onClick(MainActivity.java:25) W/System.err: at android.view.View.performClick(View.java:6259) W/System.err: at android.view.View 1.onClick(MainActivity.java:25)W/System.err:atandroid.view.View.performClick(View.java:6259)W/System.err:atandroid.view.ViewPerformClick.run(View.java:24732)
W/System.err: at android.os.Handler.handleCallback(Handler.java:789)
W/System.err: at android.os.Handler.dispatchMessage(Handler.java:98)
W/System.err: at android.os.Looper.loop(Looper.java:164)
W/System.err: at android.app.ActivityThread.main(ActivityThread.java:6592)
W/System.err: at java.lang.reflect.Method.invoke(Native Method)
W/System.err: at com.android.internal.os.Zygote M e t h o d A n d A r g s C a l l e r . r u n ( Z y g o t e . j a v a : 240 ) W / S y s t e m . e r r : a t c o m . a n d r o i d . i n t e r n a l . o s . Z y g o t e I n i t . m a i n ( Z y g o t e I n i t . j a v a : 769 ) W / S y s t e m . e r r : C a u s e d b y : j a v a . l a n g . S e c u r i t y E x c e p t i o n : N e i t h e r u s e r 10224 n o r c u r r e n t p r o c e s s h a s a n d r o i d . p e r m i s s i o n . R E B O O T . W / S y s t e m . e r r : a t a n d r o i d . o s . P a r c e l . r e a d E x c e p t i o n ( P a r c e l . j a v a : 1942 ) W / S y s t e m . e r r : a t a n d r o i d . o s . P a r c e l . r e a d E x c e p t i o n ( P a r c e l . j a v a : 1888 ) W / S y s t e m . e r r : a t a n d r o i d . o s . I P o w e r M a n a g e r MethodAndArgsCaller.run(Zygote.java:240) W/System.err: at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:769) W/System.err: Caused by: java.lang.SecurityException: Neither user 10224 nor current process has android.permission.REBOOT. W/System.err: at android.os.Parcel.readException(Parcel.java:1942) W/System.err: at android.os.Parcel.readException(Parcel.java:1888) W/System.err: at android.os.IPowerManager MethodAndArgsCaller.run(Zygote.java:240)W/System.err:atcom.android.internal.os.ZygoteInit.main(ZygoteInit.java:769)W/System.err:Causedby:java.lang.SecurityException:Neitheruser10224norcurrentprocesshasandroid.permission.REBOOT.W/System.err:atandroid.os.Parcel.readException(Parcel.java:1942)W/System.err:atandroid.os.Parcel.readException(Parcel.java:1888)W/System.err:atandroid.os.IPowerManagerStub$Proxy.shutdown(IPowerManager.java:787)
W/System.err: … 12 more

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值