`

java内醒

 
阅读更多
经过多方面的资料搜集整理,写下了这篇文章,本文主要讲解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,;
    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,;
    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,;
    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 项目,其实应该说几乎所有的项目都或多或少的采用这两种技术。在实际应用过程中二者要相互结合方能发挥真正的智能化以及高度可扩展性
分享到:
评论

相关推荐

    Java课件第一章java概述课件

    Java 的名称来源于两种说法:一是源自印度尼西亚的爪哇岛,寓意提供一种如同咖啡般提神醒脑的力量;二是源于美国俚语,象征着活力和热情。 Java 的特性包括: 1. **面向对象**:Java 强调面向对象编程,允许程序员...

    Java 程序设计语言前言摘要

    由于商标问题,Oak先后更名为Green,最终定名为Java,名字来源于印度尼西亚的一个咖啡岛屿,寓意其如咖啡般提神醒脑的特性。 Java的成功在于它的跨平台特性,通过Java虚拟机(JVM),Java程序可以在任何支持JVM的...

    《JAVA面试题》--轻醒Java面试题.zip

    十余年JAVA从业经验,精通JAVA技术体系,有志于做JAVA技能提升的朋友可与我联系,交个朋友 十余年JAVA从业经验,精通JAVA技术体系,有志于做JAVA技能提升的朋友可与我联系,交个朋友 十余年JAVA从业经验,精通JAVA...

    轻醒Java面试题.zip

    【一线互联网大厂Java核心面试题库】Java基础、异常、集合、并发编程、JVM、Spring全家桶、MyBatis、Redis、数据库、中间件MQ、Dubbo、Linux、Tomcat、ZooKeeper、Netty等等..

    java笔试题.txt

    根据提供的文件信息,我们可以从中提炼出与Java相关的多个知识点,具体包括异常处理、类与对象的概念、线程管理、错误与异常的区别、final关键字的使用、堆栈的区别、自定义数据结构、图形系统的设计、匿名内部类的...

    28个java常用的工具类源码

    7. **`java.util.Random`**:用于生成随机数,可以生成指定范围内的整数或浮点数。 8. **`java.io.*`和`java.nio.*`**:I/O流库,`java.io`是传统的字节流和字符流API,`java.nio`提供了一种基于缓冲区的非阻塞I/O...

    java实现网络五子棋

    在Java编程语言中实现一个网络五子棋游戏是一项有趣且具有挑战性的任务,它涉及到多个IT领域的知识。首先,我们要理解五子棋的基本规则:两名玩家轮流在棋盘上下棋,先连成五子直线(横、竖或斜)的玩家获胜。网络...

    Java对数据库CRUD DEMO

    Java对数据库的CRUD(Create, Read, Update, Delete)操作是编程中常见的任务,尤其在Web应用开发中。这个DEMO提供了简化数据库交互的方法,避免了手动编写大量重复的SQL语句。以下是对这个"Java对数据库CRUD DEMO...

    java开发常用jar包

    commons-beanutils.jar Apache Commons包中的一个,包含了一些Bean工具类类。必须使用的jar包。 commons-collections.jar ...velocity基于java的模板引擎,允许用简单的模板语言来引用java定义的对

    java生成验证码demo

    在本示例中,我们关注的是使用Java编程语言来生成验证码的实践。这个"java生成验证码demo"提供了生成自定义验证码图片的方法,包括设置图片尺寸、验证码长度,以及添加各种视觉混淆元素,以增加机器识别的难度。 ...

    java elasticsearch 存储查询

    Java与Elasticsearch的整合是现代大数据处理和搜索场景中的常见实践。Elasticsearch是一个分布式、RESTful风格的搜索引擎,以其高效、灵活和强大的全文检索能力而闻名。在这个压缩包中,我们找到了一系列用于Java与...

    Excel生成Java 带JPA注解的实体类

    在企业级开发中,Java JPA(Java Persistence API)被用来处理对象关系映射,简化数据库操作。本话题将探讨如何利用Apache POI库从Excel模板生成带有JPA注解的Java实体类,实现高效的数据模型自动化创建。 Apache ...

    javaJUI贪吃蛇

    【标题】:“Java JUI贪吃蛇” 在Java编程领域,Java Swing或JavaFX(统称为JUI,Java用户界面)被广泛用于构建图形化应用程序。这个“javaJUI贪吃蛇”项目就是一个很好的示例,它展示了如何利用Java的GUI库来创建...

    [Java程序设计实用教程(第5版)][叶核亚][电子教案(PPT版本)].zip

    《Java程序设计实用教程(第5版)》是由叶核亚编著的一本经典Java学习教程,本教程针对初学者和有一定基础的程序员,旨在帮助读者深入理解和掌握Java编程语言的核心概念和技术。作为电子教案的PPT版本,它将理论与...

    java网络爬虫模拟登入抓取数据

    Java网络爬虫是一种技术,用于自动化地从互联网上抓取信息。在许多情况下,特别是涉及到登录后才能访问的内容,如人人网这样的社交平台,我们不能简单地通过HTTP请求获取数据,因为这些页面需要用户身份验证。这就...

    Java生成二维码图片

    Java作为一种广泛使用的编程语言,提供了丰富的库和工具来帮助开发者实现这一功能。本篇文章将详细探讨如何使用Java生成二维码图片,以及如何在Web前端展示这些二维码。 首先,Java中的二维码生成通常依赖于第三方...

    TUserMapper.java

    TUserMapper.java

    Excel模板Java实体生成器

    【Excel模板Java实体生成器】是一种工具,它利用Apache POI库来解析Excel文件,无论是2003版的老格式还是2007版的新格式,都能够被它识别并读取。这个工具的主要目的是帮助开发者根据Excel表格中的数据结构自动生成...

Global site tag (gtag.js) - Google Analytics