Java开发知识之JAVA的反射

Posted iBinary

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java开发知识之JAVA的反射相关的知识,希望对你有一定的参考价值。

Java中的反射机制

一丶什么是反射机制

  • 反射就是指Java在运行期间,可以获得类的一些信息.比如构造方法(私有共有...)以及属性

二丶Class类,自描述类.获取类名的三种方法

1.使用静态属性Class获取

代码:

	Class cls = Person.class; //获取自描述类
	String strClassName = cls.getName();获取类的包名 + 类名
	输出(strClassName);

首先我们的Class类是自描述类.可以返回这个类的描述信息.
有很多方法.比如获取类名字,方法.构造等等.

2.使用Class方法 forName()获取自描述Class

public static void main(String[] args) throws Exception {
		// TODO 自动生成的方法存根
		Class cls = Class.forName("com.ibinary_01.Person");//传入完整类名获取Class
		String clsName = cls.getName();
		System.out.println(clsName);
}

可以看到,我们使用forName必须传入完整的类名.

3.使用Object自带的getClass返回自描述Class

  • 因为我们的类,继承了Object,所以需要实例,进行调用getClass方法.
    代码如下
public static void main(String[] args) throws Exception {
		// TODO 自动生成的方法存根
		Class cls = new Person().getClass();
		String clsName = cls.getName();
		System.out.println(clsName);
	}

三个的输出结果都一样,演示一下

三丶了解一下自描述类Class方法.

1.Class针对包,类,父类.接口的方法


通过以上图片可以看到. 我们可以获得包的路径. 我们上面使用的getName(); 那么我们还可以使用getSuperclss();
获得父类自描述类(Class),然后继续调用getName();获取父类的名称.
代码例子如下:

	public static void main(String[] args) throws Exception {
		// TODO 自动生成的方法存根
		Class cls = new Person().getClass();
		Class supercls = cls.getSuperclass();
		System.out.println(supercls.getName());
	}

当然,我使用的是getClass();首先获得自己的Class,然后在获取父类的Class,最后调用父类的Class的getName();获得类名.
我们也可以使用上面所说的三种方法.

2.Class对构造方法提供的方法


可以看到,以上是我们Class为构造方法提供的方法获取.

3.Class对方法提供的方法.

上面标题有点绕,反正就是对我们的类的方法.提供方法去操作.或者获取.

4.Class对成员变量提供的方法

一样的,对我们的成员变量也提供了方法操作.

5.对内部类提供的方法

总结

  1. 针对我们的类,方法.构造.以及成员方法都提供了方法支持.首先要熟悉类的操作.
  2. 在通过getFilds()和getMethods()方法一次获得权限为public的成员变量和方法的时候.将包含从父类中继承到的成员变量和方法.也就是说会获得父类的公共方法和成员变量
  3. getDeclaredFields()和GETdeclaredMethods();只是获得奔雷中定义的所有成员变量和方法.

四丶反射访问构造方法

  • 方面我们讲了,Class中提供了对构造方法的支持.返回Constructior这个类.
  • 这个类也提供了对构造方法的操作.

1.构造方法类提供的方法.

构造方法

请注意 newInstance.我们可以利用指定参数.创建一个该类对象.在以后的安卓逆向中.可能会需要.

2.Modifier类

这个类则是传入类型则是调用Construct类中的getModifiers();返回构造方法的类型.传入.然后可以判断是什么属性.
提供的方法

  • 四种获得构造方法方式代码
    四种获得构造方法,其实是Class类提供的方法.在我们一开始已经截图给出了.
    具体就是代码例子使用了.

1.获取所有公共权限的构造方法

public static void main(String[] args) throws Exception {
		// TODO 自动生成的方法存根
		getAllConstruct(Person.class);
		
	}
	static void getAllConstruct(Class<?> cls) {
		//第一种,获得所有的构造方法
		Constructor c[] = cls.getConstructors();
		//遍历输出
		for(Constructor i:c) {
			System.out.println(i);
		}
		
	}

可以看到,上边调用的就是getConstructors():获得了所有公共权限的方法,进行遍历输出了.
输出结果:

2.获取公共权限指定的构造方法

private static void getPublicConstruct(Class<Person> cls) {
		// TODO 自动生成的方法存根
		try {
			Constructor<Person> c = cls.getConstructor(String.class,int.class);//获取指定公共权限的Construct
			c.newInstance("1",2);    //创建构造方法,并且调用
			
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

获取公共的权限的指定构造方法,我们就需要传入类型的Class. 比如参数为 int.Class 代表一个参数.
下方调用newInstance();就是在使用这个构造.

3.获取所权限的构造方法,放到数组中

	private static void getAllConstruct(Class<Person> cls) {
		// TODO 自动生成的方法存根
		Constructor a[] = cls.getDeclaredConstructors();
		for(Constructor i:a) {
			System.out.println(i);
		}
	}

输出结果:

4.获取指定的构造方法

获得指定的就跟上面一样.比如这次我们获得priate权限的方法.并且调用它.

	private static void getPrivateConstructAnNewInstance(Class<Person> cls) throws InstantiationException, IllegalAccessException {
		// TODO 自动生成的方法存根
		
		try {
			Constructor<Person> a = cls.getDeclaredConstructor(int.class,int.class,int.class);//获得指定的
			a.setAccessible(true);
			a.newInstance(1,1,1);
		} catch (Exception e) {
			// TODO 自动生成的 catch 块
			e.printStackTrace();
		} 
		
	}

首先我们要使用 getDeclaredConstructor(int.Class...);获取指定的构造方法
因为是私有的,所以我们要使用构造方法类提供的 setAccessible(true);设置为共有的.
此时我们在使用newInstance(对应参数类型的参数); 即可实现调用.
结果图:

5.获取构造方法所有参数类型

其实这个方法上面也提供了.叫做 getParameterTypes();这个方法返回一个Class数组.
我们遍历这个数组即可.

private static void getPrivateConstructAnNewInstance(Class<Person> cls) throws InstantiationException, IllegalAccessException {
		// TODO 自动生成的方法存根
		
		try {
			Constructor<?>[] a = cls.getConstructors();
			for(Constructor i:a) {
				System.out.println("构造函数 = " + i);//输出构造函数
				Class Par[] = i.getParameterTypes();
				for(Class j:Par) {
					System.out.print("      			 	"  + j);//输出参数类型
				}
				System.out.println();
			}
			
		} catch (Exception e) {
			// TODO 自动生成的 catch 块
			e.printStackTrace();
		} 
		
	}

输出结果:

五丶成员变量的获取以及使用

1.获取所有公共权限成员变量

private static void getPublicAllFields(Class<Person> cls) {
		// TODO 自动生成的方法存根
		Field[] f = cls.getFields();//获取所有成员变量
		for(Field i:f) {
			System.out.println(i);
		}
	}

使用方法getFields()即可.输出之后会显示所有的名字.

2.获取公共权限指定的成员变量

使用方法: getField(String name);

	private static void getPublicAllFields(Class<Person> cls) {
		// TODO 自动生成的方法存根
		Field f;
		try {
			f = cls.getField("pid");
			System.out.println(f);
		} catch (Exception e) {
			// TODO 自动生成的 catch 块
			e.printStackTrace();
		} 
	}

结果同上,我们需要传入名字才可以.

3.获取所有权限的成员变量

private static void getPublicAllFields(Class<Person> cls) {
		// TODO 自动生成的方法存根
		Field a[] = cls.getDeclaredFields();
		for(Field i:a) {
			System.out.println(i);
		}
	}

输出结果:

4.获得指定权限的成员变量

	private static void getPublicAllFields(Class<Person> cls) {
		// TODO 自动生成的方法存根
		try {
			Field a = cls.getDeclaredField("name");
			System.out.println(a);
		} catch (Exception e) {
			// TODO 自动生成的 catch 块
			e.printStackTrace();
		} 
	}

5.设置跟使用成员变量

我们上面可以获取.但是我们要使用.或者要更改值.
二个步骤.
1.获取Cls new出来的实例对象.
2.使用Set方法设置值.
如果设置私有,则要设置权限才可以.

private static void getPublicAllFields(Class<Person> cls) throws Exception {
		// TODO 自动生成的方法存根
		Object obj = cls.newInstance();
		try {
			Field a = cls.getDeclaredField("name");
			a.setAccessible(true); //使用私有就要设置权限.
			a.set(obj, "张三");
			System.out.println(a.get(obj));//并且获取设置的成员变量值.
		} catch (Exception e) {
			// TODO 自动生成的 catch 块
			e.printStackTrace();
		} 
	}

结果:

以上是关于Java开发知识之JAVA的反射的主要内容,如果未能解决你的问题,请参考以下文章

java 反射代码片段

Android插件化开发之Hook StartActivity方法

JAVA之反射知识点整理

Java安全-入门篇开发基础(MavenJavaMVC反射类加载代码远程调试)

Java基础知识之反射机制简介

java开发--反射技术