java内省机制 + 内省是什么 + 内省实现方式 + 和反射的区别

2024-06-04 16:08

本文主要是介绍java内省机制 + 内省是什么 + 内省实现方式 + 和反射的区别,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

 

前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到教程。

一、内省是什么、实现方式:

内省(Introspector)是Java语言对Bean类属性、事件的一种缺省处理方法。
例如类A中有属性name,那我们可以通过getName,setName来得到其值或者设置新的值。
通过getName/setName来访问name属性,这就是默认的规则。
Java中提供了一套API用来访问某个属性的getter/setter方法,通过这些API可以使你不需要了解这个规则,
这些API存放于包java.beans中。Java语言对Bean类属性、事件的一种缺省处理方法。例如类A中有属性name,那我们可以通过getName,setName来得到其值或者设置新的值。通过getName/setName来访问name属性,这就是默认的规则。Java中提供了一套API用来访问某个属性的getter/setter方法,通过这些API可以使你不需要了解这个规则(但你最好还是要搞清楚),这些API存放于包java.beans中。一般的做法是通过类Introspector来获取某个对象的BeanInfo信息,
然后通过BeanInfo来获取属性的描述器(PropertyDescriptor),
通过这个属性描述器就可以获取某个属性对应的getter/setter方法,
然后我们就可以通过反射机制来调用这些方法。一般的做法是通过类Introspector来获取某个对象的BeanInfo信息,然后通过BeanInfo来获取属性的描述器(PropertyDescriptor),通过这个属性描述器就可以获取某个属性对应的getter/setter方法,然后我们就可以通过反射机制来调用这些方法。

内省的解释

内省在wiki上的解释:

在计算机科学中,内省是指计算机程序在运行时(Run time)检查对象(Object)类型的一种能力,通常也可以称作运行时类型检查。 
不应该将内省和反射混淆。相对于内省,反射更进一步,是指计算机程序在运行时(Run time)可以访问、检测和修改它本身状态或行为的一种能力。

二、内省和反射区别:区别:

反射式在运行状态把Java类中的各种成分映射成相应的Java类,可以动态的获取所有的属性以及动态调用任意一个方法,强调的是运行状态。 
内省机制是通过反射来实现的,BeanInfo用来暴露一个bean的属性、方法和事件,以后我们就可以操纵该JavaBean的属性


这里写图片描述
在Java内省中,用到的基本上就是上述几个类。 
通过BeanInfo这个类就可以获取到类中的方法和属性,具体的可以参考JDK文档

Web 开发框架 Struts 中的 FormBean 就是通过内省机制来将表单中的数据映射到类的属性上,
因此要求 FormBean 的每个属性要有 getter/setter 方法。
但也并不总是这样,什么意思呢?就是说对一个 Bean 类来讲,
我可以没有属性,但是只要有 getter/setter 方法中的其中一个,
那么 Java 的内省机制就会认为存在一个属性,比如类中有方法 setMobile ,
那么就认为存在一个 mobile 的属性。
将 Java 的反射以及内省应用到程序设计中去可以大大的提供程序的智能化和可扩展性。
有很多项目都是采取这两种技术来实现其核心功能,例如我们前面提到的 Struts ,
还有用于处理 XML 文件的 Digester 项目,其实应该说几乎所有的项目都或多或少的采用这两种技术。
在实际应用过程中二者要相互结合方能发挥真正的智能化以及高度可扩展性。 

三、内省代码演示:

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则是将该属性的值转换成字符串后才返回

 

这篇关于java内省机制 + 内省是什么 + 内省实现方式 + 和反射的区别的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1030491

相关文章

Spring Boot整合Redis注解实现增删改查功能(Redis注解使用)

《SpringBoot整合Redis注解实现增删改查功能(Redis注解使用)》文章介绍了如何使用SpringBoot整合Redis注解实现增删改查功能,包括配置、实体类、Repository、Se... 目录配置Redis连接定义实体类创建Repository接口增删改查操作示例插入数据查询数据删除数据更

Java Lettuce 客户端入门到生产的实现步骤

《JavaLettuce客户端入门到生产的实现步骤》本文主要介绍了JavaLettuce客户端入门到生产的实现步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要... 目录1 安装依赖MavenGradle2 最小化连接示例3 核心特性速览4 生产环境配置建议5 常见问题

Java使用Swing生成一个最大公约数计算器

《Java使用Swing生成一个最大公约数计算器》这篇文章主要为大家详细介绍了Java使用Swing生成一个最大公约数计算器的相关知识,文中的示例代码讲解详细,感兴趣的小伙伴可以了解一下... 目录第一步:利用欧几里得算法计算最大公约数欧几里得算法的证明情形 1:b=0情形 2:b>0完成相关代码第二步:加

linux ssh如何实现增加访问端口

《linuxssh如何实现增加访问端口》Linux中SSH默认使用22端口,为了增强安全性或满足特定需求,可以通过修改SSH配置来增加或更改SSH访问端口,具体步骤包括修改SSH配置文件、增加或修改... 目录1. 修改 SSH 配置文件2. 增加或修改端口3. 保存并退出编辑器4. 更新防火墙规则使用uf

Java 的ArrayList集合底层实现与最佳实践

《Java的ArrayList集合底层实现与最佳实践》本文主要介绍了Java的ArrayList集合类的核心概念、底层实现、关键成员变量、初始化机制、容量演变、扩容机制、性能分析、核心方法源码解析、... 目录1. 核心概念与底层实现1.1 ArrayList 的本质1.1.1 底层数据结构JDK 1.7

Java Map排序如何按照值按照键排序

《JavaMap排序如何按照值按照键排序》该文章主要介绍Java中三种Map(HashMap、LinkedHashMap、TreeMap)的默认排序行为及实现按键排序和按值排序的方法,每种方法结合实... 目录一、先理清 3 种 Map 的默认排序行为二、按「键」排序的实现方式1. 方式 1:用 TreeM

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

C++中unordered_set哈希集合的实现

《C++中unordered_set哈希集合的实现》std::unordered_set是C++标准库中的无序关联容器,基于哈希表实现,具有元素唯一性和无序性特点,本文就来详细的介绍一下unorder... 目录一、概述二、头文件与命名空间三、常用方法与示例1. 构造与析构2. 迭代器与遍历3. 容量相关4

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

C++中悬垂引用(Dangling Reference) 的实现

《C++中悬垂引用(DanglingReference)的实现》C++中的悬垂引用指引用绑定的对象被销毁后引用仍存在的情况,会导致访问无效内存,下面就来详细的介绍一下产生的原因以及如何避免,感兴趣... 目录悬垂引用的产生原因1. 引用绑定到局部变量,变量超出作用域后销毁2. 引用绑定到动态分配的对象,对象