A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

© whocases 黑马帝   /  2012-6-23 12:51  /  2009 人查看  /  5 人回复  /   0 人收藏 转载请遵从CC协议 禁止商业使用本文

本帖最后由 whocases 于 2012-6-24 21:43 编辑

看了内省,似懂非懂的,那么到底什么是内省呢?为什么要用内省呢?

5 个回复

倒序浏览
首先,得从JavaBean说起,javaBean是一种特殊的java类,主要用于传递数据信息,这种java类中的方法主要用于访问私有字段,且方法名符合某种规则。
如果两个模块之间传递多个信息,可以将这些信息封装到javaBean中,这种javaBean的实例对象称之为值得对象(Value Object  简称VO),这些信息在类中用私有字段来存储,如果读取或设置这些字段的值,则需要通过一些相应的方法来访问。javaBean的属性是根据其中的setter和getter方法来确定的,而不是根据其中的成员变量。如果方法名为setId()中文意思是设置id的值,至于你把它存储到那个变量上,不用管。如果方法为getId()表示获取id,至于是从哪个变量上获取的,也不重要。去掉set和get就是属性名,如果剩余部分的第二个字母是小写的,把剩余部分的首字母改成小写。
总之,一个类被当做javaBean使用时,javaBean的属性是根据其方法名推断出来的,他根本看不到类内部的成员变量。
一个符合javaBean特点的类可以当做普通类一样使用,但是把他当做javaBean使用时可定会带来一些好处。
在java EE 开发中经常熬使用到javaBean;很多环境就要求按JavaBean的方式进行操作。
JDK提供了对JavaBean进行操作的一些API。这套API就称为内省(IntroSpector).如果你要自己通过getxxx方法来访问私有的xxx属性有一定的难度。用内省这同API操作JavaBean比用普通类要便捷得多。
回复 使用道具 举报
内省是java语言对bean类属性、事件的一种处理方法
为什么要学内省?开发框架时,经常需要使用java对象的属性来封装程序的数据,每次都使用反射技术完成此类操作过于麻烦,所以sun公司开发了一套API,专门用于操作java对象的属性。

内省访问JavaBean属性的两种方式:
通过PropertyDescriptor类操作Bean的属性
通过Introspector类获得Bean对象的 BeanInfo,然后通过 BeanInfo 来获取属性的描述器( PropertyDescriptor ),通过这个属性描述器就可以获取某个属性对应的 getter/setter 方法,然后通过反射机制来调用这些方法。
回复 使用道具 举报
内省是 Java 语言对 Bean 类属性、事件的一种缺省处理方法

例如类A中有属性name, 那我们可以通过getName,setName来得到其值或者设置新的值。通过getName/setName来访问name属性,这就是默认的规则。Java中提供了一套API用来访问某个属性的getter/setter法,通过这些API可以使你不需要了解这个规则,这些API存放于包java.beans中。

一般的做法是通过类Introspector来获取某个对象的BeanInfo信息,然后通过BeanInfo来获取属性的描述器( PropertyDescriptor ),通过这个属性描述器就可以获取某个属性对应的getter/setter方法,然后我们就可以通过反射机制来调用这些方法。
回复 使用道具 举报
上面的哥们们说的都挺好的,
我就说说我的理解的一些内省和好处吧.
1.提高了复用性,如果我们定义了个person类,那么我们就可以通过内省对其进行操作,而不需要直接进行.
2.现在许多的架构都使用了内省的理论,主要就是可以提高复用性,架构就是可以复用的哪一部分,
不知道我理解的对不对,大家见笑了.....
回复 使用道具 举报
原本我也不是很明白内省,本来想等有人回答的比较满意了我也来借鉴下答案呢,不过这么长时间了都没有人答的让人满意;今天我查了下相关的资料找了了一些参考;虽然我仍然不能几句话给你讲清楚,不过脑子里还是有一些概念了;求人不如求自己啊;给你分享下吧;希望可以帮到你;
内省:
        IntroSpector//英语直译有,自省,反省,内省;等几种说法;Java取了内省;
下面是API中的介绍感觉比较抽象:


Introspector 类为通过工具学习有关受目标 Java Bean 支持的属性、事件和方法的知识提供了一个标准方法。

对于这三种信息,Introspector 将分别分析 bean 的类和超类,寻找显式或隐式信息,使用这些信息构建一个全面描述目标 bean 的 BeanInfo 对象。

对于每个 "Foo" 类,如果存在相应的 "FooBeanInfo" 类,显式信息可能是可用的,查询这些信息时,FooBeanInfo 类会提供一个非 null 值。通过获得目标 bean 类的完全受限定包名称并追加 "BeanInfo" 形成一个新类,首先查找 BeanInfo 类。如果此操作失败,则采用此名称的最终类名称组件,在 BeanInfo 包搜索路径中指定的每个包中搜索该类。

因此对于某个类,比如说 "sun.xyz.OurButton",首先需要查找称为 "sun.xyz.OurButtonBeanInfo" 的 BeanInfo 类,如果失败,则查找 BeanInfo 搜索路径中的每个包来搜索 OurButtonBeanInfo 类。对于默认搜索路径,这意味着将查找 "sun.beans.infos.OurButtonBeanInfo"。

如果某个类提供有关其自身的显式 BeanInfo,则将它添加到从分析所有派生类得到的 BeanInfo 信息中,并将显式信息视为当前类及其基类的确定的信息,无需进一步深入超类链进行分析。

如果没有在某个类上发现显式 BeanInfo,则使用低层次的反射来研究类的方法,并应用标准设计模式来标识属性存储器、事件源或公共方法。然后深入分析类的超类,从它那里(可能在超类链的顶部)添加信息。

因为 Introspector 缓存 BeanInfo 类来获得更好的性能,所以如果在使用多个类加载器的应用程序中使用它,则需小心谨慎。通常,在破坏一个已用于 introspect 类的 ClassLoader 时,应使用 Introspector.flushCaches 或 Introspector.flushFromCaches 方法从缓存中清除所有内省的类。

有关内省和设计模式的更多信息,请参阅 JavaBeans specification。


下面是在网上找的一些关于内省的一些总结和应用;结合下面的代码应该就可以明白内省是怎么一回事了;


        是Java语言对Bean类属性、事件的一种缺省处理方法
        内省(IntroSpector)是Java语言对JavaBean 类属性、事件的一种缺省处理方法。
  例如类A中有属性name, 那我们可以通过getName,setName 来得到其值或者设置新的值。
  通过getName/setName 来访问name属性,这就是默认的规则。
  Java中提供了一套API 用来访问某个属性的getter/setter方法,通过这些API 可以使你不需要了解这个规则,这些API存放于包java.beans 中。
  一般的做法是通过类Introspector的getBeanInfo方法获取某个对象的BeanInfo 信息,然后通过BeanInfo来获取属性的描述器(PropertyDescriptor),通过这个属性描述器就可以获取某个属性对应的getter/setter方法,然后我们就可以通过反射机制来调用这些方法。
  我们又通常把javabean的实例对象称之为值对象(Value Object),因为这些bean中通常只有一些信息字段和存储方法,没有功能性方法。
  一个JavaBean类可以不当JavaBean用,而当成普通类用。JavaBean实际就是一种规范,当一个类满足这个规范,这个类就能被其它特定的类调用。一   个类被当作javaBean使用时,JavaBean的属性是根据方法名推断出来的,它根本看不到java类内部的成员变量。去掉set前缀,然后取剩余部分,如果    剩余部分的第二个字母是小写的,则把剩余部分的首字母改成小的。
  除了反射用到的类需要引入外,内省需要引入的类如下所示,它们都属于java.beans包中的类,自己写程序的时候也不能忘了引入相应的包或者类。    下面代码片断是设置某个JavaBean类某个属性的关键代码:
1 package com.ljq.test;
  2
  3  import java.beans.BeanInfo;
  4  import java.beans.IntrospectionException;
  5  import java.beans.Introspector;
  6  import java.beans.PropertyDescriptor;
  7  import java.lang.reflect.InvocationTargetException;
  8  import java.lang.reflect.Method;
  9
  10  import org.apache.commons.beanutils.BeanUtils;
  11
  12
  13  public class IntrospectorTest {
  14
  15     public static void main(String[] args) throws IllegalArgumentException,
  16         IntrospectionException, IllegalAccessException, InvocationTargetException, NoSuchMethodException {
  17
  18         UserInfo userInfo=new UserInfo("zhangsan", "123456");
  19         String propertyName="userName";
  20         Object retVal=getProperty(userInfo, propertyName);
  21         System.out.println("retVal="+retVal); //retVal=zhangsan
  22
  23         Object value="abc";
  24         setProperty(userInfo, propertyName, value);
  25         retVal=getProperty(userInfo, propertyName);
  26         System.out.println("retVal="+retVal); //retVal=abc
  27
  //使用BeanUtils工具包操作JavaBean
  28         String userName=BeanUtils.getProperty(userInfo, propertyName);
  29         System.out.println("userName="+userName);
  30         BeanUtils.setProperty(userInfo, propertyName, "linjiqin");
  31         userName=BeanUtils.getProperty(userInfo, propertyName);
  32         System.out.println("userName="+userName);
  33     }
  34
  35     /**
  36      * 设置属性
  37      *
  38      * @param clazz 对象名
  39      * @param propertyName 属性名
  40      * @param value 属性值
  41      */
  42     private static void setProperty(Object clazz, String propertyName, Object value)
  43             throws IntrospectionException,IllegalAccessException, InvocationTargetException{
  44         //方法一相对于第二种方法这个简单一些;
  45          /*PropertyDescriptor pd=new PropertyDescriptor(propertyName, clazz.getClass());
  46         Method methodSet=pd.getWriteMethod();
  47         methodSet.invoke(clazz, value);*/
  48
  49         //方法二
  50          BeanInfo beanInfo=Introspector.getBeanInfo(clazz.getClass());
  51         PropertyDescriptor[] pds=beanInfo.getPropertyDescriptors();
  52         for(PropertyDescriptor pd:pds){
  53             if(propertyName.equals(pd.getName())){
  54                 Method methodSet=pd.getWriteMethod();
  55                 methodSet.invoke(clazz, value);
  56                 break;
  57             }
  58         }
  59     }
  60
  61     /**
  62      * 获取属性
  63      *
  64      * @param clazz 对象名
  65      * @param propertyName 属性名
  66      * @return
  67      * @throws IntrospectionException
  68      * @throws InvocationTargetException
  69      * @throws IllegalAccessException
  70      * @throws IllegalArgumentException
  71      */
  72     private static Object getProperty(Object clazz, String propertyName)
  73             throws IntrospectionException, IllegalArgumentException, IllegalAccessException, InvocationTargetException{
  74         //方法一
  75          /*PropertyDescriptor pd=new PropertyDescriptor(propertyName, clazz.getClass());
  76         Method methodGet=pd.getReadMethod();
  77         return methodGet.invoke(clazz);*/
  78
  79         //方法二
  80          Object retVal=null;
  81         BeanInfo beanInfo=Introspector.getBeanInfo(clazz.getClass());
  82         PropertyDescriptor[] pds=beanInfo.getPropertyDescriptors();
  83         for(PropertyDescriptor pd:pds){
  84             if(propertyName.equals(pd.getName())){
  85                 Method methodGet=pd.getReadMethod();
  86                 retVal=methodGet.invoke(clazz);
  87                 break;
  88             }
  89         }
  90         return retVal;
  91     }
  92
  93 }
  UserInfo类
  1 package com.ljq.test;
  2
  3 public class UserInfo {
  4     private String userName;
  5     private String pwd;
  6
  7     public UserInfo(String userName, String pwd) {
  8         super();
  9         this.userName = userName;
  10         this.pwd = pwd;
  11     }
  12
  13     public String getUserName() {
  14         return userName;
  15     }
  16
  17     public void setUserName(String userName) {
  18         this.userName = userName;
  19     }
  20
  21     public String getPwd() {
  22         return pwd;
  23     }
  24
  25     public void setPwd(String pwd) {
  26         this.pwd = pwd;
  27     }
  28
  29 }

回复 使用道具 举报
您需要登录后才可以回帖 登录 | 加入黑马