(转)关于java内省机制的一些描述与说明

1).内省(introspector)是 java 语言对 bean 类属性、事件的一种缺省处理方法。例如类 a 中有属性 name, 那我们可以通过

getname,setname 来得到其值或者设置新的值。通过 getname/setname 来访问 name 属性,这就是默认的规则。 java

中提供了一套 api 用来访问某个属性的 getter/setter 方法,通过这些 api 可以使你不需要了解这个规则(但你最好还是要搞清楚),这些 api

存放于包 java.beans 中。

2).直接通过属性的描述器java.beans.propertydescriptor类,来访问属性的getter/setter 方法;

相关代码:

public class point {

private integer x;

private integer y;

public point(integer x, integer y) {

super();

this.x = x;

this.y = y;

}

public integer getx() {

return x;

}

public void setx(integer x) {

this.x = x;

}

public integer gety() {

return y;

}

public void sety(integer y) {

this.y = y;

}

}

import java.beans.propertydescriptor;

import java.lang.reflect.method;

public class reflect {

public static void main(string[] args) throws exception {

point point = new point(2, 5);

string proname = "x";

getproperty(point, proname);

setproperty(point, proname);

}

private static void setproperty(point point, string proname) throws exception

{

propertydescriptor prodescriptor = new propertydescriptor(proname,

point.class);

method methodsetx = prodescriptor.getwritemethod();

methodsetx.invoke(point, 8);

system.out.println(point.getx());// 8

}

private static void getproperty(point point, string proname) throws exception

{

propertydescriptor prodescriptor = new propertydescriptor(proname,

point.class);

method methodgetx = prodescriptor.getreadmethod();

object objx = methodgetx.invoke(point);

system.out.println(objx);// 2

}

}

3).通过类 introspector 来获取某个对象的 beaninfo 信息,然后通过 beaninfo 来获取属性的描述器(

propertydescriptor ),通过这个属性描述器就可以获取某个属性对应的 getter/setter

方法,然后我们就可以通过反射机制来调用这些方法。

相关代码:

把2中的getproperty()修改成如下形式:

private static void getproperty(point point, string proname) throws exception

{

beaninfo beaninfo = introspector.getbeaninfo(point.getclass());

propertydescriptor[] prodescriptors =

beaninfo.getpropertydescriptors();

for(propertydescriptor prop: prodescriptors){

if(prop.getname().equals(proname)){

method methodgetx = prop.getreadmethod();

system.out.println(methodgetx.invoke(point));//8

break;

}

}

}

4).我们又通常把javabean的实例对象称之为值对象(value

object),因为这些bean中通常只有一些信息字段和存储方法,没有功能性方法。一个javabean类可以不当javabean用,而当成普通类用。javabean实际就是一种规范,当一个类满足这个规范,这个类就能被其它特定的类调用。一个类被当作javabean使用时,javabean的属性是根据方法名推断出来的,它根本看不到java类内部的成员变量(javabean的成员变量通常都是私有private的)。

5).除了反射用到的类需要引入外,内省需要引入的类如下所示,它们都属于java.beans包中的类,自己写程序的时候也不能忘了引入相应的包或者类。

import java.beans.beaninfo;

import java.beans.introspectionexception;

import java.beans.introspector;

import java.beans.propertydescriptor;

6).下面讲解一些开源的工具类beanutils,需要额外下载的,commons-beanutils.jar,要使用它还必须导入commons-logging.jar包,不然会出异常;

相关代码一:

public static void main(string[] args) throws exception {

point point = new point(2, 5);

string proname = "x";

beanutils.setproperty(point, proname, "8");

system.out.println(point.getx());// 8

system.out.println(beanutils.getproperty(point, proname));// 8

system.out.println(beanutils.getproperty(point,

proname).getclass().getname());// java.lang.string

beanutils.setproperty(point, proname, 8);

system.out.println(beanutils.getproperty(point,

proname).getclass().getname());// java.lang.string

}

//我们看到虽然属性x的类型是integer,但是我们设置的时候无论是integer还是string,beanutils的内部都是当成string来处理的。

相关代码二:

beanutils支持javabean属性的级联操作;

public static void main(string[] args) throws exception {

point point = new point(2, 5);//在point中加一个属性 private date birth = new

date();并产生setter/getter方法

string proname = "birth";

date date= new date();

date.settime(10000);

beanutils.setproperty(point, proname, date);

system.out.println(beanutils.getproperty(point, proname));

beanutils.setproperty(point, "birth.time", 10000);

system.out.println(beanutils.getproperty(point,

"birth.time"));//10000

}

//之所以可以 beanutils.setproperty(point, "birth.time",

10000);这样写,那是因为date类中有gettime()和settime()方法,即date类中相当于有time这个属性。

相关代码三:

beanutils和propertyutils对比:

public static void main(string[] args) throws exception {

point point = new point(2, 5);

string proname = "x";

beanutils.setproperty(point, proname, "8");

system.out.println(beanutils.getproperty(point, proname));//8

system.out.println(beanutils.getproperty(point,

proname).getclass().getname());//java.lang.string

// propertyutils.setproperty(point, proname, "8");//exception:argument type

mismatch

propertyutils.setproperty(point, proname, 8);

system.out.println(propertyutils.getproperty(point, proname));//8

system.out.println(propertyutils.getproperty(point,

proname).getclass().getname());//java.lang.integer

}

//beanutils它以字符串的形式对javabean进行转换,而propertyutils是以原本的类型对javabean进行操作。如果类型不对,就会有argument

type mismatch异常。

6).理解了相应的原理,那些现成的工具用起来就会更舒服,如beanutils与propertyutils工具。这两个工具设置属性的时候一个主要区别是propertyutils.getpropety方法获得的属性值的类型为该属性本来的类型,而beanutils.getproperty则是将该属性的值转换成字符串后才返回。

总结

web 开发框架 struts 中的 formbean 就是通过内省机制来将表单中的数据映射到类的属性上,因此要求 formbean 的每个属性要有

getter/setter 方法。但也并不总是这样,什么意思呢?就是说对一个 bean 类来讲,我可以没有属性,但是只要有 getter/setter

方法中的其中一个,那么 java 的内省机制就会认为存在一个属性,比如类中有方法 setmobile ,那么就认为存在一个 mobile 的属性。

将 java 的反射以及内省应用到程序设计中去可以大大的提供程序的智能化和可扩展性。有很多项目都是采取这两种技术来实现其核心功能,例如我们前面提到的

struts ,还有用于处理 xml 文件的 digester

项目,其实应该说几乎所有的项目都或多或少的采用这两种技术。在实际应用过程中二者要相互结合方能发挥真正的智能化以及高度可扩展性。


======================================================
在最后,我邀请大家参加新浪APP,就是新浪免费送大家的一个空间,支持PHP+MySql,免费二级域名,免费域名绑定 这个是我邀请的地址,您通过这个链接注册即为我的好友,并获赠云豆500个,价值5元哦!短网址是http://t.cn/SXOiLh我创建的小站每天访客已经达到2000+了,每天挂广告赚50+元哦,呵呵,饭钱不愁了,\(^o^)/
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值