前段時間看了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