一切皆對象
java中所有的東西都有對應的對象,包括包,類临梗,方法宇挫,字段捧书,注解吹泡,一切皆對象。
包package對應的類java.lang.reflect.Package
類class對應的java.lang.reflect.Class
字段field對應的類是java.lang.reflect.Field
方法method對應的類是java.lang.reflect.Method
反射
獲取對象的類對象经瓷,操作對象的方法爆哑,字段。
//獲取已知類對象
Class<String> stringClz=String.class;
//獲取未知類對象
Class<?> class=Class.forName("com.demo.xx.XX");
反射類Class
- 獲取包 getPackage可以拿到類所在的包
- 獲取類名 getName() 返回的字符串包含了包名
- 獲取字段 getField() getDeclaredField() 區(qū)別就是獲取public 還是protect private
- 獲取方法 getMethod() getDeclaredMethod() 區(qū)別就是獲取public 還是protect private
- 創(chuàng)建對象 newInstance() 這個必須保證有默認構造方法
- 獲取構造方法 getConstructor()
- getSuperclass 獲取父類 默認父類是Object
- getInterfaces 獲取實現(xiàn)的接口
反射類字段
Class clz=Class.forName("com.demo.reflect.A");
/**
* 獲取構造函數 constructor(String)
* 并創(chuàng)建對象
*/
Object obj=clz.getConstructor(String.class).newInstance("哈哈");
//拿到private 或者protect的字段 name
Field nameField=clz.getDeclaredField("name");
//設置訪問權限
nameField.setAccessible(true);
//取值
System.out.println(nameField.get(obj));
反射類方法
Class clz = Class.forName("com.demo.reflect.A");
/**
* 獲取構造函數 constructor(String)
* 并創(chuàng)建對象
*/
Object obj=clz.getConstructor(String.class).newInstance("哈哈");
//拿到private 或者protect的字段 name
Field nameField=clz.getDeclaredField("name");
//設置訪問權限 這步是必須的 因為默認private和protect字段是無法取值和賦值的
nameField.setAccessible(true);
//取值
System.out.println(nameField.get(obj));
//獲取setName(String)方法
Method setNameMethod=clz.getMethod("setName", String.class);
//調用并傳參數
setNameMethod.invoke(obj,"設置的值");
//獲取getName方法
Method getNameMethod=clz.getMethod("getName");
//調用getName方法
System.out.println(getNameMethod.invoke(obj));
反射的作用
解耦且基于配置,依賴反轉
讓Computer類只依賴接口不依賴具體實現(xiàn)
public interface Cpu {
String getName();
double core();
}
public interface Disk {
int size();
}
public interface Display {
String show();
}
public class Computer {
public Cpu cpu;
public Disk disk;
public Display display;
public Computer(Cpu cpu, Disk disk, Display display) {
this.cpu = cpu;
this.disk = disk;
this.display = display;
}
public void showMessage(){
System.out.println("CPU:"+this.cpu.core()+"核心 "+this.cpu.getName());
System.out.println("硬盤:"+disk.size()/1024+"m");
System.out.println("顯示器:"+display.show());
}
}
實現(xiàn)類
public class I7Cpu implements Cpu {
@Override
public String getName() {
return "i7";
}
@Override
public double core() {
return 4;
}
}
public class I9Cpu implements Cpu {
@Override
public String getName() {
return "core i9";
}
@Override
public double core() {
return 9;
}
}
public class SansungDisplay implements Display {
@Override
public String show() {
return " 三星 1980*1070 顯示器";
}
}
public class SonyDisk implements Disk {
@Override
public int size() {
return 2048;
}
}
配置文件 computerA.txt
com.demo.reflect.impl.I9Cpu
com.demo.reflect.impl.SonyDisk
com.demo.reflect.impl.SansungDisplay
main方法
public static Object getClass(String str) throws ClassNotFoundException, IllegalAccessException, InstantiationException {
return Class.forName(str).newInstance();
}
public static void main(String agrs[]) throws Exception {
BufferedReader bufferedReader=new BufferedReader(new InputStreamReader(new FileInputStream("src/computerA.txt")));
String cpu=bufferedReader.readLine();
String disk=bufferedReader.readLine();
String display=bufferedReader.readLine();
bufferedReader.close();
new Computer((Cpu) getClass(cpu),(Disk) getClass(disk),(Display) getClass(display)).showMessage();
}
通過讀取配置文件自動構建出Computer類的實例舆吮,不修改程序的情況下 只需要修改配置文件就可以達到修改Cpu Disk和Display