反射机制的概述和字节码对象的获取方式 JavaBean的概述&BeanUtils的使用 反射操作构造方法、成员方法、成员属性

 反射介绍

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. 通过setAccessiblejvm不检查权限
        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, 简单来说就是用于封装数据的

规范:

类使用公共进行修饰

提供私有修饰的成员变量

为成员变量提供公共gettersetter方法

提供公共无参的构造

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. jarBuild 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();
			}
		}
	}







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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值