淺談Java內(nèi)省機制
概念
JavaBean
在實際編程中,我們常常需要一些用來包裝值對象的類,例如Student、 Employee、Order,這些 類中往往沒有業(yè)務方法,只是為了把需要處理的實體對象進行封裝,有這樣的特征:
- 屬性都是私有的;
- 有無參的public構造方法;
- 對私有屬性根據(jù)需要提供公有的getXxx方法以及setXxx方法;
比如:屬性名稱為name,則有getName方法返回屬性name值, setName方法設置name值;注意方法的名稱通常是get或 set加上屬性名稱,并把屬性名稱的首字母大寫;這些方法稱為getters/setters;getters必須有返回值沒有方法參數(shù); setter值沒有返回值,有方法參數(shù);
例如下面的例子:

符合這些特征的類,被稱為JavaBean;
內(nèi)省
內(nèi)省(Inspector)機制就是基于反射的基礎, Java語言對Bean類屬性、事件的一種缺省處理方法。
只要類中有getXXX方法,或者setXXX方法,或者同時有getXXX及setXXX方法,其中getXXX方 法沒有方法參數(shù),有返回值; setXXX方法沒有返回值,有一個方法參數(shù);那么內(nèi)省機制就認為 XXX為一個屬性;
例如下面代碼
Employee類中根本沒有聲明age屬性,僅僅是聲明了這樣的getter和setter.內(nèi)省機制就認為age是屬性
package com.shixun.introspector;
public class Employee {
private String name;
private Double score;
// age將被內(nèi)省認為是屬性
public int getAge(){
return 30;
}
// name將被內(nèi)省認為是屬性
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
// score將被內(nèi)省認為是屬性
public Double getScore() {
return score;
}
public void setScore(Double score) {
this.score = score;
}
public static void main(String[] args) {
}
}
相關API
與Java內(nèi)省有關的主要類及接口有:
java.beans.Introspector類: 為獲得JavaBean屬性、事件、方法提供了標準方法;通常使用其中的getBeanInfo方法返回BeanInfo對象;Java.beans.BeanInfo接口:不能直接實例化,通常通過Introspector類返回該類型對象,提供了返回屬性描述符對象(PropertyDescriptor)、方法描述符對象(MethodDescriptor) 、 bean描述符(BeanDescriptor)對象的方法;Java.beans.PropertyDescriptor類:用來描述一個屬性,該屬性有getter及setter方法;
可以使用PropertyDescriptor類的方法獲取屬性相關的信息,例如getName方法返回屬性的名字:
PropertyDescriptor類中定義了方法可以獲取該屬性的getter和setter方法
| 方法 | 方法描述 |
|---|---|
| Method getReadMethod() | 回屬性對應的getter方法對象; |
| Method getWriteMethod() | 回屬性對應的setter方法對象; |
下面我們來用代碼深入探究一下:
代碼案例:獲取屬性相關信息
Employee如上面代碼所示,繼續(xù)編寫主函數(shù)進行測試
首先用BeanInfo接口獲取BeanInfo的對象,再通過BeanInfo對象獲取PropertyDescriptor屬性描述
//獲取BeanInfo的對象
BeanInfo employeeBeanInfo = Introspector.getBeanInfo(Employee.class);
//通過BeanInfo對象獲取PropertyDescriptor屬性描述
PropertyDescriptor[] propertyDescriptors = employeeBeanInfo.getPropertyDescriptors();
System.out.println("通過Inspector內(nèi)省機制獲取JavaBean屬性======= 打印所有信息 ====================");
Arrays.stream(propertyDescriptors).forEach(f->{
System.out.println("====================================");
System.out.println("屬性名:"+f.getName());
System.out.println("類型:"+f.getPropertyType());
System.out.println("get方法:"+f.getReadMethod());
System.out.println("set方法:"+f.getWriteMethod());
});
// 或者用增強for
System.out.println("通過Inspector內(nèi)省機制獲取JavaBean屬性======= 打印所有信息 ====================");
for (PropertyDescriptor propertyDescriptor : propertyDescriptors) {
System.out.println("====================================");
System.out.println("名字:" + propertyDescriptor.getName());
System.out.println("類型:" + propertyDescriptor.getPropertyType());
System.out.println("get方法:" + propertyDescriptor.getReadMethod());
System.out.println("set方法:" + propertyDescriptor.getWriteMethod());
}
運行結果如下:

我們也可以通過反射調(diào)用這里獲取的get或set方法
//創(chuàng)建Employee的對象
Class<?> clazz = Class.forName("com.shixun.introspector.Employee");
Object employee = clazz.newInstance();
//遍歷屬性描述對象
for (PropertyDescriptor propertyDescriptor : propertyDescriptors) {
//打印屬性名稱
System.out.println(propertyDescriptor.getName());
//判斷屬性名稱是不是name
if (propertyDescriptor.getName().equals("name")) {
//setter方法
Method writeMethod = propertyDescriptor.getWriteMethod();
//調(diào)用setName方法
writeMethod.invoke(employee, "jack");
//getter方法
Method readMethod = propertyDescriptor.getReadMethod();
//調(diào)用getName方法
Object nameValue = readMethod.invoke(employee);
System.out.println("name屬性的值為:" + nameValue);
}
//判斷屬性名稱是否為score
if (propertyDescriptor.getName().equals("score")) {
//setter方法
Method scoreWriteMethod = propertyDescriptor.getWriteMethod();
//調(diào)用setScore方法
scoreWriteMethod.invoke(employee, new Double(3000));
//getter方法
Method scoreReadMethod = propertyDescriptor.getReadMethod();
Object scoreValue = scoreReadMethod.invoke(employee);
System.out.println("score屬性的值為:" + scoreValue);
}
}
System.out.println("當前對象的信息:"+employee.toString());
運行結果如下所示:

全部代碼附在最下方?。。。。?!
內(nèi)省屬性的注意事項
- 很多框架都使用了內(nèi)省機制檢索對象的屬性,定義屬性名字時,名字最好起碼以兩個小寫字母開頭,例如stuName,而不要使用sName,某些情況下,可能會導致檢索屬性失?。?/li>
- 內(nèi)省機制檢索屬性時,是根據(jù)getter和setter方法確認屬性名字,而不是根據(jù)類里聲明的成員變量名稱決定;
完整代碼
package com.shixun.introspector;
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.util.Arrays;
public class Employee {
private String name;
private Double score;
// age將被內(nèi)省認為是屬性
public int getAge() {
return 30;
}
// name將被內(nèi)省認為是屬性
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
// score將被內(nèi)省認為是屬性
public Double getScore() {
return score;
}
public void setScore(Double score) {
this.score = score;
}
@Override
public String toString() {
return "Employee{" +
"name='" + name + '\'' +
", score=" + score +
'}';
}
public static void main(String[] args) throws ClassNotFoundException, IntrospectionException, IllegalAccessException, InstantiationException, InvocationTargetException {
//獲取BeanInfo的對象
BeanInfo employeeBeanInfo = Introspector.getBeanInfo(Employee.class);
//通過BeanInfo對象獲取PropertyDescriptor屬性描述
PropertyDescriptor[] propertyDescriptors = employeeBeanInfo.getPropertyDescriptors();
// System.out.println("通過Inspector內(nèi)省機制獲取JavaBean屬性======= 打印所有信息 ====================");
// Arrays.stream(propertyDescriptors).forEach(f->{
// System.out.println("====================================");
// System.out.println("屬性名:"+f.getName());
// System.out.println("類型:"+f.getPropertyType());
// System.out.println("get方法:"+f.getReadMethod());
// System.out.println("set方法:"+f.getWriteMethod());
// });
//
//
//
// System.out.println("通過Inspector內(nèi)省機制獲取JavaBean屬性======= 打印所有信息 ====================");
//
// for (PropertyDescriptor propertyDescriptor : propertyDescriptors) {
// System.out.println("名字:" + propertyDescriptor.getName());
// System.out.println("類型:" + propertyDescriptor.getPropertyType());
// System.out.println("get方法:" + propertyDescriptor.getReadMethod());
// System.out.println("set方法:" + propertyDescriptor.getWriteMethod());
// }
//創(chuàng)建Employee的對象
Class<?> clazz = Class.forName("com.shixun.introspector.Employee");
Object employee = clazz.newInstance();
//遍歷屬性描述對象
for (PropertyDescriptor propertyDescriptor : propertyDescriptors) {
//打印屬性名稱
System.out.println(propertyDescriptor.getName());
//判斷屬性名稱是不是name
if (propertyDescriptor.getName().equals("name")) {
//setter方法
Method writeMethod = propertyDescriptor.getWriteMethod();
//調(diào)用setName方法
writeMethod.invoke(employee, "jack");
//getter方法
Method readMethod = propertyDescriptor.getReadMethod();
//調(diào)用getName方法
Object nameValue = readMethod.invoke(employee);
System.out.println("name屬性的值為:" + nameValue);
}
//判斷屬性名稱是否為score
if (propertyDescriptor.getName().equals("score")) {
//setter方法
Method scoreWriteMethod = propertyDescriptor.getWriteMethod();
//調(diào)用setScore方法
scoreWriteMethod.invoke(employee, new Double(3000));
//getter方法
Method scoreReadMethod = propertyDescriptor.getReadMethod();
Object scoreValue = scoreReadMethod.invoke(employee);
System.out.println("score屬性的值為:" + scoreValue);
}
}
System.out.println("當前對象的信息:"+employee.toString());
}
}
到此這篇關于淺談Java內(nèi)省機制的文章就介紹到這了,更多相關Java內(nèi)省機制內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
Spring Boot中單例類實現(xiàn)對象的注入方式
這篇文章主要介紹了Spring Boot中單例類實現(xiàn)對象的注入方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-08-08

