反射介绍
JAVA反射机制是在运行状态中,
对于任意一个类,都能够知道这个类的所有属性和方法,
对于任意一个对象,都能够调用它的任意一个方法
这种动态获取的以及动态调用对象的方法的功能称为java语言的反射机制.
简单来说, 就可以把.class文件比做动物的尸体, 而反射技术就是对尸体的一种解剖.
通过反射技术, 我们可以拿到该字节码文件中所有的东西, 例如成员变量, 成员方法, 构造方法, 而且还包括私有
package com.demo.test_02;
/**
*
* @author Administrator
*
* 反射:
*
* 在运行时,我们可以获取任意一个类的所有方法和属性
* 在运行时,我们可以调用任意一个对象的所有方法和属性
*
* 反射的前提:
* 要获取类的字节码对象(Class对象)
*
*
*/
public class ReflectDemo {
public static void main(String[] args) throws Exception {
/**
*
* 字节码对象获取的三种方式:
* 1.对象.getClass() 该方法来自Object对象已经存在的情况下可以使用这种方式
* 2.类名.class 类名.class这是一个静态属性只要知道类名就可以获取字节码对象
* 3.Class.forName("com.demo.test") 通过Class类中的静态方法,指定字符串 该字符串就会是(包名+类名)
* 同样此处会抛出异常ClassNotFoundException 防止传入错误的类名
*/
// 第一种方式:
Student s = new Student();
Class<? extends Student> clazz = s.getClass();
// 第二种方式:
Class calzz1= Student.class;
//第三种方式:
Class clazz2 = Class.forName("com.demo.test_02.Student");
/**
* 字节码对象是用来描述什么的?
*
* 用来描述.class文件的
* 这里我们也可以用面向对象的思想来说明:Java中描述事物都是通过类来描述的
* 而字节码对象也可以看做是事物,那如何描述这种事物呢:
* 那就看这种事物有什么组成的
* 1.成员变量
* 2.成员方法
* 3.构造方法
*
*/
}
}
反射操作构造方法
通过获取的构造创建对象
步骤:
1.获得Class对象
2获得构造
3.通过构造对象获得实例化对象
package com.demo.test_02;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
/**
*
* @author Administrator
*
* 反射操作构造方法 步骤: 1.获得Class对象 2.获得构造 3.通过构造获得实例化对象
*
* 直接使用Class类中的newIntance()和获取的getConstructor()的区别就是: newInstance()方法,
* 只能通过空参的构造方法创建对象 getConstructor(Class<T>… parameterTypes)方法,
* 方法接受一个可变参数, 可以根据传入的类型来匹配对应的构造方法
*
*/
public class ReflectDemo2 {
public static void main(String[] args) throws Exception {
Class clazz = Class.forName("com.demo.test_02.Student");
// mthoed_01(clazz);
// method_02(clazz);
method_03(clazz);
// 通过字节码对象实例化对象
Object stu = clazz.newInstance();
// System.out.println(stu);
}
private static void method_03(Class clazz) throws NoSuchMethodException, InstantiationException,
IllegalAccessException, IllegalArgumentException, InvocationTargetException {
// 获得有参构造
Constructor constructor = clazz.getConstructor(String.class, int.class);
Object stu = constructor.newInstance("小明", 23);
System.out.println(constructor);
System.out.println(stu);
}
private static void method_02(Class clazz) throws NoSuchMethodException, Exception, IllegalAccessException,
IllegalArgumentException, InvocationTargetException {
// 获得字节码对象的无参构造
Constructor constructor = clazz.getConstructor();
Object stu = constructor.newInstance();
System.out.println(constructor);
System.out.println(stu);
}
private static void mthoed_01(Class clazz) {
// 获得所有的public 修饰的构造方法
Constructor[] constructors = clazz.getConstructors();
for (Constructor constructor : constructors) {
System.out.println(constructor);
}
}
}
直接通过Class类中的newInstance()和获取getConstructor()有什么区别?
newInstance()方法, 只能通过空参的构造方法创建对象
getConstructor(Class<T>… parameterTypes)方法, 方法接受一个可变参数, 可以根据传入的类型来匹配对应的构造方法
总结
Constructor<?>[] getConstructors()
Constructor<T> getConstructor(Class<?>... parameterTypes)
方法1: 获取该类中所有的构造方法, 返回的是一个数组
方法2: 方法接受一个可变参数, 可以根据传入的类型, 来匹配对应的构造方法
反射操作公共成员变量
反射public成员变量(字段)
通过反射运行public变量流程
1. 通过反射获取该类的字节码对象
Class clazz = Class.forName("com.demo.Person");
2. 创建该类对象
Object p = clazz.newInstance();
3. 获取该类中需要操作的字段(成员变量)
getField(String name) --> 方法传入字段的名称.
注意: 此方法只能获取公共的字段
Field f = clazz.getField("age");
4. 通过字段对象中的方法修改属性值
void set(Object obj, Object value) --> 参数1): 要修改那个对象中的字段, 参数2): 将字段修改为什么值.
f.set(ps, 99);
package com.demo.test_02;
import java.lang.reflect.Field;
/*
* 反射操作公有成员变量
* 通过反射运行public成员变量流程
* 1.通过反射获取类的字节码文件
* 2.创建该类的对象
* 3.获取要操作的该类的字段
* 4.通过字段对象中的方法修改属性值
*
*/
public class ReflectDemo3 {
public static void main(String[] args) throws Exception {
// 1.通过反射获取类的字节码文件
Class<?> clazz = Class.forName("com.demo.test_02.Student");
// method_01(clazz);
// method_02(clazz);
// method_03(clazz);
}
private static void method_03(Class<?> clazz) {
// 获得所有的public公有的成员变量
Field[] fields = clazz.getFields();
for (Field field : fields) {
System.out.println(field);
}
}
private static void method_02(Class<?> clazz) {
// 获取所有的成员变量包括私有
Field[] declaredFields = clazz.getDeclaredFields();
for (Field field : declaredFields) {
System.out.println(field);
}
}
private static void method_01(Class<?> clazz)
throws InstantiationException, IllegalAccessException, NoSuchFieldException {
// 2.创建该类的对象
Object stu = clazz.newInstance();
// 3.获取要操作的该类的字段
Field f = clazz.getField("gender");
System.out.println(f);
// 4.通过字段对象中的方法修改属性值
f.set(stu, "男");
Object object = f.get(stu);
System.out.println(object);
}
}
方法总结
通过反射获取成员变量并使用
Field[] getFields() --> 返回该类所有(公共)的字段
Field getField(String name) --> 返回指定名称字段
Field[] getDeclaredFields() --> 暴力反射获取所有字段(包括私有)
Field getDeclaredField(String name) --> 暴力反射获取指定名称字段
Field:
Object get(Object obj) --> Field对象调用, 返回传入对象的具体字段
void set(Object obj, Object value) --> Field对象调用
参数1: 要修改的对象
参数2: 将此对象的字段修改为什么值.
反射操作私有成员变量
反射private成员变量(字段)
反射private属性执行流程
1. 获取学生类字节码对象
2. 获取学生对象
3. 通过getDeclaredField方法获取私有字段
4. 通过setAccessible让jvm不检查权限
5. 通过set方法设置对象为具体的值
package com.demo.test_02;
import java.lang.reflect.Field;
/**
* 反射private属性执行流程 1. 获取学生类字节码对象 2. 获取学生对象 3. 通过getDeclaredField方法获取私有字段
* 4. 通过setAccessible让jvm不检查权限 5. 通过set方法设置对象为具体的值
*
*/
public class ReflectDemo4 {
public static void main(String[] args) throws Exception {
// 1. 获取学生类字节码对象
Class<?> clazz = Class.forName("com.demo.test_02.Student");
method_01(clazz);
method_02(clazz);
}
private static void method_02(Class<?> clazz)
throws InstantiationException, IllegalAccessException, NoSuchFieldException {
// 2. 获取学生对象
Object stu = clazz.newInstance();
// 3. 通过getDeclaredField方法获取私有字段
Field f = clazz.getDeclaredField("name");
Field f1 = clazz.getDeclaredField("age");
// 4. 通过setAccessible让jvm不检查权限
f.setAccessible(true);
f1.setAccessible(true);
// 5. 通过set方法设置对象为具体的值
f.set(stu, "老王");
f1.set(stu, 24);
Object name = f.get(stu);
Object age = f1.get(stu);
System.out.println(name);
System.out.println(age);
}
private static void method_01(Class<?> clazz) throws InstantiationException, IllegalAccessException {
// 2. 获取学生对象
Object stu = clazz.newInstance();
// 3. 通过getDeclaredField方法获取私有字段
Field[] declaredFields = clazz.getDeclaredFields();
for (Field field : declaredFields) {
field.setAccessible(true);
System.out.println(field);
}
}
}
方法总结
Field[] getDeclaredFields() --> 暴力反射获取所有字段(包括私有)
Field getDeclaredField(String name) --> 暴力反射获取指定名称字段
void setAccessible(boolean flag) --> 让jvm不检查权限
通过反射获取成员方法并使用
反射获取普通成员方法
反射public方法执行流程
1. 获取学生类字节码对象
2. 反射手段创建学生对象
3. 调用getMethod方法获取Method对象, 方法形参接受方法的名字
4. 调用Method方法中的invoke()将方法运行
package com.demo.test_02;
import java.lang.reflect.Method;
/**
* 反射public方法执行流程
* 1. 获取学生类字节码对象
* 2. 反射手段创建学生对象
* 3. 调用getMethod方法获取Method对象,方法形参接受方法的名字
* 4. 调用Method方法中的invoke()将方法运行
*
*/
public class ReflectDemo5 {
public static void main(String[] args) throws Exception {
//1. 获取学生类字节码对象
Class<?> clazz = Class.forName("com.demo.test_02.Student");
//2. 反射手段创建学生对象
Object stu = clazz.newInstance();
//3. 调用getMethod方法获取Method对象,方法形参接受方法的名字
Method m = clazz.getMethod("add");
//4. 调用Method方法中的invoke()将方法运行
m.invoke(stu);
}
}
package com.demo.test_02;
public class Student {
private String name;
private int age;
public String gender;
public Student(String name, int age, String gender) {
super();
this.name = name;
this.age = age;
this.gender = gender;
}
public Student(String name, int age) {
super();
this.name = name;
this.age = age;
}
public Student() {
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
public String toString() {
return "Student [name=" + name + ", age=" + age + "]";
}
public void add(){
System.out.println("helloWorld");
}
}
方法总结
Class:
Method getMethod(String name, Class<?>... parameterTypes)
// 此方法由字节码对象调用
// 参数1: 要反射的方法名称
// 参数2: 此方法需要接受的参数类型(注意,传入的都是字节码)
Method:
Object invoke(Object obj, Object... args)
// 方法由Method对象调用
// 参数1: 要由那个对象调用方法
// 参数2: 方法需要的具体实参(实际参数)
问题: 私有的成员方法怎么玩?
// 获取字节码对象
Class clazz = Class.forName("com.demo.Student");
// 创建学生对象
Object stu = clazz.newInstance();
// 暴力反射获取方法
Method method = clazz.getDeclaredMethod("method");
// 让jvm不检查权限
method.setAccessible(true);
// 执行方法
method.invoke(stu);
JavaBean的概述和规范
JavaBean的概述:
将需要操作的多个属性封装成JavaBean, 简单来说就是用于封装数据的
规范:
类使用公共进行修饰
提供私有修饰的成员变量
为成员变量提供公共getter和setter方法
提供公共无参的构造
package com.demo.test_03;
import java.io.Serializable;
public class Student implements Serializable{
/**
*
*/
private static final long serialVersionUID = 1L;
private String name;
private int age;
private String gender;
public Student() {
}
public Student(String name, int age, String gender) {
super();
this.name = name;
this.age = age;
this.gender = gender;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public String getGender() {
return gender;
}
public void setGender(String gender) {
this.gender = gender;
}
@Override
public String toString() {
return "Student [name=" + name + ", age=" + age + ", gender=" + gender + "]";
}
}
BeanUtils的概述
BeanUtils的由来
之前我们使用的类都是来自Java编写好的源代码
而这个BeanUtils却是一个叫做Apache的组织编写.
那么这个组织编写的代码当中, 有一个系列可以很方便的提高我们今后的开发效率.
这个系列为Commons, BeanUtils就是其中之一
准备工作
1. 导入两个jar包
commons-beanutils-1.8.3.jar
commons-logging-1.1.1.jar
2. 将jar包Build path 配置到当前的classpath环境变量中
2.3BeanUtils的常用方法
static void setProperty(Object bean, String name, Object value)
static String getProperty(Object bean, String name)
static void populate(Object bean, Map properties)
setProperty 用来给对象中的属性赋值(了解)
参数1: 需要设置属性的对象
参数2: 需要修改的属性名称
参数3: 需要修改的具体元素
getProperty 用来获取对象中的属性(了解)
参数1: 要获取的javaBean对象
参数2: 对象中的哪个属性
Populate 用来给对象中的属性赋值(掌握)
参数1: 要设置属性的对象
参数2: 将属性以Map集合的形式传入
Key : 属性的名称
Value: 属性具体的值
package com.demo_02;
import java.lang.reflect.InvocationTargetException;
import java.util.HashMap;
import java.util.Map;
import org.apache.commons.beanutils.BeanUtils;
/*
* BeanUtils:Apache commons提供的一个组件,主要功能就是为了简化JavaBean封装数据的操作
* static void setProperty(Object bean, String name, Object value)
* static String getProperty(Object bean, String name)
* static void populate(Object bean, Map properties)
*
* 注意:BeanUtils的setProperty和getProperty方法底层并不是直接操作成员变量,而是操作和成员变量名有关的get和set方法
*/
public class BeanUtilsDemo {
public static void main(String[] args) throws ReflectiveOperationException {
//static void populate(Object bean, Map properties)
Person p = new Person();
Map<String,Object> map = new HashMap<String,Object>();
map.put("name", "laowang");
map.put("age", 38);
map.put("gender", "male");
BeanUtils.populate(p,map);
System.out.println(p);
}
private static void method() throws IllegalAccessException, InvocationTargetException, NoSuchMethodException {
Person p = new Person();
//System.out.println(p);
//static void setProperty(Object bean, String name, Object value) :给JavaBean对象的成员变量进行赋值
BeanUtils.setProperty(p, "name", "zhangsan");
//BeanUtils.setProperty(p, "age", 18);
//System.out.println(p);
//static String getProperty(Object bean, String name)
String name = BeanUtils.getProperty(p, "name");
System.out.println(name);
}
}
方法总结
三个方法底层是通过反射实现, 而且反射操作的是setXxx方法和getXxx方法.
所以编写JavaBean的时候一定要注意格式
2.4自定义BeanUtils的赋值和获取方法实现.
2.4.1功能分析
定义MyBeanUtils工具类, 实现与BeanUtils相同的功能
public static void setProperty(Object bean,String name,Object value)
// 设置任意对象的, 任意属性, 为任意的值
public static String getProperty(Object bean,String name)
// 获取任意对象的任意属性
public static void populate(Object bean,Map map)
// 修改任意对象中的属性, 为传入Map集合中的键和值
package com.demo_03;
import java.lang.reflect.Field;
public class MyBeanUtils {
private MyBeanUtils() {}
//public static void setProperty(Object bean,String name,Object value)
public static void setProperty(Object bean,String name,Object value) throws ReflectiveOperationException {
//根据JavaBean对象获取对应的字节码对象
Class clazz = bean.getClass();
//根据字节码对象获取对应的Field对象
Field f = clazz.getDeclaredField(name);
//设置权限,让虚拟机不进行访问的检查
f.setAccessible(true);
//赋值
f.set(bean, value);
}
//public static String getProperty(Object bean,String name)
public static String getProperty(Object bean,String name) throws ReflectiveOperationException {
Class clazz = bean.getClass();
Field f = clazz.getDeclaredField(name);
f.setAccessible(true);
Object obj = f.get(bean);
return obj.toString();
}
}
2.5自定义BeanUtils的populate方法实现
2.5.1功能分析
public static void populate(Object bean,Map map)
// 修改任意对象中的属性, 为传入Map集合中的键和值
思路:
1.获取传入对象的字节码对象
2.获取map集合中所有的键和值
3.调用Class中的getDeclaredField()方法将每一个键传入, 得到Field对象
4.通过Field对象中的set方法赋值
5.Try catch捕获getDeclaredField方法可能发生的异常.(为了方式传入错误的值)
//public static void populate(Object bean,Map map)
public static void populate(Object bean,Map map) throws ReflectiveOperationException {
//通过JavaBean对象来获取对应的字节码对象
Class clazz = bean.getClass();
//获取Map中所有的key
Set keys = map.keySet();
for (Object key : keys) {
try {
//根据key来获取对应的Field对象
Field f = clazz.getDeclaredField(key.toString());
//根据key来获取Map中对应的value
Object value = map.get(key);
f.setAccessible(true);
f.set(bean, value);
} catch(NoSuchFieldException e) {
//e.printStackTrace();
}
}
}