Java 学习笔记(15)——反射
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种:
- 可以通过 Class 类的静态方法 forName 传入类名获取
- 可以通过具体对象的getClass 方法获取,这种方式的前提是我们拿到了目标对象,也就是需要内存中已经加载了对应的对象,相对来说第一种方法相对方便。
- 通过类的静态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类:
- 类必须被public 修饰
- 类必须提供对应的getter 与 setter方法
- 类必须提供空参的构造方法
- 成员变量必须用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)——反射的更多相关文章
- java学习笔记:反射
1.什么是反射? Reflection(反射)是被视为动态语言的关键,反射机制允许程序做执行期间借助于ReflectionAPI取得任何类的内部信息,并能直接操作任意对象内部属性及方法 2.反射相关的 ...
- 【java学习笔记】反射基础
一.反射 反射就是在剖析一个类,了解这个类的构造,创建这个类对应的对象. Class 代表字节码的类,代表类的类 Field 代表属性的类 Method 代表方法的类 Constructor 代表构造 ...
- 8.2(java学习笔记)反射
一.基础知识 对象是表示或封装一些数据,一个类被加载后JVM会创建一个对应该类的Class对象, 类的整个结构信息会被放在对应的对象中,通过这个对象我们可以获取改类的全部信息, 而这些操作称为反射. ...
- java学习笔记之反射—Class类实例化和对象的反射实例化
反射之中所有的核心操作都是通过Class类对象展开的,可以说Class类是反射操作的根源所在,但是这个类的实例化对象,可以采用三种方式完成. java.lang.Class类的定义: public f ...
- Java学习笔记15
do-while循环时while循环的变体语法如下:do{ // 循环体 语句(组);}while(循环继续条件); 如果循环中的语句至少需要执行一次,那么建议使用do-while循环. for循环 ...
- Thinking In Java 学习笔记 1-5 章
第1章 对象导论 本章主要讲OOP的思想及一些OOP基本概念 1.抽象过程:万物都是对象,对象具有状态.行为和标识.对象拥有属性和方法,以及在内存中的唯一地址. 2.每个对象都有一个接口:通过接口给对 ...
- java学习笔记15(String 类,StringBuffer)
/* * String类的特点: * 所有的""都是String的对象 * 字符串一旦创建就是常量,不能改变 */ public class StringDemo { public ...
- java学习笔记之反射—反射和工厂模式
简单工厂模式又称为静态工厂方法模式,它是由工厂对象来决定要创建哪一种类的实例化对象. 静态工厂代码: class Factory{ private Factory() {} public static ...
- Java学习笔记15(面向对象八:匿名对象、内部类)
匿名对象: 是指创建对象的时候,只有创建对象的语句,却没有把对象地址值赋给某个变量 创建一个普通对象: Person p = new Person(); 创建一个匿名对象: new Person(); ...
随机推荐
- SPSS函数之期和时间函数
SPSS函数之期和时间函数 CTIME.DAYS(timevalue)数值.返回 timevalue 中的天数(包括有小数位的天数),timevalue 必须为时间格式的数值或表达式,如 TIME.x ...
- hdu1532&&poj1273 最大流
Dinic算法: 层次图:根据源点到该点的距离建图,这里设相邻的都差1. (下面部分转) 在这幅图中我们首先要增广1->2->4->6,这时可以获得一个容量为2的流,但是如果不建立4 ...
- el标签 2016-06-05 21:39 477人阅读 评论(15) 收藏
JSP EL语言定义 E L(Expression Language) 目的:为了使JSP写起来更加简单. 表达式语言的灵感来自于 ECMAScript 和 XPath 表达式语言,它提供了在 JSP ...
- Calendar类实现当前日期的日历
package com.sxt.home1; import java.text.DateFormat; import java.text.ParseException; import java.tex ...
- laravel 随笔
laravel5.5 1.laravel 查询数据库默认返回对象,如何改成 返回值为数组 答:在 App\Providers\EventServiceProvider 文件中 第一步: use Il ...
- 17-3 cookie和session
一 . Cookie 1.cookie 是什么? 保存在浏览器端的键值对! 服务端在返回响应的时候,告诉浏览器保存的键值对!浏览器可以拒绝保存Cookie. 2. 为什么要有cookie? HTTP请 ...
- 无旋treap hfq-treap
怎么代码都这么长... #include<iostream> #include<stdio.h> #include<string.h> #include<al ...
- 2018-8-10-win10-uwp-如何打包Nuget给其他人
title author date CreateTime categories win10 uwp 如何打包Nuget给其他人 lindexi 2018-08-10 19:16:50 +0800 20 ...
- oracle避免在索引列上使用IS NULL和IS NOT NULL
避免在索引中使用任何可以为空的列,ORACLE将无法使用该索引 .对于单列索引,如果列包含空值,索引中将不存在此记录. 对于复合索引,如果每个列都为空,索引中同样不存在此记录. 如果至少有一个列不为空 ...
- CODE FESTIVAL 2017 qual A D Four Coloring(补题)
这题看了好几天才看懂,一直误解题解中的d * d了 题解中说把大的格子划分成d * d的方格,我划分的时候把格子当作点来算的,一直觉得那明明是(d-1) * (d-1),昨天刚反映过来 思路:把格子旋 ...