1. 动态语言 VS 静态语言
[1] 动态语言: 是一类在运行时可以改变其结构的语言: 例如新的函数,对象,甚至代码可以被引进,已有的函数可以被删除或是其他结构上的变化。通俗点说就是运行时代码可以根据某些条件改变目录结构,主要的动态语言有:Object-C, C#, JavaScript, PHP, Python等
[2] 静态语言:与动态语言相对的,运行时结构不可变的语言就是静态语言,如Java, C, C++。
Java不是动态语言,但是Java可以称为“准动态语言”。即Java有一定的动态性,我们可以利用反射机制获得类似动态语言的特性。Java的动态性让编程的时候更灵活。 ===> java Reflection
<1> Reflection(反射)是Java被视为动态语言的关键,反射机制允许程序在执行期间借助于Reflection API 取得任何类内部信息,并能直接操作任意对象内部属性及方法。
例子:
Class c = Class.forName("java.lang.String");
<2> 加载完类之后,在堆内崔的方法区中就产生了一个Class类型的对象(一个类只有一个Class对象),这个对象就包含了完整的类的结构信息。我们可以通过这个对象看到类的结构,这个对象就像一面镜子,透过这个镜子看到类的结构。所以,我们形象的称之为:反射。
我们通过代码了解一下一个类只有一个Class对象
package com.gs.reflection;
//什么叫反射
public class Test02 {
public static void main(String[] args) throws ClassNotFoundException {
//通过反射获取Class对象
Class c1 = Class.forName("com.gs.reflection.User");
System.out.println(c1);
//一个类在内存中只有一个Class对象
// 一个类加载后,类的整个结构会被封装在Class对象中
Class c2 = Class.forName("com.gs.reflection.User");
Class c3 = Class.forName("com.gs.reflection.User");
System.out.println(c2.hashCode());
System.out.println(c3.hashCode());
}
}
// 实体类 pojo,entity
class User{
private String name;
private int id;
private int age;
public User() {
}
public User(String name, int id, int age) {
this.name = name;
this.id = id;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
public String toString() {
return "User{" +
"name='" + name + '\'' +
", id=" + id +
", age=" + age +
'}';
}
}
我们可以看到两个Class对象的hashcode值是相同的,则他们是同一个对象
2. Java反射机制研究及应用
Java反射机制提供的功能:
[1] 在运行时判断任意一个对象所属的类
[2] 在运行时构造任意一个类的对象
[3] 在运行时判断任意一个类具有成员变量和方法
[4] 在运行时获取泛型信息
[5] 在运行时调用任意一个对象的成员变量和方法
[6] 在运行时处理注解
[7] 生成动态代理
Java反射的优点和缺点
优点: 可以实现动态创建对象和编译,体现出很大的灵活性。
缺点:对性能有影响,使用反射基本上是一种解释操作,我们可以告诉JVM,我们希望做什么,并且让他们满足我们的要求,这类操作总是慢于直接执行相同的操作
反射相关的的主要API
[1] java.lang.Class: 代表一个类
//com.gs.reflection.User是我们上面所举例的User对象
Class c1 = Class.forName("com.gs.reflection.User");
[2] java.lang.reflection.Method: 代表类的方法
//获得对象c1的public方法
Method[] methods = c1.getMethods();
//获得本类的所有方法
methods = c1.getDeclaredMethods();
[3] java.lang.reflect.Field: 代表类的成员变量
//只能找到public属性
Field[] fields = c1.getFields();
//找到全部的属性
fields = c1.getDeclaredFields();
[4] java.lang.reflect.Constructor: 代表类的构造器
Constructor[] constructors = c1.getConstructors();
constructors = c1.getDeclaredConstructors();