java反射意思是,你只要告诉我一个类的名字,我就能拿到你这个类里边的任何东西,包括成员构造方法,变量,其他方法,以及成员变量和方法的修饰符,annotation等等。说白了,就是你写的代码没啥神秘的,jdk知道你写的每个类的所有细节。这就提供给了我们除了用new之外另外一种创建对象的方法。
假设我们定义了一个Person类:
public class Person {
private String name;
private int age;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public void speak() {
System.out.println("LaLaLa");
}
}
第一步,需要通过传进来的的class的名字获取他的Class对象:
Class clazz = Class.forName("se.ericsson.refleaction.Person");
第二步调用这个Class对象的newInstance方法创建对象:
Person person = (Person) clazz.newInstance();
我们还可以通过Class对象拿到类的方法:
Method setName = clazz.getMethod("setName", String.class);
然后调用person对象的这个方法:
setName.invoke(person, "simin");
当然你也可以直接拿到类里边的成员变量:
Field age = clazz.getDeclaredField("age");
然后调用set方法给person设值(如果是private类型需要首先通过setAccessible打开access权限):
age.setAccessible(true);
age.set(person, 30);
java的反射有什么优势呢?根据上边的用法,我们可以知道,你只需要给个Class的名字(或者其他方式,比如说你用一个property文件,key是你自己定义的字符串,value是class的名字,这时候你只要传入key值就行),jdk就能根据这个字符串对应到相应的类,进而构建对象,这叫动态编程,你只需控制字符串的值就能轻松控制你的程序的行为,而不是去改你的程序的源代码。
当然反射用的更多的是在一些大型框架中,比如说:
eclipse的提示功能:在eclipse等IDE中,比如说你想调用person对象的某个方法,你只要写person. IDE就能把person的所有可以调用的方法和成员变量列出来,你只需要选中回车就行,不需要一个字母一个字母的敲这个方法。
spring 的IOT/DI:在springboot工程中你想要新建一个对象,只需要用@Autowired关键字标示这个对象的类名就行:
@Autowired
private InstallationService installationService;
当然你需要有一个实现InstallationService 的Service
@Service
public class InstallationServiceImpl implements InstallationService
springboot加载的时候会帮我们找到合适的实现类,然后通过反射构造出对象传给我们,这叫做控制反转(IOT)或者依赖注入(DI)
Testng/Junit:我们测试代码的每个method需要注明@Test,@BeforeTest,@AfterTest,测试框架就知道先执行哪个方法再执行哪个方法,原因就是反射机制通过获取这些Annotation来觉得方法的调用顺序,进而完成Test的执行
@Test
public void test() throws Exception {
log.info(mapper.selectByPrimaryKey(10000013).getMessage());
}
Herbernate/Mybatis:在这些数据库的操作框架中,我们只需要写一个interface,interface里边列出操作sql的方法和方法的返回值,框架就能自动帮我们完成查询数据库->拿到返回数据->把返回数据转换成对象(Installation)的过程
Installation selectByPrimaryKey(Integer id);
Java Dynamic Proxy: 要想做到动态代理,类名和方法名字是动态传进去的,在代理类中通过调用方法的invoke方法触发传入的类的这个方法实现:
method.invoke(target, args);