前言

使用内省相对于直接使用反射更加安全可靠,Java的反射机制比较特殊,它不同于一般的编程方式,稍不小心就容易破坏类的封装性。练的不好,就容易走火入魔。没关系,很多时候我们还可以使用Java的内省机制哦。

本文会讲Java的内省机制是什么和怎么使用。

请先给二当家的一个三连,然后接着读下去吧,多谢。

对了,还不懂反射的小伙伴可以先去读读二当家的这篇文章 - <<java的上乘武功,反射,好好玩哦,绝对值得收藏>>

本文由 二当家的白帽子 https://le-yi.blog.csdn.net/ 博客原创,转载请注明来源,谢谢~


文章目录

  • 前言
  • Java的内省机制是什么?
  • 使用内省替代直接使用反射可以防止破坏类的封装
  • 使用内省也一样可以写出通用的工具
  • 尾声

Java的内省机制是什么?

内省(Introspection )在心理学中,它是心理学基本研究方法之一。内省法又称自我观察法。它是发生在内部的,我们自己能够意识到的主观现象。也可以说是对于自己的主观经验及其变化的观察。正因为它的主观性,内省法自古以来就成为心理学界长期的争论。争论于它是否客观,是否可靠。另外内省也可看作自我反省,也是儒家强调的自我思考。从这个角度说它可以应用于计算机领域,例如Java内省机制和cocoa内省机制。

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

以上就是百科的解释。Java的内省最终是用Java的反射实现的。那为什么不直接用反射,要使用内省呢?


使用内省替代直接使用反射可以防止破坏类的封装

我们定义一个人的类型,其中包括年龄和是否成年两个属性。在修改年龄属性的时候会同时修改是否成年的属性。我们假设18岁和18岁以上就是成年,否则就是未成年。

import java.beans.IntrospectionException;
import java.beans.PropertyDescriptor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.text.MessageFormat;class Person {/*** 18岁成年*/private static final int ADULT_AGE = 18;/*** 年龄*/private int     age;/*** 是否成年*/private boolean adult;public int getAge() {return age;}public void setAge(int age) {this.age = age;this.adult = age >= ADULT_AGE;}public boolean isAdult() {return adult;}public String toString() {return MessageFormat.format("age:{0},adult:{1}", age, adult);}
}/*** @author 二当家的白帽子 https://le-yi.blog.csdn.net/*/
public class Test {/*** 利用反射修改对象属性* @param o* @param fieldName* @param value* @throws NoSuchFieldException* @throws IllegalAccessException*/public static void changeObjectFieldByReflection(Object o, String fieldName, Object value) throws NoSuchFieldException, IllegalAccessException {Field field = o.getClass().getDeclaredField(fieldName);field.setAccessible(true);field.set(o, value);}/*** 利用内省修改对象属性* @param o* @param fieldName* @param value* @throws NoSuchFieldException* @throws IllegalAccessException*/public static void changeObjectFieldByIntrospector(Object o, String fieldName, Object value) throws IntrospectionException, InvocationTargetException, IllegalAccessException {PropertyDescriptor pd = new PropertyDescriptor(fieldName, o.getClass());pd.getWriteMethod().invoke(o, value);}public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException, IntrospectionException, InvocationTargetException {Person p = new Person();changeObjectFieldByReflection(p, "age", 20);System.out.println("反射修改属性破坏类的封装,使其内部状态错误:");System.out.println(p);changeObjectFieldByIntrospector(p, "age", 18);System.out.println("内省修改属性未破坏类的封装:");System.out.println(p);}
}

可以看到,反射由于是直接修改属性,所以破坏了类中封装的逻辑(20岁却不是成年)。

而内省由于修改属性还是调用了set方法,也就是说和正常修改对象属性调用了相同的方法,所以类的封装性不会遭到破坏。

当然由于内省其实本质也是反射,可以说是封装了反射,所以如果反射用的正确,也是安全的,我们可以根据属性名去获取相应的set和get方法,然后再去调用,但是这种情况下内省使用起来就更方便,毕竟没有必要重复发明一个车轮子,圆形轮子已经是很多年很多年智慧的结晶了。

那么问题来了,既然内省就是调用set和get方法,那我为什么不直接调用set和get方法,而要使用内省呢?


使用内省也一样可以写出通用的工具

既然内省可以动态获取信息,那就和反射一样,可以实现出通用工具或者框架哦。我们这里实现一个可以拷贝任意类型两个对象的属性值的工具方法。

import java.beans.BeanInfo;
import java.beans.IntrospectionException;
import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.text.MessageFormat;class Person {/*** 18岁成年*/private static final int ADULT_AGE = 18;/*** 名字*/private final String  name;/*** 身高*/private       int     height;/*** 年龄*/private       int     age;/*** 是否成年*/private       boolean adult;public Person(String name) {this.name = name;}public String getName() {return name;}public int getHeight() {return height;}public void setHeight(int height) {this.height = height;}public int getAge() {return age;}public void setAge(int age) {this.age = age;this.adult = age >= ADULT_AGE;}public boolean isAdult() {return adult;}public String toString() {return MessageFormat.format("name:{0},height:{1},age:{2},adult:{3}", name, height, age, adult);}
}/*** @author 二当家的白帽子 https://le-yi.blog.csdn.net/*/
public class Test {/*** 将orig的可读属性值拷贝到dest的可写属性中* @param dest* @param orig* @param <T>* @throws IntrospectionException* @throws InvocationTargetException* @throws IllegalAccessException*/public static <T> void copyProperties(T dest, T orig) throws IntrospectionException, InvocationTargetException, IllegalAccessException {BeanInfo beanInfo = Introspector.getBeanInfo(orig.getClass());PropertyDescriptor[] pds = beanInfo.getPropertyDescriptors();for (PropertyDescriptor pd : pds) {Method rm = pd.getReadMethod();Method wm = pd.getWriteMethod();if (rm != null&& wm != null) {Object value = rm.invoke(orig);wm.invoke(dest, value);}}}public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException, IntrospectionException, InvocationTargetException {Person p2 = new Person("二当家的");p2.setAge(18);p2.setHeight(180);System.out.println(p2);Person p1 = new Person("大当家的");System.out.println(p1);System.out.println("将二当家的可读属性值拷贝给大当家的可写属性:");copyProperties(p1, p2);System.out.println(p1);}
}

可以看到,名字没有被拷贝,其他的属性值都顺利拷贝了。这也是我们期望的结果。

内省很好的保证了类的封装性,同时又具有动态获取对象属性,和动态修改对象属性的能力。

另外是我二当家的让大当家的长高,长大,成年的。这便宜我偷偷占了,开森。


尾声

和反射一样,一般的程序可能也用不到写内省的代码。但是像apache的beanutils这样方便的工具,如果没有反射也没有内省,我真的想不出如何实现呢。哪怕永远不需要用内省,了解机制对我们都有着莫大的好处。

java的反射用不好容易走火入魔?还可以用内省啊!相关推荐

  1. 【Java】反射、枚举、Lambda表达式

    一.反射 1 定义 2 用途(了解) 3 反射基本信息 4 反射相关的类(重要) 4.1 Class类(反射机制的起源 ) 4.1.1 Class类中的相关方法(方法的使用方法在后边的示例当中) 4. ...

  2. java初反射_初始 java 反射机制 (一)

    反射机制详解 JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取的信息以及动态调用对象的方法的功能称为j ...

  3. java反射机制(三)---java的反射和代理实现IOC模式 模拟spring

    IOC(Inverse of Control)可翻译为"控制反转",但大多数人都习惯将它称为"依赖注入".在Spring中,通过IOC可以将实现类.参数信息等配 ...

  4. sqlite字段是否存在_学习廖雪峰的JAVA教程---反射(访问字段)

    对任意的一个Object实例,只要我们获取了它的Class,就可以获取它的一切信息. 我们先看看如何通过Class实例获取字段信息.Class类提供了以下几个方法来获取字段: Field getFie ...

  5. java小游戏毕业论文,你不懂还不学?

    1. Spring 特点 Spring 主要有如下特点: 轻量级:Spring 是非侵入式,其中的对象不依赖 Spring 的特定类: 控制反转(IoC):通过 IoC,促进了低耦合,一个对象依赖的其 ...

  6. Java 使用反射处理注解

    Java 使用反射处理注解 自定义注解的格式: [public|final] @interface 注解名//@interface 表明:这是一个自定义注解 {注解元素//注解元素 是无参数的方法 } ...

  7. java原理—反射机制

    http://www.cnblogs.com/forlina/archive/2011/06/21/2085849.html 一.什么是反射: 反射的概念是由Smith在1982年首次提出的,主要是指 ...

  8. 学习:java原理—反射机制

    一.什么是反射: 反射的概念是由Smith在1982年首次提出的,主要是指程序可以访问.检测和修改它本身状态或行为的一种能力.这一概念的提 出很快引发了计算机科学领域关于应用反射性的研究.它首先被程序 ...

  9. Java —— Reflect反射机制

    JAVA反射机制是在运行时,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取的信息以及动态调用对象的方法的功能称为java的反射机制. ...

最新文章

  1. HBase 在京东人资数据预处理平台中的实践!
  2. python stm32-STM32 上面跑Python
  3. labview如何进行串口通讯(转)
  4. 零基础学python:魔法函数都有了解吗
  5. java 定时器框架_java定时器
  6. django调用java_07.手把手教将深度学习利用Django将模型发布成服务供java调用
  7. Python学习 :文件操作
  8. 自学web前端课程大纲分享,适合所有人学习
  9. 【场景化解决方案】OA付款审批同步到金蝶KIS
  10. tl-wr842n服务器未响应,TL-WR842N路由器怎么重启? 重启路由器的技巧
  11. linux capability详解与容器中的capability
  12. Springboot整合支付宝支付(沙箱)
  13. 计算机二级wps知识点,计算机二级MS office和WPS office如何备考?
  14. Android文字跑马灯简单实现的三种方法
  15. 阿里飞猪一员工贩卖机票报销发票获利超千万被判六年
  16. 各种HIC处理数据之间的相互转化
  17. 基于深度学习的显著性目标检测方法综述
  18. 《程序员面试金典(第6版)》 面试题 08.11. 硬币(动态规划,组合问题,C++)
  19. Tobii pro lab学习笔记1
  20. 给定一个单链表,把所有的奇数节点和偶数节点分别排在一起。请注意,这里的奇数节点和偶数节点指的是节点编号的奇偶性,而不是节点的值的奇偶性。

热门文章

  1. 关于中兴客户认证端破解的猜想
  2. 更新 freenode irc 邮箱
  3. xshell关闭后 数据库进程_关闭命令行或者xshell窗口后后台依旧执行命令
  4. 端到端安全能力,态势感知抵御潜在风险
  5. 【正点原子FPGA连载】第四十九章OV5640摄像头HDMI灰度显示实验 -摘自【正点原子】新起点之FPGA开发指南_V2.1
  6. Could not resolve placeholder ‘‘“ in value ““
  7. getParameter方法的用法
  8. windows 10/11 多用户同时远程登陆
  9. hive如何创建视图
  10. c语言中冒号运算符,冒号等于 - G之间的差异:=和=运算符