Java学习之反射
来源:互联网 发布:复杂网络 pdf 编辑:程序博客网 时间:2024/06/11 19:05
反射,在开发项目时, 需要使用DBUtils,好奇心的驱使之下, 阅读了一下DBUtils的源码, 发现在将数据库结果集转换为JavaBean
就是通过反射实现的,但对反射没有一个较深入的了解!
一. 反射的概念
反射是指程序可以访问、检测和修改它本身状态或行为的一种能力,允许以编程方式访问有关加载类的字段,方法和构造函数的信息,
以及在安全限制内使用反射的字段,方法和构造函数对其底层进行操作。
反射是JDK 1.1引入的机制, 便于创建高度灵活的代码, 代码可以运行时装配,但是如果使用不当, 使用成本会很高,建议阅读DBUtils
源码先小试牛刀, 如果有兴趣, 可以继续阅读spring源码。
注意: 不要和内省的概念相混淆, 内省用于在运行时检测某个对象的类型和其包含的属性; 而反射除了检测还可以修改, 如C++只支
持内省, 但是不支持反射。
二. 反射机制的作用
1. 反编译: class文件反编译为Java文件
2. 通过反射机制访问bean的属性、方法、构造方法; 可以动态的根据传入的数据选择正确的类型接收处理;修改
构造函数、方法、属性的可见性;例如在Struts框架中, 框架确定处理请求的Action类之后, 就是通过反射机制来实现
的Action类的加载。
三、反射类
java.lang.Classjava.lang.reflect.Constructorjava.lang.reflect.Field java.lang.reflect.Methodjava.lang.reflect.Modifier
很多对象中的方法、属性等操作这四个类中查询。API才是成就霸业的军师。
四、实例
public class User { private int id; private int age; private String username; private String password; }
1. 通过反射机制从对象获取类
// 方式1Class userClass1 = Class.forName("User");// 方式2Class userClass2 = User.class;// 方式3Class userClass3 = new User().getClass();
2. 从已有对象创建新对象
User user01 = new User();User user02 = user01.getClass().newInstance();3. 获取对象所有属性
public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException { Class c = Class.forName("java.lang.String"); Field[] fs = c.getDeclaredFields(); StringBuffer sb = new StringBuffer(); sb.append(Modifier.toString(c.getModifiers()) + " class " + c.getSimpleName() + "{\n"); for (Field field : fs) { sb.append("\t");//tab sb.append(Modifier.toString(field.getModifiers()) + " "); sb.append(field.getType().getSimpleName() + " ");// 属性的类型 sb.append(field.getName() + ";\n");// 属性名 } sb.append("}"); System.out.println(sb); }
4. 获取对象单个属性
public static void main(String[] args) throws NoSuchFieldException, SecurityException, ClassNotFoundException, InstantiationException, IllegalAccessException { Class c = Class.forName("User"); Field username = c.getDeclaredField("username"); Object user = c.newInstance(); username.setAccessible(true); // 使用反射机制可以打破封装性,导致了java对象的属性不安全。 // username属性赋值"harry" username.set(user, "harry"); System.out.println(username.get(user)); }
5. 获取对象方法及调用方法(有参和无参)
public class MyReflect { public static void main(String[] args) { Method method; Task t = new Task(); try { // 1. 获取带参数的方法 method = t.getClass().getMethod("setAll", new Class<?>[]{int.class, String.class}); // 2. 调用带参数的方法 method.invoke(t, 10, "update"); // 1. 获取无参方法 method = t.getClass().getMethod("print", new Class<?>[0]); // 2. 调用无参方法 method.invoke(t); // 1. 获取无参方法 method = t.getClass().getMethod("getTaskComment", new Class<?>[0]); //method = t.getClass().getMethod("getTaskComment", null); // 使用这种方式时, 会产生警告 // 2. 调用方法并获取返回值 Object comment = method.invoke(t); System.out.println(comment); } catch (Exception e) { e.printStackTrace(); } } private static class Task { private int taskId; private String taskComment; public int getTaskId() { return taskId; } public void setTaskId(int taskId) { this.taskId = taskId; } public String getTaskComment() { return taskComment; } public void setTaskComment(String taskComment) { this.taskComment = taskComment; } public void print() { System.out.println(this.taskId + " " + this.taskComment); } public void setAll(int taskId, String taskComment) { this.taskComment = taskComment; this.taskId = taskId; } }}
6. 从类名字符串创建对象
public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException { Class<?> clazz = Class.forName("com.main.User"); User user = (User)clazz.newInstance(); user.print(); }
7. 获取构造方法并从构造方法创建对象
public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, SecurityException, IllegalArgumentException, InvocationTargetException { Class<?> clazz = Student.class; // 1. 从带参构造方法获取实例 Constructor<?> conMethod = clazz.getConstructor(new Class<?>[] { int.class, String.class }); Object o = conMethod.newInstance(20, "harry"); Method print = clazz.getMethod("print", new Class<?>[0]); print.invoke(o); // 2. 从无参构造方法获取实例 Constructor<?> conMethod1 = clazz.getConstructor(new Class<?>[0]); Object o1 = conMethod1.newInstance(); clazz.getMethod("setUsername", String.class).invoke(o1, "natasha"); System.out.println(clazz.getMethod("getUsername").invoke(o1)); // 3. 获取所有构造方法 Constructor<?> cons[] = clazz.getConstructors(); // 获取参数个数 System.out.println(cons[0].getParameterCount()); // 获取参数类型 System.out.println(cons[1].getParameters()[1]); // 通过构造方法创建对象, 并使用对象调用方法 Student s1 = (Student) cons[0].newInstance(); Student s2 = (Student) cons[1].newInstance(30, "joenas"); s2.print(); } private static class Student { private int id; private String username; public String getUsername() { return username; } public void setUsername(String username) { this.username = username; } public Student() { } public Student(int id, String username) { this.id = id; this.username = username; } public void print() { System.out.println(this.id + " " + this.username); } }
8. 获取实现的接口、父类、声明的属性等
public static void main(String[] args) throws NoSuchFieldException, SecurityException { Class<?> clazz = Student.class; // 1. 获取实现的接口 Class<?> inters[] = clazz.getInterfaces(); System.out.println(inters[0]); // 2. 获取父类 Class<?> parent = clazz.getSuperclass(); System.out.println(parent); // 3. 获取所有属性 Student stu = new Student(); Field fileds[] = clazz.getFields(); fileds = stu.getClass().getFields(); for (Field s : fileds) { System.out.println(s.getName()); } // 4. 获取单个属性 Field field = clazz.getField("username"); System.out.println(field.getName()); // 此行代码将报错, 不能获取private属性 field = clazz.getField("id"); } private static class Student implements Serializable{ private int id; public String username; public Student() { } public Student(int id, String username) { this.id = id; this.username = username; } public void print() { System.out.println(this.id + " " + this.username); } }
9. 使用反射修改数组大小
public static void main(String[] args) { int[] intArray = { 1, 2, 3, 4, 5 }; int[] newIntArray = (int[]) changeArraySize(intArray, 8); print(newIntArray); String[] atr = { "a", "b", "c", "d", "e" }; String[] str1 = (String[]) changeArraySize(atr, 8); print(str1); } /* * 修改数组大小, 原理为 * 1. 获取原数组的类型 * 2. 按照原类型新长度创建数组 * 3. 将原数组中数据赋值到新数组中 */ public static Object changeArraySize(Object obj, int len) { // 获取数组元素类型 Class<?> type = obj.getClass().getComponentType(); Object newArray = Array.newInstance(type, len); // 复制数组 int count = Array.getLength(obj); System.arraycopy(obj, 0, newArray, 0, count); return newArray; } // 输出数组内容 public static void print(Object obj) { Class<?> c = obj.getClass(); if (!c.isArray()) { return; } System.out.println("Array length: " + Array.getLength(obj)); System.out.print("["); for (int index = 0; index < Array.getLength(obj); index++) { System.out.print(Array.get(obj, index) + ","); } System.out.println("]"); }
当然、反射所能做的事情不限于如上所列, 还可以实现很多功能, 比如获取方法返回值类型,参数类型等; 在动态加载配置时反射也可以发挥大作用。
文章如有错误, 欢迎指正。
阅读全文
1 0
- Java学习之反射
- java学习之反射
- java学习之反射
- Java学习之反射
- Java学习之反射
- java学习之反射机制
- Java学习之-----反射机制
- java学习之反射机制
- Java学习总结之反射
- Java学习之反射机制
- Java学习之反射--Reflection
- Java学习笔记之反射
- Java学习笔记之反射
- Java学习日记之反射
- java学习笔记之反射
- Java学习之反射机制
- java学习之reflection反射
- Java学习笔记之反射
- iOS开发-国际化-配置App多语言
- 跨进程通信中AIDL文件的理解
- 经验 | 游戏人成长唯一有效的途径:接触更多更优秀的人
- IDE中配置文件导入和打开新文件的一些问题
- 打印hibernate的SQL语句的几种办法
- Java学习之反射
- js中__proto__和prototype的区别和关系?
- RTMP开发记录 测试服务器搭建篇
- 几种软件滤波方法
- 判断单链表有环的三种方法
- POJ3304---Segments (基础计算几何:叉积判断线段相交)
- java解析JSONArray对象
- 卷积神经网络实践
- Windows搭建GitHub