JAVA語言中的反射機制:
在Java 運行時 環境中,對于任意一個類,能否知道這個類有哪些屬性和方法?
對于任意一個對象,能否調用他的方法?這些答案是肯定的,這種動態獲取類的信息,以及動態調用類的方法的功能來源于JAVA的反射。從而使java具有動態語言的特性。
JAVA反射機制主要提供了以下功能:
1.在運行時判斷任意一個對象所屬的類
2.在運行時構造任意一個類的對象
3.在運行時判斷任意一個類所具有的成員變量和方法(通過反射甚至可以調用private方法)
4.在運行時調用任意一個對象的方法(*****注意:前提都是在運行時,而不是在編譯時)
Java 反射相關的API簡介:
位于java。lang。reflect包中
--Class類:代表一個類
--Filed類:代表類的成員變量
--Method類:代表類的方法
--Constructor類:代表類的構造方法
--Array類:提供了動態創建數組,以及訪問數組的元素的靜態方法。該類中的所有方法都是靜態方法
----Class類
在 java 的Object類中的申明了數個應該在所有的java類中被改寫的methods:
hashCode(), equals(),clone(),toString(),getClass()等,其中的getClass()返回yige
Class 類型的對象。
Class類十分的特殊,它和一般的類一樣繼承自Object,其實體用以表達java程序運行
時的 class和 interface,也用來表達 enum,array,primitive,Java Types 以及關鍵字void
,當加載一個類,或者當加載器(class loader)的defineClass()被JVM調用,便產生一個Class
對象,
Class是Reflection起源,針對任何你想探勘的class(類),唯有現為他產生一個Class
的對象,接下來才能經由后者喚起為數十多個的反射API。
Java允許我們從多種途徑為一個類class生成對應的Class對象。
--運用 getClass():Object類中的方法,每個類都擁有此方法
String str="abc";
Class cl=str.getClass();
--運用 Class。getSuperclass():Class類中的方法,返回該Class的父類的Class
--運用 Class。forName()靜態方法:
--運用 ,Class:類名.class
--運用primitive wrapper classes的TYPE語法: 基本類型包裝類的TYPE,如:Integer.TYPE
注意:TYPE的使用,只適合原生(基本)數據類型
----運行時生成instance
想生成對象的實體,在反射動態機制中有兩種方法,一個針對無變量的構造方法,一個針對帶參數的
構造方法,,如果想調用帶參數的構造方法,就比較的麻煩,不能直接調用Class類中的newInstance()
,而是調用Constructor類中newInstance()方法,首先準備一個Class[]作為Constructor的參數類型。
然后調用該Class對象的getConstructor()方法獲得一個專屬的Constructor的對象,最后再準備一個
Object[]作為Constructor對象昂的newInstance()方法的實參。
在這里需要說明的是 只有兩個類擁有newInstance()方法,分別是Class類和Constructor類
Class類中的newInstance()方法是不帶參數的,而Constructro類中的newInstance()方法是帶參數的
需要提供必要的參數。
例:
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);
----運行時調用Method
這個動作首先準備一個Class[]{}作為getMethod(String name,Class[])方法的參數類型,接下來準備一個
Obeject[]放置自變量,然后調用Method對象的invoke(Object obj,Object[])方法。
注意,在這里調用
----運行時調用Field內容
變更Field不需要參數和自變量,首先調用Class的getField()并指定field名稱,獲得特定的Field對象后
便可以直接調用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 {
-
-
-
-
-
-
-
-
-
-
-
- public Object copy(Object obj) throws IllegalArgumentException, SecurityException, InstantiationException, IllegalAccessException, InvocationTargetException, NoSuchMethodException{
-
-
- Class classType=obj.getClass();
- System.out.println("該對象的類型是:"+classType.toString());
-
-
- Object objectCopy=classType.getConstructor(new Class[]{}).newInstance(new Object[]{});
-
-
- Field[] fields=classType.getDeclaredFields();
-
- for(int i=0;i
-
- Field field=fields[i];
-
- String fieldName=field.getName();
- String stringLetter=fieldName.substring(0, 1).toUpperCase();
-
-
- String getName="get"+stringLetter+fieldName.substring(1);
- String setName="set"+stringLetter+fieldName.substring(1);
-
-
- Method getMethod=classType.getMethod(getName, new Class[]{});
- Method setMethod=classType.getMethod(setName, new Class[]{field.getType()});
-
-
- Object value=getMethod.invoke(obj, new Object[]{});
- System.out.println(fieldName+" :"+value);
-
-
- 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 {
-
-
-
-
-
- public static void main(String[] args) throws ClassNotFoundException {
-
- Class classType=Class.forName("java.lang.String");
-
- Object array= Array.newInstance(classType,10);
-
-
- Array.set(array, 5, "hello");
-
- String s=(String)Array.get(array, 5);
-
- System.out.println(s);
-
-
-
- for(int i=0;i<((String[])array).length;i++){
-
- String str=(String)Array.get(array, i);
-
- System.out.println(str);
- }
-
- }
-
- }
聲明:JavaEye文章版權屬于作者,受法律保護。沒有作者書面許可不得轉載。