Java中的反射機制(二) 一個利用反射進行物件拷貝的例子

weixin_34344677發表於2013-01-26

 

  本部落格已有的關於反射的討論:

  http://www.cnblogs.com/mengdd/archive/2012/08/18/2645553.html

  http://www.cnblogs.com/mengdd/archive/2013/01/26/2877972.html

 

  如下例程ReflectTester類進一步演示了Reflection API的基本使用方法。

  ReflectTester類有一個copy(Object object)方法,這個方法能夠建立一個和引數object同樣型別的物件,然後把object物件中的所有屬性拷貝到新建的物件中,並將它返回。

  這個例子只能複製簡單的類,假定類的每個屬性都有public型別的getXXX()和setXXX()方法。

  

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;

public class ReflectTester
{

    // 該方法實現對Customer物件的拷貝操作
    public Object copy(Object object) throws Exception
    {
        Class<?> classType = object.getClass();


        /* 生成新的物件的討論
        // 獲得Constructor物件,此處獲取第一個無引數的構造方法的
        Constructor cons = classType.getConstructor(new Class[] {});//不帶引數,所以傳入一個為空的陣列
        // 通過構造方法來生成一個物件
        Object obj = cons.newInstance(new Object[] {});

        // 以上兩行程式碼等價於:
        Object obj11 = classType.newInstance();  // 這行程式碼無法處理建構函式有引數的情況
        
        //用第二個帶引數的構造方法生成物件
        Constructor cons2 = classType.getConstructor(new Class[] {String.class, int.class});
        Object obj2 = cons2.newInstance(new Object[] {"ZhangSan",20});
        
        */
        
        Object objectCopy = classType.getConstructor(new Class[]{}).newInstance(new Object[]{});
        
        //獲得物件的所有成員變數
        Field[] fields = classType.getDeclaredFields();
        for(Field field : fields)
        {
            //獲取成員變數的名字
            String name = field.getName();    //獲取成員變數的名字,此處為id,name,age
            //System.out.println(name);

            //獲取get和set方法的名字
            String firstLetter = name.substring(0,1).toUpperCase();    //將屬性的首字母轉換為大寫            
            String getMethodName = "get" + firstLetter + name.substring(1);
            String setMethodName = "set" + firstLetter + name.substring(1);            
            //System.out.println(getMethodName + "," + setMethodName);
            
            //獲取方法物件
            Method getMethod = classType.getMethod(getMethodName, new Class[]{});
            Method setMethod = classType.getMethod(setMethodName, new Class[]{field.getType()});//注意set方法需要傳入引數型別
            
            //呼叫get方法獲取舊的物件的值
            Object value = getMethod.invoke(object, new Object[]{});
            //呼叫set方法將這個值複製到新的物件中去
            setMethod.invoke(objectCopy, new Object[]{value});

        }

        return objectCopy;

    }

    public static void main(String[] args) throws Exception
    {
        Customer customer = new Customer("Tom",20);
        customer.setId(1L);
        ReflectTester tester = new ReflectTester();
        
        Customer customer2 = (Customer)tester.copy(customer);
        
        System.out.println(customer2.getId() + "," + customer2.getName() + "," + customer2.getAge());
        
    }
}

class Customer
{
    private long id;
    private String name;
    private int age;

    public Customer()
    {
    }

    public Customer(String name, int age)
    {
        this.name = name;
        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;
    }

    public int getAge()
    {
        return age;
    }

    public void setAge(int age)
    {
        this.age = age;
    }
}

 

 

參考資料:

  聖思園張龍老師Java SE系列視訊教程。

相關文章