- 浏览: 887437 次
- 性别:
- 来自: 北京
文章分类
- 全部博客 (687)
- java (127)
- servlet (38)
- struts (16)
- spring (22)
- hibernate (40)
- javascript (58)
- jquery (18)
- tomcat (51)
- 设计模式 (6)
- EJB (13)
- jsp (3)
- oracle (29)
- RUP (2)
- ajax (3)
- java内存管理 (4)
- java线程 (12)
- socket (13)
- path (5)
- XML (10)
- swing (2)
- UML (1)
- JBPM (2)
- 开发笔记 (45)
- Note参考 (15)
- JAXB (4)
- Quartz (2)
- 乱码 (2)
- CSS (2)
- Exception (4)
- Tools (7)
- sqlserver (3)
- DWR (7)
- Struts2 (47)
- WebService (2)
- 问题解决收藏 (7)
- JBOSS (7)
- cache (10)
- easyUI (19)
- jQuery Plugin (11)
- FreeMarker (6)
- Eclipse (2)
- Compass (2)
- JPA (1)
- WebLogic (1)
- powerdesigner (1)
- mybatis (1)
最新评论
-
bugyun:
受教了,谢谢
java 正则表达式 过滤html标签 -
xiongxingxing_123:
学习了,感谢了
java 正则表达式 过滤html标签 -
wanmeinange:
那如果无状态的。对同一个任务并发控制怎么做?比如继承Quart ...
quartz中参数misfireThreshold的详解 -
fanjieshanghai:
...
XPath 元素及属性查找 -
tianhandigeng:
还是没明白
quartz中参数misfireThreshold的详解
一、java反射机制
JAVA反射机制是在运行状态中,
对于任意一个类,都能够得到这个类的所有属性和方法;
对于任意一个对象,都能够调用它的任意一个方法;
这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制.
概括一下:
反射就是让你可以通过名称来得到对象(类,属性,方法)的技术。
例如我们可以通过类名来生成一个类的实例;
知道了方法名,就可以调用这个方法;
知道了属性名就可以访问这个属性的值。
1、获取类对应的Class对象
运用(已知对象)getClass():Object类中的方法,每个类都拥有此方法。
如: String str = new String();
Class strClass = str.getClass();
运用(已知子类的class) Class.getSuperclass():Class类中的方法,返回该Class的父类的Class;
运用(已知类全名):Class.forName() 静态方法
运用(已知类): 类名.class
2、通过类名来构造一个类的实例
a、调用无参的构造函数:
Class newoneClass = Class.forName(类全名);
newoneClass.newInstance();
b、调用有参的构造函数:我们可以自定义一个函数。
public Object newInstance(String className, Object[] args) throws Exception {
//args为参数数组
Class newoneClass = Class.forName(className);
//得到参数的Class数组(每个参数的class组成的数组),由此来决定调用那个构造函数
Class[] argsClass = new Class[args.length];
for (int i = 0, j = args.length; i < j; i++) {
argsClass[i] = args[i].getClass();
}
Constructor cons = newoneClass.getConstructor(argsClass); //根据argsClass选择函数
return cons.newInstance(args); //根据具体参数实例化对象。
}
3、得到某个对象的属性
a、非静态属性:首先得到class,然后得到这个class具有的field,然后以具体实例为参数调用这个field
public Object getProperty(Object owner, String fieldName) throws Exception {
Class ownerClass = owner.getClass();//首先得到class
Field field = ownerClass.getField(fieldName);
//然后得到这个class具有的field,也可以通过getFields()得到所有的field
Object property = field.get(owner);
//owner指出了取得那个实例的这个属性值,如果这个属性是非公有的,这里会报IllegalAccessException。
return property;
}
b、静态属性:
只有最后一步不同,由于静态属性属于这个类,所以只要在这个类上调用这个field即可
Object property = field.get(ownerClass);
4、执行某对象的方法
public Object invokeMethod(Object owner, String methodName, Object[] args) throws Exception {
Class ownerClass = owner.getClass(); //也是从class开始的
//得到参数的class数组,相当于得到参数列表的类型数组,来取决我们选择哪个函数。
Class[] argsClass = new Class[args.length];
for (int i = 0, j = args.length; i < j; i++) {
argsClass[i] = args[i].getClass();
}
//根据函数名和函数类型来选择函数
Method method = ownerClass.getMethod(methodName, argsClass);
return method.invoke(owner, args);//具体实例下,具体参数值下调用此函数
}
5、执行类的静态方法
和上面的相似只是最后一行不需要指定具体实例
return method.invoke(null, args);
6、判断是否为某个类的实例
public boolean isInstance(Object obj, Class cls) {
return cls.isInstance(obj);
}
测试bean类:SimpleBean.java
- package com.royzhou.bean;
- public class SimpleBean {
- private String name;
- private String[] hobby;
- public SimpleBean() {
- }
- public SimpleBean(String name, String[] hobby) {
- this.name = name;
- this.hobby = hobby;
- }
- public void setName(String name) {
- this.name = name;
- }
- public String getName() {
- return this.name;
- }
- public void setHobby(String[] hobby) {
- this.hobby = hobby;
- }
- public String[] getHobby() {
- return this.hobby;
- }
- public String toString() {
- String returnValue = super.toString() + "\n";
- returnValue += "name:=" + this.name + "\n";
- if(this.hobby != null) {
- returnValue += "hobby:";
- for(String s : this.hobby) {
- returnValue += s + ",";
- }
- returnValue += "\n";
- }
- return returnValue;
- }
- }
package com.royzhou.bean; public class SimpleBean { private String name; private String[] hobby; public SimpleBean() { } public SimpleBean(String name, String[] hobby) { this.name = name; this.hobby = hobby; } public void setName(String name) { this.name = name; } public String getName() { return this.name; } public void setHobby(String[] hobby) { this.hobby = hobby; } public String[] getHobby() { return this.hobby; } public String toString() { String returnValue = super.toString() + "\n"; returnValue += "name:=" + this.name + "\n"; if(this.hobby != null) { returnValue += "hobby:"; for(String s : this.hobby) { returnValue += s + ","; } returnValue += "\n"; } return returnValue; } }
反射测试类:ReflectTest.java
- package com.royzhou.bean;
- import java.lang.reflect.Constructor;
- import java.lang.reflect.Field;
- import java.lang.reflect.Method;
- public class ReflectTest {
- public static void main(String[] args) throws Exception {
- Class clazz = SimpleBean.class;
- //使用无参构造函数实例化bean
- SimpleBean sb = (SimpleBean)clazz.newInstance();
- System.out.println(sb);
- //使用有参构造函数实例化bean
- Constructor constructor = clazz.getConstructor(new Class[]{String.class, String[].class});
- sb = (SimpleBean)constructor.newInstance(new Object[]{"royzhou",new String[]{"football","basketball"}});
- System.out.println(sb);
- //为name字段设置值
- Field field = clazz.getDeclaredField("name");
- field.setAccessible(true); //避免private不可访问抛出异常
- field.set(sb, "royzhou1985");
- System.out.println("modify name using Field:=" + sb.getName() + "\n");
- //列出类SimpleBean的所有方法
- Method[] methods = clazz.getDeclaredMethods();
- System.out.println("get methods of class SimpleBean:");
- for(Method method : methods) {
- if("setHobby".equals(method.getName())) {
- //动态调用类的方法来为hobby设置值
- method.invoke(sb, new Object[]{new String[]{"tennis","fishing"}});
- }
- System.out.println(method.getName());
- }
- System.out.println("\nset by invoke Method");
- System.out.println(sb);
- }
- }
package com.royzhou.bean; import java.lang.reflect.Constructor; import java.lang.reflect.Field; import java.lang.reflect.Method; public class ReflectTest { public static void main(String[] args) throws Exception { Class clazz = SimpleBean.class; //使用无参构造函数实例化bean SimpleBean sb = (SimpleBean)clazz.newInstance(); System.out.println(sb); //使用有参构造函数实例化bean Constructor constructor = clazz.getConstructor(new Class[]{String.class, String[].class}); sb = (SimpleBean)constructor.newInstance(new Object[]{"royzhou",new String[]{"football","basketball"}}); System.out.println(sb); //为name字段设置值 Field field = clazz.getDeclaredField("name"); field.setAccessible(true); //避免private不可访问抛出异常 field.set(sb, "royzhou1985"); System.out.println("modify name using Field:=" + sb.getName() + "\n"); //列出类SimpleBean的所有方法 Method[] methods = clazz.getDeclaredMethods(); System.out.println("get methods of class SimpleBean:"); for(Method method : methods) { if("setHobby".equals(method.getName())) { //动态调用类的方法来为hobby设置值 method.invoke(sb, new Object[]{new String[]{"tennis","fishing"}}); } System.out.println(method.getName()); } System.out.println("\nset by invoke Method"); System.out.println(sb); } }
输出结果:
com.royzhou.bean.SimpleBean@757aef
name:=null
com.royzhou.bean.SimpleBean@d9f9c3
name:=royzhou
hobby:football,basketball,
modify name using Field:=royzhou1985
get methods of class SimpleBean:
setHobby
getHobby
getName
toString
setName
set by invoke Method
com.royzhou.bean.SimpleBean@d9f9c3
name:=royzhou1985
hobby:tennis,fishing,
二、java内省机制
内省是 Java 语言对 Bean 类属性、事件的一种处理方法(也就是说给定一个javabean对象,我们就可以得到/调用它的所有的get/set方法)。
例如类 A 中有属性 name, 那我们可以通过 getName,setName 来得到其值或者设置新的值。通过 getName/setName 来访问 name 属性,这就是默认的规则。
Java 中提供了一套 API 用来访问某个属性的 getter/setter 方法,通过这些 API 可以使你不需要了解这个规则,这些 API 存放于包 java.beans 中。
一般的做法是通过类 Introspector 的 getBeanInfo方法 来获取某个对象的 BeanInfo 信息,然后通过 BeanInfo 来获取属性的描述器(PropertyDescriptor),
通过这个属性描述器就可以获取某个属性对应的 getter/setter 方法,然后我们就可以通过反射机制来调用这些方法。
内省测试类:Introspector.java
- package com.royzhou.bean;
- import java.beans.BeanInfo;
- import java.beans.Introspector;
- import java.beans.MethodDescriptor;
- import java.beans.PropertyDescriptor;
- public class IntrospectorTest {
- public static void main(String[] args) throws Exception {
- SimpleBean sb = new SimpleBean("royzhou",new String[]{"football","backetball"});
- System.out.println(sb);
- /**
- * 使用Introspector.getBeanInfo(SimpleBean.class)将Bean的属性放入到BeanInfo中。
- * 第二个参数为截止参数,表示截止到此类之前,不包括此类。
- * 如果不设置的话,那么将会得到本类以及其所有父类的info。
- * BeanInfo中包含了SimpleBean的信息
- */
- BeanInfo beanInfo = Introspector.getBeanInfo(sb.getClass());
- //将每个属性的信息封装到一个PropertyDescriptor形成一个数组其中包括属性名字,读写方法,属性的类型等等
- PropertyDescriptor[] propertys = beanInfo.getPropertyDescriptors();
- for(PropertyDescriptor property : propertys) {
- System.out.println("属性名:" + property.getName());
- System.out.println("类型:" + property.getPropertyType());
- }
- System.out.println();
- System.out.println("列出SimpleBean的所有方法");
- MethodDescriptor[] methods = beanInfo.getMethodDescriptors();
- for(MethodDescriptor method : methods) {
- System.out.println(method.getName());
- }
- System.out.println();
- /**
- * 重新设置属性值
- */
- for(PropertyDescriptor property : propertys) {
- if(property.getPropertyType().isArray()){ //getPropertyType得到属性类型。
- if(property.getPropertyType().isArray()) {
- if("hobby".equals(property.getName())) {
- //getComponentType()可以得到数组类型的元素类型
- if(property.getPropertyType().getComponentType().equals(String.class)) {
- //getWriteMethod()得到此属性的set方法----Method对象,然后用invoke调用这个方法
- property.getWriteMethod().invoke(sb, new Object[]{new String[]{"tennis","fishing"}});
- }
- }
- }
- } else if("name".equals(property.getName())) {
- property.getWriteMethod().invoke(sb, new Object[] { "royzhou1985" });
- }
- }
- /**
- * 获取对象的属性值
- */
- System.out.println("获取对象的属性值");
- for(PropertyDescriptor property : propertys) {
- if(property.getPropertyType().isArray()){ //getPropertyType得到属性类型。
- //getReadMethod()得到此属性的get方法----Method对象,然后用invoke调用这个方法
- String[] result=(String[]) property.getReadMethod().invoke(sb, new Object[]{});
- System.out.print(property.getName()+":");//getName得到属性名字
- for (int j = 0; j < result.length; j++) {
- System.out.print(result[j] + ",");
- }
- System.out.println();
- }
- else{
- System.out.println(property.getName()+":"+property.getReadMethod().invoke(sb, new Object[]{}));
- }
- }
- }
- }
package com.royzhou.bean; import java.beans.BeanInfo; import java.beans.Introspector; import java.beans.MethodDescriptor; import java.beans.PropertyDescriptor; public class IntrospectorTest { public static void main(String[] args) throws Exception { SimpleBean sb = new SimpleBean("royzhou",new String[]{"football","backetball"}); System.out.println(sb); /** * 使用Introspector.getBeanInfo(SimpleBean.class)将Bean的属性放入到BeanInfo中。 * 第二个参数为截止参数,表示截止到此类之前,不包括此类。 * 如果不设置的话,那么将会得到本类以及其所有父类的info。 * BeanInfo中包含了SimpleBean的信息 */ BeanInfo beanInfo = Introspector.getBeanInfo(sb.getClass()); //将每个属性的信息封装到一个PropertyDescriptor形成一个数组其中包括属性名字,读写方法,属性的类型等等 PropertyDescriptor[] propertys = beanInfo.getPropertyDescriptors(); for(PropertyDescriptor property : propertys) { System.out.println("属性名:" + property.getName()); System.out.println("类型:" + property.getPropertyType()); } System.out.println(); System.out.println("列出SimpleBean的所有方法"); MethodDescriptor[] methods = beanInfo.getMethodDescriptors(); for(MethodDescriptor method : methods) { System.out.println(method.getName()); } System.out.println(); /** * 重新设置属性值 */ for(PropertyDescriptor property : propertys) { if(property.getPropertyType().isArray()){ //getPropertyType得到属性类型。 if(property.getPropertyType().isArray()) { if("hobby".equals(property.getName())) { //getComponentType()可以得到数组类型的元素类型 if(property.getPropertyType().getComponentType().equals(String.class)) { //getWriteMethod()得到此属性的set方法----Method对象,然后用invoke调用这个方法 property.getWriteMethod().invoke(sb, new Object[]{new String[]{"tennis","fishing"}}); } } } } else if("name".equals(property.getName())) { property.getWriteMethod().invoke(sb, new Object[] { "royzhou1985" }); } } /** * 获取对象的属性值 */ System.out.println("获取对象的属性值"); for(PropertyDescriptor property : propertys) { if(property.getPropertyType().isArray()){ //getPropertyType得到属性类型。 //getReadMethod()得到此属性的get方法----Method对象,然后用invoke调用这个方法 String[] result=(String[]) property.getReadMethod().invoke(sb, new Object[]{}); System.out.print(property.getName()+":");//getName得到属性名字 for (int j = 0; j < result.length; j++) { System.out.print(result[j] + ","); } System.out.println(); } else{ System.out.println(property.getName()+":"+property.getReadMethod().invoke(sb, new Object[]{})); } } } }
输出结果:
com.royzhou.bean.SimpleBean@757aef
name:=royzhou
hobby:football,backetball,
属性名:class
类型:class java.lang.Class
属性名:hobby
类型:class [Ljava.lang.String;
属性名:name
类型:class java.lang.String
列出SimpleBean的所有方法
hashCode
setHobby
equals
wait
wait
notify
getClass
toString
notifyAll
getHobby
setName
wait
getName
获取对象的属性值
class:class com.royzhou.bean.SimpleBean
hobby:tennis,fishing,
name:royzhou1985
三、总结
将 Java 的反射以及内省应用到程序设计中去可以大大的提供程序的智能化和可扩展性,有很多项目都是采取这两种技术来实现其核心功能,
其实应该说几乎所有的项目都或多或少的采用这两种技术,在实际应用过程中二者要相互结合方能发挥真正的智能化以及高度可扩展性。
发表评论
-
操作系统的目标和作用
2012-07-05 23:46 1314操作系统的目标 目前存在着多种类型的OS,不同类型的OS ... -
利用(ffmpeg)生成视频缩略图(java)
2012-07-01 01:11 0对于上传视频生成缩略图使用的是ffmpeg进行生成的。 自己在 ... -
对Java多线程技术中所有方法的详细解析
2012-06-06 11:32 775一、run()和start() 这两个 ... -
java乱码
2012-06-06 11:33 961自从接触Java和JSP以来, ... -
学习apache commons-io类库中的文件清除器
2011-07-06 23:26 1429学习apache commons-io 1.4类库中的File ... -
java 正则表达式 过滤html标签
2011-05-24 15:10 5346前段时间开发的时候要读取一篇文章的简介内容(也就是前200个字 ... -
转---Eclipse中web-inf和meta-inf文件夹的信息
2011-05-24 13:08 1126Eclipse中web-inf和meta-inf ... -
logback与Log4J的区别
2011-05-17 23:34 1420Logback和log4j是非常相似 ... -
性能优化
2011-04-14 16:10 1162(1)jdbc性能优化 jdbc程序的性能主要由两个因素决定 ... -
JAVA的Random类(转)
2011-04-12 00:21 903Random类中实现的随机算法是伪随机,也就是有规则的随机。在 ... -
非阻塞的Socket链接
2011-04-10 21:59 886import java.io.IOException; ... -
创建临时文件
2011-04-10 21:55 1087package net.java2000.io; ... -
面向对象设计的基本原则
2011-04-07 10:28 1139摘自:http://soft6.com/tech/6/6501 ... -
proxool
2011-04-02 15:01 850属性列表说明: fatal-sql- ... -
当前Java软件开发中几种认识误区
2011-04-01 10:12 871越来越多人开始使用Java ... -
Java中查看一个方法被调用的层次(Reflection、StackTrace)
2011-04-01 00:53 2025package test; public class Mai ... -
反序列化时恢复transient字段
2011-03-30 13:20 1209我们知道将字段设置为transient,可以避免该自动被序列化 ... -
用socket连接服务器直接发送接收邮件
2011-03-22 17:22 1295首页 新闻 论坛 问答 博客 招聘 更多 ▼ 专栏 &l ... -
利用JavaMail收/发Gmail邮件(SSL)
2011-03-22 17:21 2378Gmail目前已经启用了POP3和SMTP服务,具体情况请看 ... -
关于jdk的注解@SuppressWarnings详解
2011-03-01 15:23 1489@SuppressWarnings J2SE 提供的一个批注或 ...
相关推荐
Java反射与内省是Java编程语言中的核心特性,它们为开发者提供了强大的能力,可以在运行时检查和操作类、接口、对象及其属性。本PPT深入探讨了这些概念,旨在帮助开发者充分利用Java平台的灵活性。 首先,Java反射...
反射机制的分类包括反射 API、反射与数组、反射与泛型、反射源码与性能开销、反射优缺点、反射与内省等。 反射 API 是 Java 语言提供的一组 Application Programming Interface(API),用于在运行时获取类的信息和...
### Java中的内省与反射详解 #### 一、引言 在深入了解Java语言的过程中,我们会遇到两个重要的概念:反射和内省。这两种技术在实际开发中有着广泛的应用,尤其是在需要进行动态操作或元数据处理的场景下尤为重要...
### Java反射机制详解 #### 一、引言 在Java面试中,经常会出现与反射机制相关的题目。这是因为Java反射机制不仅是Java语言的一项重要特性,也是理解Java动态特性的关键所在。通过本文,我们将深入探讨Java反射...
本篇文章将深入探讨如何利用反射和内省技术来实现一个简单的Spring IOC。首先,我们来理解什么是反射和内省。 反射是Java提供的一种强大的动态类型机制,它允许程序在运行时检查类的信息,如类名、属性、方法等,并...
Java反射机制是Java编程语言中的一个重要特性,它允许程序在运行时获取和操作任何已知名称的类的内部信息。这一机制使得Java具备了一定的动态性,虽然在传统的分类中Java被视为静态类型语言。通过反射,开发者可以在...
"JAVA的内省机制(introspector)与反射机制(reflection)" JAVA 的内省机制(introspector)和反射机制(reflection)是两个重要的概念,在 JAVA 编程中扮演着至关重要的角色。那么,什么是内省机制和反射机制?它们...
### Java反射机制详解 #### 一、引言 Java反射机制是Java语言中一项重要的功能,使得Java在一定程度上具备了动态语言的特性。通过Java反射机制,可以在运行时获取类的信息并操作类的对象,这为Java应用开发带来了...
Java反射机制是Java被视为动态语言的关键特性之一,它允许程序在运行时通过Reflection APIs获取任何已知名称的类的内部信息。这一机制使得开发者能够在运行时检查类的修饰符(如public、static等)、超类(如Object...
在这个"酷炫javaweb后台系统"中,包含了一些核心的JavaWeb技术和编程理念,如封装、反射和内省等。这些概念是Java编程的重要组成部分,对于理解和构建高效的应用程序至关重要。 首先,让我们来探讨一下"封装"。封装...
Java IOC(Inversion of Control,控制反转)和AOP(Aspect-Oriented ...在源码和工具方面,我们可以学习到如何利用Java的内省API进行类信息的获取,以及如何利用反射和动态代理来实现依赖注入和面向切面编程。
内省在实际开发中有着广泛的应用,例如在框架中进行数据绑定、属性的动态修改、序列化和反序列化,以及XML与Java对象之间的映射等。了解并熟练掌握Java内省机制,可以提高代码的灵活性和可维护性,尤其是在处理基于...
Java反射机制是一种在运行时查询、监视和操作类、方法、字段和对象的机制,其提供了一种强大的编程能力,允许程序在运行时进行自我检查和调整行为。Java Reflection in Action这本书作为Java反射机制的经典之作,...
在Java编程语言中,反射和内省是两个强大的特性,它们允许程序在运行时检查类、接口、字段和方法的信息,甚至动态地调用方法和访问属性。"minimalcode-reflect"项目提供了一个简约的框架,专注于JavaBean的反射和...
内省(Introspection)是Java编程语言中的一个重要特性,它允许程序在运行时检查对象的属性和方法,以及类的信息。在Java中,内省主要用于动态操作对象,特别是那些在设计时没有完全预知其行为的对象。内省机制使得...
Java反射API主要包括以下几个核心类: 1. **Class类**:每个类在JVM中都有一个对应的Class对象,它包含了类的所有信息。我们可以通过`Class.forName()`或者对象的`getClass()`方法来获取对应的Class对象。 2. **...