java的反射機制

前段時間看了java的反射機制的內容,但是好像僅停留在理論層面,下面簡單的以代碼的形式介紹下,自己的理解吧。

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


public class reflectDemo {
    public static void main(String[]args)  throws  Exception
    {
        //getFieldDemo();

        getMethoddemo2();
    }

    private static void getMethoddemo2() throws Exception {
        Class clazz = Class.forName("InterviewDirectory.reflect.person");
        Method method = clazz.getMethod("getAge",null);   //獲取空參數的一般方法
        Object obj = clazz.newInstance();
        method.invoke(obj,null);

        Method method1 = clazz.getMethod("setAge", int.class);
        Object object = clazz.newInstance();
        method1.invoke(object, 4576);

    }

    //獲取字節碼文件的字段。
    private static void getFieldDemo() throws Exception {
        Class clazz = Class.forName("InterviewDirectory.reflect.person");
        Field field = null;
        //field = clazz.getField("age");//只能獲取公有的。
        field = clazz.getDeclaredField("age");  //只獲取本類但包含私有。
        //反射破壞了對象的封裝特性,因爲可以訪問私有成員。
        System.out.println(field);
        //對私有字段的訪問取消權限檢查,暴力訪問。
        field.setAccessible(true);
        //直接訪問字段。默認爲0.
        Object obj = clazz.newInstance();
        Object o = field.get(obj);
        System.out.println(o);
        //反射設置字段的值。不用使用對象進行設置。
        field.set(obj, 89);
        Object o1 = field.get(obj);
        System.out.println(o1);


    //    當獲取指定名稱對應類中所體現的對象時,
        // 而對象初始化不使用空參數的構造函數怎麼辦?
    //    可以獲取指定的構造函數,通過字節碼文件對象即可完成。
        // 該方法: getConstructor(paramterType)

        String name = "InterviewDirectory.reflect.person";
    //    尋找該名稱的文件,並加載進內存,併產生class對象。
        Class class1 = Class.forName(name);
        //獲取指定構造函數對象。
        Constructor constructor = class1.getConstructor(int.class);
    //    通過該構造器對象的newInstance方法進行對象初始化。
        Object obj1 = constructor.newInstance(46);
        Object o2 = field.get(obj1);
        System.out.println(o2);


        Method[] methods = class1.getMethods();//獲取公有的方法
        methods = class1.getDeclaredMethods();    //只獲取本類的方法,包含私有
        for(Method me : methods)
        {
            System.out.println(me);
        }
    }
}

反射破壞了對象的封裝特性,因爲可以訪問私有成員。

反射實現多態:

首先主板類:

package InterviewDirectory.reflect;

/**
 * Created by huali on 2018/5/19.
 */
public class MainBoard {
    public void run() {
        System.out.println("main board run!!1");
    }

    public void UsePCI(PCI pci) {
        pci.open();
        pci.close();
    }
}

然後接口類:

package InterviewDirectory.reflect;

/**
 * Created by huali on 2018/5/19.
 */
public interface PCI {
    public void open();
    public  void close();
}

然後網卡類:

package InterviewDirectory.reflect;

/**
 * Created by huali on 2018/5/19.
 */
public class netcard implements PCI{
    @Override
    public void open() {
        System.out.println("netcars open");
    }

    @Override
    public void close() {
        System.out.println("netcard close");
    }
}

然後聲卡類:

package InterviewDirectory.reflect;

/**
 * Created by huali on 2018/5/19.
 */
public class soundcard implements PCI{
    @Override
    public void open() {
        System.out.println("soundcard open");
    }

    @Override
    public void close() {
        System.out.println("soundcard close");
    }
}

然後配置文件

pci1 = InterviewDirectory.reflect.soundcard
pci2 = InterviewDirectory.reflect.netcard

然後測試類:

package InterviewDirectory.reflect;

import java.io.File;
import java.io.FileInputStream;
import java.util.Properties;

/**
 * Created by huali on 2018/5/19.
 */
public class reflectDemo2 {
    public static void main(String[]args)  throws  Exception
    {
        MainBoard md = new MainBoard();
        md.run();
    //    每次添加一個設備都需要修改代碼傳遞一個新創建的對象,(也就是多態問題)
    //    md.UsePCI()
    //    不用new老獲得,只獲得class的文件,其內部實現創建對象的動作。
        File configfile = new File("E:\\JavaBook\\MyprojectTouTiao\\MyAlgorithms\\src\\InterviewDirectory\\reflect\\PCI.properties");
        Properties prop = new Properties();
        FileInputStream fis = new FileInputStream(configfile);
        prop.load(fis);
        for(int x =0;x<prop.size();x++)
        {
            String pciname = prop.getProperty("pci"+(x+1));
            Class clazz = Class.forName(pciname);
            PCI p = (PCI)(clazz.newInstance());
            md.UsePCI(p);
        }
        fis.close();

    }

}

結果展示:

main board run!!1
soundcard open
soundcard close
netcars open
netcard close
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章