- 浏览: 890268 次
- 性别:
- 来自: 北京
文章分类
- 全部博客 (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 1326操作系统的目标 目前存在着多种类型的OS,不同类型的OS ... -
利用(ffmpeg)生成视频缩略图(java)
2012-07-01 01:11 0对于上传视频生成缩略图使用的是ffmpeg进行生成的。 自己在 ... -
对Java多线程技术中所有方法的详细解析
2012-06-06 11:32 782一、run()和start() 这两个 ... -
java乱码
2012-06-06 11:33 967自从接触Java和JSP以来, ... -
学习apache commons-io类库中的文件清除器
2011-07-06 23:26 1439学习apache commons-io 1.4类库中的File ... -
java 正则表达式 过滤html标签
2011-05-24 15:10 5352前段时间开发的时候要读取一篇文章的简介内容(也就是前200个字 ... -
转---Eclipse中web-inf和meta-inf文件夹的信息
2011-05-24 13:08 1131Eclipse中web-inf和meta-inf ... -
logback与Log4J的区别
2011-05-17 23:34 1432Logback和log4j是非常相似 ... -
性能优化
2011-04-14 16:10 1172(1)jdbc性能优化 jdbc程序的性能主要由两个因素决定 ... -
JAVA的Random类(转)
2011-04-12 00:21 906Random类中实现的随机算法是伪随机,也就是有规则的随机。在 ... -
非阻塞的Socket链接
2011-04-10 21:59 888import java.io.IOException; ... -
创建临时文件
2011-04-10 21:55 1090package net.java2000.io; ... -
面向对象设计的基本原则
2011-04-07 10:28 1142摘自:http://soft6.com/tech/6/6501 ... -
proxool
2011-04-02 15:01 855属性列表说明: fatal-sql- ... -
当前Java软件开发中几种认识误区
2011-04-01 10:12 875越来越多人开始使用Java ... -
Java中查看一个方法被调用的层次(Reflection、StackTrace)
2011-04-01 00:53 2030package test; public class Mai ... -
反序列化时恢复transient字段
2011-03-30 13:20 1211我们知道将字段设置为transient,可以避免该自动被序列化 ... -
用socket连接服务器直接发送接收邮件
2011-03-22 17:22 1300首页 新闻 论坛 问答 博客 招聘 更多 ▼ 专栏 &l ... -
利用JavaMail收/发Gmail邮件(SSL)
2011-03-22 17:21 2384Gmail目前已经启用了POP3和SMTP服务,具体情况请看 ... -
关于jdk的注解@SuppressWarnings详解
2011-03-01 15:23 1493@SuppressWarnings J2SE 提供的一个批注或 ...
相关推荐
51jobduoyehtml爬虫程序代码QZQ2
白色大气风格的商务英语学习培训网站模板.zip
锡林郭勒市五险一金办事指南
警务处内务规定
白色扁平化风格的设计创业公司模板下载.zip
白色大气风格的土建设计公司模板下载.zip
白色大气风格的响应式CSS3模板下载.zip
白色大气风格的个人简历网页模板下载.zip
白色调简洁的咨询公司整站模板下载.zip
1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于计算机科学与技术等相关专业,更为适合;
Origin教程008:热图所需练习数据
白色扁平化风格的企业创意联系方式模板下载.zip
白色大气风格的欧美穿衣风格网站模板下载.zip
白色干净大气的企业网站模板下载.zip
内容概要:本文介绍了一段简单的Python代码,用于在控制台中输出一棵带有装饰的圣诞树。具体介绍了代码结构与逻辑,包括如何计算并输出树形的各层,如何加入装饰元素以及打印树干。还提供了示例装饰字典,允许用户自定义圣诞树装饰位置。 适用人群:所有对Python编程有一定了解的程序员,尤其是想要学习控制台图形输出的开发者。 使用场景及目标:适用于想要掌握如何使用Python代码创建控制台艺术,特别是对于想要增加节日氛围的小项目。目标是帮助开发者理解和实现基本的字符串操作与格式化技巧,同时享受创造乐趣。 其他说明:本示例不仅有助于初学者理解基本的字符串处理和循环机制,而且还能激发学习者的编程兴趣,通过调整装饰物的位置和树的大小,可以让输出更加个性化和丰富。
1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于计算机科学与技术等相关专业,更为适合;
白色大气风格响应式通用后台管理网站模板.zip
白色风格的广告设计公司网站模板下载.rar
2008-2020年各省技术服务水平相关指标数据 1.时间:2008-2020年 2.指标:行政区划代码、地区、年份、信息传输、软件和信息技术服务业城镇单位就业人员(万人)、软件业务收入(万元)、高技术产品出口额占商品出口额比重(%) 3.范围:31省 4.来源:科技年鉴、国家统计J
1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于计算机科学与技术等相关专业,更为适合;