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(); ...
随机推荐
- Hdu 4810
2014-05-02 15:53:50 题目连接 2013年南京现场赛的题目,现场的时候,排在我们前面的队伍基本都过了这题,我们后面的队伍也有不少过了这题,唯独我们没有.. 后来是Qingyu Sha ...
- Maximum Depth of Binary Tree 树的最大深度
Given a binary tree, find its maximum depth. The maximum depth is the number of nodes along the long ...
- 【Leetcode链表】旋转链表(61)
题目 给定一个链表,旋转链表,将链表每个节点向右移动 k 个位置,其中 k 是非负数. 示例 1: 输入: 1->2->3->4->5->NULL, k = 2 输出: ...
- BZOJ 1008 越狱题解
其实这题很水,显然n个房间有m种宗教,总共有n^m种情况, 我们再考虑不合法的情况,显然第一个房间有m种情况,而后一种只有m-1种情况(因为不能相同) 所以不合法的情况有(m-1)^(n-1)*m种情 ...
- 让开发部署提速 8 倍,我参与贡献这款 IDE 插件的全过程
如何像参与开源那样,去参与一款 IDE 插件的设计? 作为一款 IDE 插件的使用者,我是否能决定下一个版本的功能? 自从产品经理银时小伙和他的开发小哥们在去年12月发布 Cloud Toolkit( ...
- 【C++】为什么构造函数没有返回值?(转载)
为什么构造函数没有返回值? 意见(1) 我认为构造函数隐含的返回值就是this,因为构造函数是在类的对象产生时自动调用.构造函数被调用也就意味着产生了一个对象,而this指针是与对象实体相关联的, ...
- Kubernetes1.3新特性:新的资源回收控制器
(一) 核心概念 在kubernetes1.3中新增了一个资源回收控制器GarbaseCollector,用这个控制器来替代kubernetes1.3中的资源回收控制器GC. 如下为kubernet ...
- HTTP Status 500 - java.lang.ClassNotFoundException: org.apache.jsp.register_jsp
你搜一下你的页面中是不是有<!---->的注释 去掉就好了 改成jsp的注释 1).JSP页面中的HTML注释 JSP页面中的HTML注释使用“<!—”和“-->”创建,它的具 ...
- Java练习 SDUT-1689_斐波那契?
斐波那契? Time Limit: 1000 ms Memory Limit: 32768 KiB Problem Description 给出一个数列的递推公式,希望你能计算出该数列的第N个数.递推 ...
- window执行python文件
@echo offD:cd D:\pythonstart python2 del.pyexit