java反射 内存_Java反射

什么是反射?

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

首先我们来了解一下类的加载过程:Class对象的由来是将class文件读入内存,并为之创建一个Class对象。

下图所示:

5f3664332489

类的加载过程

1、当我们new一个对象的时候,JVM会去本地寻找对象的.class文件,并加载到jvm内存中;

2、将.class文件读入内存后生产一个.class对象,一个类只生产一个.class对象;

反射的应用场景

在Java程序中许多对象在运行是都会出现两种类型:编译时类型和运行时类型。

编译时的类型由声明对象时的类型来决定,运行时的类型由实际赋值给对象的类型决定,如:

Person person=new Student();

其中,编译时类型为Person,运行时类型为Student;

java程序在运行时可以接收到外部传入的对象,这些对象在编译时的类型为Object,但是程序有需要调用该对象的运行时类型的方法;为了解决这些问题,程序需要在运行时发现对象和类的真实信息,如果编译时根本无法预知该对象和类属于哪些类, 程序只能依靠运行时的信息来发现该对象和类的真实信息,此时就必须要使用反射了;

Java反射的API

Class类:反射的核心类,可以获取类的属性,方法等信息;

Field类:Java.lang.reflec包中的类,表示类的成员变量,可以用来获取和设置类之中的属性值;

Method类:Java.lang.reflec包中的类,表示类的方法,它可以用来获取类中的方法信息或者执行方法;

Constructor类:Java.lang.reflec包中的类,表示类的构造方法;

我们来了解一下反射的步骤:

1、获取想要操作类的Class对象;

2、调用Class类中的方法;

3、使用反射API来操作这些信息;

获取Class对象的方法

1、调用某个对象的getClass()方法

Student student = new Student();

Class clazz = student.getClass();

2、调用某个类的class属性来获取该类对应的Class对象

Class clazz=Student.class;

3、使用Class类中的forName()静态方法; (最安全/性能最好 推荐)

Class clazz=Class.forName("com.jiao.mytest.proxy.Student");//类的全路径

反射获取构造方法

Constructor类:Java.lang.reflec包中的类,表示类的构造方法;

我们来举个例子,获取对象的构造方法并使用,我们用Student类来举例,Student类代码如下:

public class Student {

//默认构造方法

Student(String name) {

Log.d("Test", "调用了默认的构造方法 name=:" + name);

}

//无参构造方法

public Student() {

Log.d("Test", "调用了共有、无参构造方法");

}

//有多个参数的构造方法

public Student(String name, int age) {

Log.d("Test", "调用了多个参数的构造方法 姓名:" + name + "年龄:" + age);

}

//受保护的构造方法

protected Student(boolean n) {

Log.d("Test", "调用了受保护的构造方法 n = " + n);

}

//私有构造方法

private Student(int age) {

Log.d("Test", "调用私有的构造方法 年龄:" + age);

}

}

然后我们在主程序中通过反射来操作该类的构造方法

private void reflect() throws Exception {

Class clazz = Class.forName("com.jiao.mytest.proxy.Student");//类的全路径

Log.d("Test", "获取所有公有构造方法");

Constructor[] consArray = clazz.getConstructors();

for (Constructor c : consArray) {

Log.d("Test", c.toString());

}

Log.d("Test", "获取所有构造方法(包括:私有、受保护、默认、公有)");

Constructor[] consArray1 = clazz.getDeclaredConstructors();

for (Constructor c : consArray1) {

Log.d("Test", c.toString());

}

Log.d("Test", "获取公有、无参构造方法");

Constructor con = clazz.getConstructor(null);

Log.d("Test", "con= " + con);

Object obj = con.newInstance();

Log.d("Test", "获取私有构造方法");

Constructor con1 = clazz.getDeclaredConstructor(int.class);

Log.d("Test", "con1= " + con1);

con1.setAccessible(true);//暴力反射

Object obj1 = con1.newInstance(20);

}

控制台输出:

5f3664332489

构造方法反射测试

反射获取成员变量

Field类:Java.lang.reflec包中的类,表示类的成员变量,可以用来获取和设置类之中的属性值;

下面举例说明如何获取类的成员变量并操作,还是以Student类为例

public class Student {

public String name;

protected int age;

boolean sex;

private String phone;

public String toString() {

return "Student [name=" + name + ",age=" + age + ",性别=" + sex + ",phone=" + phone + "]";

}

}

主程序代码测试:

private void reflect() throws Exception {

Class clazz = Class.forName("com.jiao.mytest.proxy.Student");//类的全路径

Log.d("Test", "获取所有字段(包括:私有、受保护、默认、公有)");

Field[] fields1 = clazz.getDeclaredFields();

for (Field f : fields1) {

Log.d("Test", f.toString());

}

Log.d("Test", "获取公有字段,并调用");

Field f = clazz.getField("name");

Log.d("Test", "f= " + f.toString());

Object objStudent = clazz.getConstructor().newInstance();

f.set(objStudent, "柳岩");

Student student = (Student) objStudent;

Log.d("Test", "student.name=" + student.name);

Log.d("Test", "获取私有字段,并调用");

Field field = clazz.getDeclaredField("phone");

field.setAccessible(true);//暴力反射

field.set(objStudent, "18888888888");

Log.d("Test", "student.phone=" + student.toString());

}

控制台输出:

5f3664332489

反射获取成员变量测试

反射获取成员方法

Method类:Java.lang.reflec包中的类,表示类的方法,它可以用来获取类中的方法信息或者执行方法;

反射获取类的成员方法,并调用,还是以Student为例:

public class Student {

public void test1(String s) {

Log.d("Test", "调用了公有的String参数的方法test1 s=" + s);

}

protected void test2() {

Log.d("Test", "调用了受保护的无参的方法test2");

}

private int test3() {

Log.d("Test", "调用了私有的无参有返回值的方法test3");

return 100;

}

}

主程序中使用:

private void reflect() throws Exception {

Class clazz = Class.forName("com.jiao.mytest.proxy.Student");//类的全路径

Log.d("Test", "获取所有公有方法");

Method[] methods = clazz.getMethods();

for (Method m : methods) {

Log.d("Test", m.toString());

}

Log.d("Test", "获取所有方法(包括:私有、受保护、默认、公有)");

Method[] methods1 = clazz.getDeclaredMethods();

for (Method m : methods) {

Log.d("Test", m.toString());

}

Log.d("Test", "获取公有方法,并调用");

Method method1 = clazz.getMethod("test1", String.class);

Object objStudent = clazz.getConstructor().newInstance();

Student student = (Student) objStudent;

method1.invoke(objStudent, "柳岩");

Log.d("Test", "获取私有方法,并调用");

Method method2 = clazz.getDeclaredMethod("test3");

method2.setAccessible(true);//暴力反射

Log.d("Test", "私有test3方法返回:" + method2.invoke(objStudent, null));

}

控制台输出:

5f3664332489

反射获取方法测试

ok以上就是对java反射的简单介绍,本篇主要讲解了

1、反射的概念

2、java中类的加载过程

3、为什么要用到反射

4、反射API的使用

接下来我还会继续总结反射的高级应用;

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值