什么是反射
反射是一种使java的编译过程从静态编译转变到动态编译的机制,能够有效的降低系统的耦合,是运行中的程序检查自己和软件运行环境的能力,它可以根据它发现的进行改变。通俗的讲就是反射可以在运行时根据指定的类名获得类的信息。要理解反射,首先我们先明确两个概念:
-
静态编译:
在编译时就确定引用类型变量所引用的数据类型,也就是说将引用类型变量的值在编译时就确定下来了,它不会根据程序的运行而发生改变。 -
动态编译:
在运行时确定引用类型变量所引用的数据类型,也就是说引用类型变量的值会根据程序的运行而发生改变。
反射是怎么来的
在我们没有使用反射的时候我们的编程是这样的:
package test;
public class test {
public static void main(String[] args) {
A a1= new B();
a1.see();
A a2 = new C();
a2.see();
}
}
abstract class A{
abstract void see();
}
class B extends A{
@Override
void see() {
System.out.println("This is B");
}
}
class C extends A{
@Override
void see() {
System.out.println("this is C");
}
}
更好一点的话是使用接口来进行引用:
package test;
public class test {
public static void main(String[] args) {
A a1= new B();
a1.see();
A a2 = new C();
a2.see();
}
}
interface A{
void see();
}
class B implements A{
@Override
public void see() {
System.out.println("This is B");
}
}
class C implements A{
@Override
public void see() {
System.out.println("this is C");
}
}
但是使用这种方式来引用的话会出现一些问题,比如,我现在想要使a1引用C的实例,a2引用B的实例,那么我就必须要修改源码,使得a1引用C的实例,a2引用B的实例。但是在实际开发中这是不可能实现的,难道还要用户每次都卸载重装吗?
所以这时候就引入了反射这个概念,我们先来看一下怎么实现的吧:
package fanShe;
//定义一个office接口,使word、ppt、excel三个类实现这个接口
public interface office {
void print();
}
----------------------------------------------------
package fanShe;
public class ppt implements office {
@Override
public void print() {
System.out.println("ppt print");
}
}
----------------------------------------------------
package fanShe;
public class word implements office{
@Override
public void print() {
System.out.println("word print");
}
}
---------------------------------------------------
package fanShe;
public class excel implements office {
@Override
public void print() {
System.out.println("excel print");
}
}
-----------------------------------------------------
//主方法
package fanShe;
public class main {
public static void main(String[] args) {
try {
// 通过Class.forName()方法动态获取要创建的类
office i = (office) Class.forName("fanShe.ppt").newInstance();
i.print();
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
}
通过这种方法创建类只需要更改Class.forName()
中的参数就可以在运行的时候动态的进行类的创建。
还可以再做一些改进:
package fanShe;
//定义一个office接口,使word、ppt、excel三个类实现这个接口
public interface office {
void print();
}
----------------------------------------------------
package fanShe;
public class ppt implements office {
@Override
public void print() {
System.out.println("ppt print");
}
}
----------------------------------------------------
package fanShe;
public class word implements office{
@Override
public void print() {
System.out.println("word print");
}
}
---------------------------------------------------
package fanShe;
public class excel implements office {
@Override
public void print() {
System.out.println("excel print");
}
}
-----------------------------------------------------
//主方法
package fanShe;
import java.io.*;
public class main {
public static void main(String[] args) {
try {
BufferedReader br = new BufferedReader(new InputStreamReader(new FileInputStream("D:\\demo\\javaBase\\src\\config")));
String str = br.readLine();
// 通过Class.forName()方法动态获取要创建的类
office i = (office) Class.forName(str).newInstance();
i.print();
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
}
-----------------------------------------------------
//config文件
fanShe.excel
建立一个外部的配置文件config,通过输入流对文件进行读入,而我们也可以在外部通过改变配置文件的方式来改变程序的运行状态,源代码也没有发生任何更改,这样的一个方法无疑是比较有用的。