JAVA語(yǔ)言中的反射機(jī)制:
在Java 運(yùn)行時(shí) 環(huán)境中,對(duì)于任意一個(gè)類,能否知道這個(gè)類有哪些屬性和方法?
對(duì)于任意一個(gè)對(duì)象,能否調(diào)用他的方法?這些答案是肯定的,這種動(dòng)態(tài)獲取類的信息,以及動(dòng)態(tài)調(diào)用類的方法的功能來(lái)源于JAVA的反射。從而使java具有動(dòng)態(tài)語(yǔ)言的特性。
JAVA反射機(jī)制主要提供了以下功能:
1.在運(yùn)行時(shí)判斷任意一個(gè)對(duì)象所屬的類
2.在運(yùn)行時(shí)構(gòu)造任意一個(gè)類的對(duì)象
3.在運(yùn)行時(shí)判斷任意一個(gè)類所具有的成員變量和方法(通過(guò)反射甚至可以調(diào)用private方法)
4.在運(yùn)行時(shí)調(diào)用任意一個(gè)對(duì)象的方法(*****注意:前提都是在運(yùn)行時(shí),而不是在編譯時(shí))
Java 反射相關(guān)的API簡(jiǎn)介:
位于java。lang。reflect包中
--Class類:代表一個(gè)類
--Filed類:代表類的成員變量
--Method類:代表類的方法
--Constructor類:代表類的構(gòu)造方法
--Array類:提供了動(dòng)態(tài)創(chuàng)建數(shù)組,以及訪問(wèn)數(shù)組的元素的靜態(tài)方法。該類中的所有方法都是靜態(tài)方法
----Class類
在 java 的Object類中的申明了數(shù)個(gè)應(yīng)該在所有的java類中被改寫(xiě)的methods:
hashCode(), equals(),clone(),toString(),getClass()等,其中的getClass()返回yige
Class 類型的對(duì)象。
Class類十分的特殊,它和一般的類一樣繼承自O(shè)bject,其實(shí)體用以表達(dá)java程序運(yùn)行
時(shí)的 class和 interface,也用來(lái)表達(dá) enum,array,primitive,Java Types 以及關(guān)鍵字void
,當(dāng)加載一個(gè)類,或者當(dāng)加載器(class loader)的defineClass()被JVM調(diào)用,便產(chǎn)生一個(gè)Class
對(duì)象,
Class是Reflection起源,針對(duì)任何你想探勘的class(類),唯有現(xiàn)為他產(chǎn)生一個(gè)Class
的對(duì)象,接下來(lái)才能經(jīng)由后者喚起為數(shù)十多個(gè)的反射API。
Java允許我們從多種途徑為一個(gè)類class生成對(duì)應(yīng)的Class對(duì)象。
--運(yùn)用 getClass():Object類中的方法,每個(gè)類都擁有此方法
String str="abc";
Class cl=str.getClass();
--運(yùn)用 Class。getSuperclass():Class類中的方法,返回該Class的父類的Class
--運(yùn)用 Class。forName()靜態(tài)方法:
--運(yùn)用 ,Class:類名.class
--運(yùn)用primitive wrapper classes的TYPE語(yǔ)法: 基本類型包裝類的TYPE,如:Integer.TYPE
注意:TYPE的使用,只適合原生(基本)數(shù)據(jù)類型
----運(yùn)行時(shí)生成instance
想生成對(duì)象的實(shí)體,在反射動(dòng)態(tài)機(jī)制中有兩種方法,一個(gè)針對(duì)無(wú)變量的構(gòu)造方法,一個(gè)針對(duì)帶參數(shù)的
構(gòu)造方法,,如果想調(diào)用帶參數(shù)的構(gòu)造方法,就比較的麻煩,不能直接調(diào)用Class類中的newInstance()
,而是調(diào)用Constructor類中newInstance()方法,首先準(zhǔn)備一個(gè)Class[]作為Constructor的參數(shù)類型。
然后調(diào)用該Class對(duì)象的getConstructor()方法獲得一個(gè)專屬的Constructor的對(duì)象,最后再準(zhǔn)備一個(gè)
Object[]作為Constructor對(duì)象昂的newInstance()方法的實(shí)參。
在這里需要說(shuō)明的是 只有兩個(gè)類擁有newInstance()方法,分別是Class類和Constructor類
Class類中的newInstance()方法是不帶參數(shù)的,而Constructro類中的newInstance()方法是帶參數(shù)的
需要提供必要的參數(shù)。
例:
Class c=Class.forName("DynTest");
Class[] ptype=new Class[]{double.class,int.class};
Constructor ctor=c.getConstructor(ptypr);
Object[] obj=new Object[]{new Double(3.1415),new Integer(123)};
Object object=ctor.newInstance(obj);
System.out.println(object);
----運(yùn)行時(shí)調(diào)用Method
這個(gè)動(dòng)作首先準(zhǔn)備一個(gè)Class[]{}作為getMethod(String name,Class[])方法的參數(shù)類型,接下來(lái)準(zhǔn)備一個(gè)
Obeject[]放置自變量,然后調(diào)用Method對(duì)象的invoke(Object obj,Object[])方法。
注意,在這里調(diào)用
----運(yùn)行時(shí)調(diào)用Field內(nèi)容
變更Field不需要參數(shù)和自變量,首先調(diào)用Class的getField()并指定field名稱,獲得特定的Field對(duì)象后
便可以直接調(diào)用Field的 get(Object obj)和set(Object obj,Object value)方法
java 代碼
- package cn.com.reflection;
-
- import java.lang.reflect.Field;
- import java.lang.reflect.InvocationTargetException;
- import java.lang.reflect.Method;
-
- public class ReflectTester {
-
- /**
- * 在這個(gè)類里面存在有copy()方法,根據(jù)指定的方法的參數(shù)去 構(gòu)造一個(gè)新的對(duì)象的拷貝
- * 并將他返回
- * @throws NoSuchMethodException
- * @throws InvocationTargetException
- * @throws IllegalAccessException
- * @throws InstantiationException
- * @throws SecurityException
- * @throws IllegalArgumentException
- */
- public Object copy(Object obj) throws IllegalArgumentException, SecurityException, InstantiationException, IllegalAccessException, InvocationTargetException, NoSuchMethodException{
-
- //獲得對(duì)象的類型
- Class classType=obj.getClass();
- System.out.println("該對(duì)象的類型是:"+classType.toString());
-
- //通過(guò)默認(rèn)構(gòu)造方法去創(chuàng)建一個(gè)新的對(duì)象,getConstructor的視其參數(shù)決定調(diào)用哪個(gè)構(gòu)造方法
- Object objectCopy=classType.getConstructor(new Class[]{}).newInstance(new Object[]{});
-
- //獲得對(duì)象的所有屬性
- Field[] fields=classType.getDeclaredFields();
-
- for(int i=0;i
- //獲取數(shù)組中對(duì)應(yīng)的屬性
- Field field=fields[i];
-
- String fieldName=field.getName();
- String stringLetter=fieldName.substring(0, 1).toUpperCase();
-
- //獲得相應(yīng)屬性的getXXX和setXXX方法名稱
- String getName="get"+stringLetter+fieldName.substring(1);
- String setName="set"+stringLetter+fieldName.substring(1);
-
- //獲取相應(yīng)的方法
- Method getMethod=classType.getMethod(getName, new Class[]{});
- Method setMethod=classType.getMethod(setName, new Class[]{field.getType()});
-
- //調(diào)用源對(duì)象的getXXX()方法
- Object value=getMethod.invoke(obj, new Object[]{});
- System.out.println(fieldName+" :"+value);
-
- //調(diào)用拷貝對(duì)象的setXXX()方法
- setMethod.invoke(objectCopy,new Object[]{value});
-
-
- }
-
- return objectCopy;
-
- }
-
-
- public static void main(String[] args) throws IllegalArgumentException, SecurityException, InstantiationException, IllegalAccessException, InvocationTargetException, NoSuchMethodException {
- Customer customer=new Customer();
- customer.setName("hejianjie");
- customer.setId(new Long(1234));
- customer.setAge(19);
-
- Customer customer2=null;
- customer2=(Customer)new ReflectTester().copy(customer);
- System.out.println(customer.getName()+" "+customer2.getAge()+" "+customer2.getId());
-
- System.out.println(customer);
- System.out.println(customer2);
-
-
- }
-
- }
-
-
- class Customer{
-
- private Long id;
-
- private String name;
-
- private int age;
-
-
- public Customer(){
-
- }
-
- public int getAge() {
- return age;
- }
-
-
- public void setAge(int age) {
- this.age = age;
- }
-
-
- public Long getId() {
- return id;
- }
-
-
- public void setId(Long id) {
- this.id = id;
- }
-
-
- public String getName() {
- return name;
- }
-
-
- public void setName(String name) {
- this.name = name;
- }
-
- }
java 代碼
- package cn.com.reflection;
-
- import java.lang.reflect.Array;
-
- public class ArrayTester1 {
-
- /**
- * 此類根據(jù)反射來(lái)創(chuàng)建
- * 一個(gè)動(dòng)態(tài)的數(shù)組
- */
- public static void main(String[] args) throws ClassNotFoundException {
-
- Class classType=Class.forName("java.lang.String");
-
- Object array= Array.newInstance(classType,10); //指定數(shù)組的類型和大小
-
- //對(duì)索引為5的位置進(jìn)行賦值
- Array.set(array, 5, "hello");
-
- String s=(String)Array.get(array, 5);
-
- System.out.println(s);
-
-
- //循環(huán)遍歷這個(gè)動(dòng)態(tài)數(shù)組
- for(int i=0;i<((String[])array).length;i++){
-
- String str=(String)Array.get(array, i);
-
- System.out.println(str);
- }
-
- }
-
- }