Java中的类文件最终会被编译为.class 文件,也就是Java字节码。这个字节码中会存储Java 类的相关信息。在JVM执行这些代码时首先根据 java 命令中指定的类名找到.class 文件然后使用类加载器将class字节码文件加载到内存,在加载时会调用Class类的classLoader方法,读取.class 文件中保存类的成员变量、方法、构造方法,并将这些内容在需要时创建对应的对象。这个就是java中的反射机制。反射机制提供了由字符串到具体类对象的映射,提高了程序的灵活性,在一些框架中大量使用映射,做到根据用户提供的xml配置文件来动态生成并创建类对象

反射机制最关键的就是从字节码文件中加载类信息并封装为对应的结构。在Java中专门提供了一个 Class 类,用于存储从.class 文件中读取出来的类的信息。 该类的定义和常用方法如下:

public final class Class<?> extends Object implements Serializable, GenericDeclaration, Type, AnnotatedElement{
String getName(); //获取类名
ClassLoader getClassLoader(); //返回类的加载器
static Class<T> forName(String className); //根据类名返回对应类的Class对象
Field getField(String name); //根据名称返回对应的Filed对象
Field[] getFields(); //返回所有的Filed 对象
Field getDeclaredField(String name) //;返回一个 Field对象。
Field[] getDeclaredFields();//返回的数组 Field对象 Constructor<T> getConstructor(Class<?>... parameterTypes);
Constructor<?>[] getConstructors();
Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes);
Constructor<?>[] getDeclaredConstructors(); Method getDeclaredMethod(String name, Class<?>... parameterTypes);
Method[] getDeclaredMethods();
Method getMethod(String name, Class<?>... parameterTypes);
Method[] getMethods();
}

获取 Class 对象

获取Class对象常见的有3种:

  1. 可以通过 Class 类的静态方法 forName 传入类名获取
  2. 可以通过具体对象的getClass 方法获取,这种方式的前提是我们拿到了目标对象,也就是需要内存中已经加载了对应的对象,相对来说第一种方法相对方便。
  3. 通过类的静态class 成员来获取。

下面是3中方式对应的代码

 //1. 使用forName 来获取
try {
Class<?> student = Class.forName("Student");
System.out.println(student.getName());
} catch (ClassNotFoundException e) {
e.printStackTrace();
} //2. 使用getClass 对象
Class<? extends Student> aClass = new Student().getClass();
System.out.println(aClass.getName()); //3. 使用class 静态变量
Class<Student> studentClass = Student.class;
System.out.println(studentClass.getName());

需要注意的是在每个进程中 一个类的 Class 只有一个,拿上面的代码来说,即使 我们获取了3次,在内存中只有一个对应的Class 对象。

获取类成员变量

通过一定的方法,我们已经获取到了对应的Class 成员,之前说过Class是对字节码中记录的类信息的封装,类的成员变量被封装到了Field对象中,我们可以使用上述4个与Field有关的方法来获取对应的成员变量。

public class Student {
public String name;
private int age;
public String sex;
private float gress;
} //main
try {
Class<?> student = Class.forName("Student");
Field name = student.getField("name");
System.out.println(name.getName()); Field[] fields = student.getFields();
for (Field field: fields) {
System.out.println(field.getName());
}
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (NoSuchFieldException e) {
e.printStackTrace();
}

上述的代码执行后会发现只能得到两个变量名,这两个函数只能获取 public 修饰的变量。要获取所有的可以使用 getDeclaredField(s) 函数组

获取类方法

Class 对象将类方法的信息封装到了 Method 对象中。我们可以使用 Method 对应的获取方法,同样的对应的Declared 方法能获取所有的,其他的只能获取公共的。

这次我们实现一个 给Java Bean对象赋值的通用类。

Java Bean是指满足这样一些条件的标准Java类:

  1. 类必须被public 修饰
  2. 类必须提供对应的getter 与 setter方法
  3. 类必须提供空参的构造方法
  4. 成员变量必须用private 修饰

为了方便代码的编写,针对Java bean对象的getter/setter 方法命名有一个规定,尽量使用 get + 成员变量名(第一个字母大写)的方式来命名。同时定义类的属性值是 getter/setter 方法名去掉get/set 并将剩余词第一个字母小写得到属性名。

针对这些定义,我们来实现一个根据字典值来给Java Bean赋值的方法。

//默认已经给上述的student类添加了对应的getter/setter 方法,并且为了方便将所有成员都改为String
static void BeanPopulate(Object bean, Map properties) throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
Iterator iterator = properties.entrySet().iterator();
while(iterator.hasNext()){
Map.Entry entry = (Map.Entry)iterator.next();
String key = (String) entry.getKey(); //首字母转大写
char[] chars = key.toCharArray();
chars[0] = (char) (chars[0] + ('A' - 'a'));
String name = "set" + new String(chars); Method method = bean.getClass().getMethod(name, String.class); //第二个参数是方法的参数列表
method.invoke(bean, entry.getValue()); ////第一个参数是对象,第二个是方法的参数列表
}
} Student student = new Student();
HashMap<String, String> map = new HashMap<>();
map.put("name", "tom");
map.put("age", "23");
map.put("sex", "男");
map.put("gress", "89.9");
try {
BeanPopulate(student, map);
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}

通过Class创建对象

上述的方法还有 Constructor 函数组没有说,这个函数组用来获取类的构造方法,有了这个方法,我们就可以创建对象了,那么我们将上面的例子给改一改,实现动态创建类并根据传入的map来设置值

static Object BeanPopulate(Class beanClass, Map properties) throws NoSuchMethodException, InvocationTargetException, IllegalAccessException, InstantiationException {
Iterator iterator = properties.entrySet().iterator();
Constructor constructor = beanClass.getConstructor(null); //构造方法的参数列表
Object bean = constructor.newInstance(null); //根据构造方法创建一个对象 while(iterator.hasNext()){
Map.Entry entry = (Map.Entry)iterator.next();
String key = (String) entry.getKey();
char[] chars = key.toCharArray();
chars[0] = (char) (chars[0] + ('A' - 'a'));
String name = "set" + new String(chars); Method method = bean.getClass().getMethod(name, String.class);
method.invoke(bean, entry.getValue());
}
return bean;
}

Java 学习笔记(15)——反射的更多相关文章

  1. java学习笔记:反射

    1.什么是反射? Reflection(反射)是被视为动态语言的关键,反射机制允许程序做执行期间借助于ReflectionAPI取得任何类的内部信息,并能直接操作任意对象内部属性及方法 2.反射相关的 ...

  2. 【java学习笔记】反射基础

    一.反射 反射就是在剖析一个类,了解这个类的构造,创建这个类对应的对象. Class 代表字节码的类,代表类的类 Field 代表属性的类 Method 代表方法的类 Constructor 代表构造 ...

  3. 8.2(java学习笔记)反射

    一.基础知识 对象是表示或封装一些数据,一个类被加载后JVM会创建一个对应该类的Class对象, 类的整个结构信息会被放在对应的对象中,通过这个对象我们可以获取改类的全部信息, 而这些操作称为反射. ...

  4. java学习笔记之反射—Class类实例化和对象的反射实例化

    反射之中所有的核心操作都是通过Class类对象展开的,可以说Class类是反射操作的根源所在,但是这个类的实例化对象,可以采用三种方式完成. java.lang.Class类的定义: public f ...

  5. Java学习笔记15

    do-while循环时while循环的变体语法如下:do{  // 循环体 语句(组);}while(循环继续条件); 如果循环中的语句至少需要执行一次,那么建议使用do-while循环. for循环 ...

  6. Thinking In Java 学习笔记 1-5 章

    第1章 对象导论 本章主要讲OOP的思想及一些OOP基本概念 1.抽象过程:万物都是对象,对象具有状态.行为和标识.对象拥有属性和方法,以及在内存中的唯一地址. 2.每个对象都有一个接口:通过接口给对 ...

  7. java学习笔记15(String 类,StringBuffer)

    /* * String类的特点: * 所有的""都是String的对象 * 字符串一旦创建就是常量,不能改变 */ public class StringDemo { public ...

  8. java学习笔记之反射—反射和工厂模式

    简单工厂模式又称为静态工厂方法模式,它是由工厂对象来决定要创建哪一种类的实例化对象. 静态工厂代码: class Factory{ private Factory() {} public static ...

  9. Java学习笔记15(面向对象八:匿名对象、内部类)

    匿名对象: 是指创建对象的时候,只有创建对象的语句,却没有把对象地址值赋给某个变量 创建一个普通对象: Person p = new Person(); 创建一个匿名对象: new Person(); ...

随机推荐

  1. TIJ——Chapter Three:Operators

    Operators 本章节比较简单,所以简单的做一些笔记: 几个要点: 1.When the compiler sees a String followed by a "+" fo ...

  2. React Native-组件的引用

    之前文章中,我们使用了许多React Native组件,也定义了一些组件.但是我们都没有定义组件的标识,我们都是通过回调方法处理组件对应的事件,这种情况能满足绝大多数需求,有些情况我们需要对组件进行操 ...

  3. 前端学习☞jquery

    一 什么是jQuery对象? jQuery 对象就是通过jQuery包装DOM对象后产生的对象.jQuery 对象是 jQuery 独有的. 如果一个对象是 jQuery 对象, 那么它就可以使用 j ...

  4. SGU 103 Traffic Lights【最短路】

    题目链接: http://acm.hust.edu.cn/vjudge/problem/visitOriginUrl.action?id=16530 题意: 给定每个点最初的颜色,最初颜色持续时间,以 ...

  5. 中断源记录 INT0 INT1

    中断源记录 INT0 INT1 用到一个单片机 使用的 P3.1 P3.3 作为唤醒口,后来发一 P3.1 和 P3.3 使用的同一个中断 INT1,这个尴尬了,只能两选 一. 查看规格书,还好 P3 ...

  6. oracle函数 VARIANCE([distinct|all]x)

    [功能]统计数据表选中行x列的方差. [参数]all表示对所有的值求方差,distinct只对不同的值求方差,默认为all 如果有参数distinct或all,需有空格与x(列)隔开. [参数]x,只 ...

  7. 选择阿里云数据库HBase版十大理由

    根据Gartner的预计,全球非关系型数据库(NoSQL)在2020~2022预计保持在30%左右高速增长,远高于数据库整体市场. 阿里云数据库HBase版也是踏着技术发展的节奏,伴随着NoSQL和大 ...

  8. PMC亮相IDF展示12G SAS分层存储解决方式

    引领大数据连接.传送以及存储,提供创新半导体及软件解决方式的PMC公司(纳斯达克代码:PMCS)出席了2014年4月2-3日在深圳举办的2014 IDF英特尔开发人员论坛. 此次,PMC将在 1层展示 ...

  9. HDFS概念名称节点和数据节点-基本模型

  10. iptables SNAT与伪装

    Source NAT(SNAT)的主要應用,是让同一內部網路上的多部主机,可共用同一条Internet实体连線.直接与Internet相连的闸道器,可使用SNAT(搭配连線追蹤)来来改写內部网络与In ...