Java反射机制的那些事儿,java反射机制的原理

package com.test.demo;

public class Student {

public String name;

private int age;

public Student() {

}

private Student(String name, int age) {

this.name = name;

this.age = age;

}

public void show(String msg){

System.out.println("show方法 = " + msg);

}

private void speak(String msg,int number){

System.out.println(“speak方法 = " + msg +”:"+ number );

}

@Override

public String toString() {

return “Student{” +

“name=’” + name + ‘’’ +

“, age=” + age +

‘}’;

}

}

反射的使用1:构造器(Constructor)的反射


再次之前,我们可以通过公共的空参构造new一个Student,但是无法new私有的满参构造。

Student student = new Student();

现在来反射构造构造器(反射的形式创建实例)

public static void main(String[] args)

throws NoSuchMethodException, IllegalAccessException,

InvocationTargetException, InstantiationException {

//获取Class对象

Class<?> clazz = Student.class;

/*

根据参数类型获取相应的构造器

参数类型是形参类型

*/

Constructor<?> constructor = clazz.getConstructor();

/*

创建实例

参数类型是实参类型(形参一一对应)

*/

Object obj = constructor.newInstance();

System.out.println("obj = " + obj);

}

这样获取到的Student对象和new出来的空参构造器new出来的对象效果一样的(实际业务开发并没有意义)。

前者通过new创建出来对象的方式相比用反射创建的对象更被动,前者 是被new出来的,而用反射,是自己创建自己(对象),构造方法反客为主。

还有一种方式,就是直接通过Class对象创建构造器:

public static void main(String[] args)

throws IllegalAccessException, InstantiationException {

//获取Class对象

Class<?> clazz = Student.class

《一线大厂Java面试题解析+后端开发学习笔记+最新架构讲解视频+实战项目源码讲义》

【docs.qq.com/doc/DSmxTbFJ1cmN1R2dB】 完整内容开源分享

;

/*

默认调用空参构造创建一个实例

jdk9中已过时

*/

Object obj = clazz.newInstance();

System.out.println("obj = " + obj);

}

在Student类中 ,还有一个私有的构造器,正常方式下是不能通过私有构造器创建对象的。,但是反射可以做到:

public static void main(String[] args)

throws NoSuchMethodException, IllegalAccessException,

InvocationTargetException, InstantiationException {

//获取Class对象

Class<?> clazz = Student.class;

/*

获取构造

因为权限是私有,但getConstructor()只能获取public修饰的方法

getDeclaredConstructor():获取声明的方法。只要声明的就可以

*/

Constructor<?> constructor = clazz.getDeclaredConstructor(String.class, int.class);

System.out.println(“满参私有构造 :” + constructor);

/*

私有构造,newInstance会产生非法访问异常:java.lang.IllegalAccessException

所以要改变权限setAccessible() -->暴力反射

*/

constructor.setAccessible(true);

Object obj = constructor.newInstance(“小明”,20);

System.out.println("obj = " + obj);

}

以上就是利用反射来创建一个对象(反射构造器)。

反射的使用2:方法(Method)的反射


接下来看看Student对象内两个方法的反射

我们之前(外部)使用方法,都是都是通过对象调用(非私有)方法,如果是静态方法就是类直接调用。

那么,使用反射调用(非私有)方法,该怎么做?

public static void main(String[] args)

throws NoSuchMethodException, IllegalAccessException,

InvocationTargetException {

//获取Class对象

Student student = new Student();

Class<? extends Student> clazz = student.getClass();

/*

getMethod():获取Class对象里的方法

参数一:方法名

参数二:参数列表类型

*/

Method show = clazz.getMethod(“show”, String.class);

/*

调用show方法需要对象和参数

invoke()方法:调用的意思

参数一:调用此方法的对象

参数二:调用此方法需要传入的实参

*/

show.invoke(student, “hello public show”);

}

反射可以理解为语言语法上的倒装句:

我们平时写代码都是我(对象)去调用方法,这里就是:

new Student().show(“对象调用方法”);

而在 show.invoke(student, “hello public show”); 中,

show方法考虑的是谁来调用我,然后Student对象说,我来调用你(student作为参数)。

扩展:如果公共的show方法加上static关键字,会影响方法调用吗?

提示:静态与对象无关.

答:加上static关键字,普通代码即使不new对象也可以调用,这个大家都知道,那么,在show.invoke(student, “hello public show”); 中参数1 写 null 也是不影响的,因为,show方法来自于 Student的Class对象。

接下来看看私有方法的反射如何实现?

ps: 反射通道的API都很有规律,可读性很强

public static void main(String[] args)

throws NoSuchMethodException, IllegalAccessException,

InvocationTargetException {

//获取Class对象

Student student = new Student();

Class<? extends Student> clazz = student.getClass();

/*

getDeclaredMethod():获取Class对象里的声明过的方法(包括)

参数一:方法名

参数二:参数列表类型

*/

Method speak = clazz.getDeclaredMethod(“speak”, String.class, int.class);

//私有方法,暴力反射

speak.setAccessible(true);

/*

调用show方法需要对象和参数

invoke()方法:调用的意思

参数一:调用此方法的对象

参数二:调用此方法需要传入的实参

*/

speak.invoke(student, “hello private speak”,2018);

}

反射的使用3:属性(Field)的反射


在Student实体中有一个共有属性一个私有属性,我们可以通过对象来设置共有属性的值,那么通过反射如何实现所有属性的赋值?

先来看看共有属性name的赋值

public static void main(String[] args)

throws ClassNotFoundException, NoSuchFieldException,

IllegalAccessException, InstantiationException {

//获取Class对象,参数全限定名

Class<?> clazz = Class.forName(“com.test.demo.Student”);

/*

getField():通过属性名获取属性

*/

Field name = clazz.getField(“name”);

//获取对象

Object obj = clazz.newInstance();

/*

设置一个值

参数一:哪个对象的属性值

参数二:参数

*/

name.set(obj,“张三”);

System.out.println(obj);

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值