java反射

一切皆对象

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

反射类字段

Class clz=Class.forName("com.demo.reflect.A");
/**
 * 获取构造函数 constructor(String)
 * 并创建对象
 */
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)
 * 并创建对象
 */
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类只依赖接口不依赖具体实现


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());
    }
}

实现类

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