反射

来源:互联网 发布:淘宝店铺起名字 编辑:程序博客网 时间:2024/06/04 19:46

如果想要对指定名称的字节码文件进行加载并获取其中的内容并调用。怎么实现呢?这时就使用到了反射技术。


Class类

提供获取字节码文件中的内容。比如:名称,字段,构造函数,一般函数。

该类就可以获取字节码文件中的所有内容,那么反射就是依靠该类完成的。

想要对一一个类文件进行解析,只有获取到该类的字节码文件对象即可。


Java 反射机制

在运行状态中,对于任意一个类(class文件),都能够知道这个类的所有属性和方法,
 对于任意一个对象,都能够调用它的任意一个方法和属性,这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
 动态获取类的信息。
 类的剖析,想要剖析必须要有字节码文件对象,如何获取字节码文件对象呢?
 
public class ReflectDemo {


public static void main(String[] args) throws Exception {
getFieldDemo();
}
/*获取字节码对象的方式:
* 1 Object类中的getClass()方法
* 想要用这种方式,必须要明确具体的类,并创建对象。

* 2 任何数据类型都具备一个静态的属性,class来获取其对应的Class对象。
* 相对简单还是要明确用到类中的静态成员。

* 3 只要通过给定的类的字符串名称就可以获取该类,更为扩展。
* 可用Class类中的方法完成。
* 该方法就是forName
* 这种方式只要有名称即可,更为方便扩展性更强

* */
public static void getClassObject_1(){
Person p = new Person();
Class clazz = p.getClass();
Person p1 = new Person();
Class clazz1 = p1.getClass();
System.out.println(clazz==clazz1);
}
public static void getClassObject_2(){
Class clazz = Person.class;

}
public static void getClassObject_3() throws ClassNotFoundException{
String str = "com.reflect.demo.Person";
Class clazz = Class.forName(str);
System.out.println(clazz);
}
public static void createNewObject() throws Exception{
//早期:new,先根据被new的类的名称找寻该类的字节码文件,并加载进内存。
//并创建该字节码文件对象,并接者创建该字节文件的对应的Person
com.reflect.demo.Person p = new com.reflect.demo.Person();

//现在:
String str = "com.reflect.demo.Person";

//找寻该名称类文件,并加载进内存,并产生Class对象。
Class clazz = Class.forName(str);

//如何产生该类的对象呢?
Object obj = clazz.newInstance();//空参数的构造函数


//当获取指定名称对应类中的所体现的对象时,而该对象初始化不使用空参数构造函数怎么办?
//既然是通过指定的构造函数进行对象的初始化,所以应该获取到该构造函数。通过字节码文件对象即可完成。
//该方法是:getConstructor(paramterTypes)
Constructor con = clazz.getConstructor(String.class,int.class);

//通过该构造器对象的newInstance方法进行对象的初始化。
Object obj1 = con.newInstance("小明",38);
 
}

//获取字节码文件中的字段
public static void getFieldDemo() throws Exception{
Class clazz = Class.forName("com.reflect.demo.Person");
Field fie = null;///clazz.getField("age");//只能获取公有的
fie = clazz.getDeclaredField("age");//只获取本类,但包含私有。
//对私有字段的访问取消权限检查
fie.setAccessible(true);
Object obj = clazz.newInstance();
fie.set(obj, 89);
Object o = fie.get(obj);
System.out.println(o);

}

//获取指定Class中的公共函数。
public static void getMethodDemo() throws Exception{
Class clazz = Class.forName("com.reflect.demo.Person");
Method[] me = clazz.getMethods();//获取公有的方法。
me = clazz.getDeclaredMethods();//获取本类全部方法,只包含私有。
Method method = clazz.getMethod("show", null);//无参数
Object obj = clazz.newInstance();
method.invoke(obj, null);

Method me1 = clazz.getMethod("paraMethod", String.class,int.class);//有参数
me1.invoke(obj, "小强",89);
}


}

运行中想要扩展功能时,想做到不修改源代码,此时需要两个东西,一个是接口,一个就是配置文件。

反射练习:

public class ReflectTest {


public static void main(String[] args) throws IOException, ClassNotFoundException, InstantiationException, IllegalAccessException {
MainBoard mb = new MainBoard();
mb.run();

File file = new File("prop.properties");
Properties p = new Properties();
FileInputStream in = new FileInputStream(file);
p.load(in);
for(int x = 0;x<p.size();x++){
String str = p.getProperty("pci"+(x+1));
Class clazz = Class.forName(str);
Pci obj = (Pci)clazz.newInstance();//多态
mb.usePci(obj);

}


}


}

public class MainBoard {
public void run(){
System.out.println("主板....运行");
}
public void usePci(Pci p){
p.open();
p.close();
}
}

public interface Pci {
public void open();
public void close();


}

public class SoundCard implements Pci {


public void open() {
    System.out.println("sound open");
}

public void close() {
System.out.println("sound close");
}
}

配置文件信息:

pci1=com.reflect.demo.SoundCard

0 0
原创粉丝点击