java反射机制的概述:
1.什么是java反射:
- Reflection(反射)是被视为动态语言的关键,反射机制允许程序在执行期 借助于Reflection API取得任何类的内部信息,并能直接操作任意对象的内 部属性及方法。
- 加载完类之后,在堆内存的方法区中就产生了一个Class类型的对象(一个 类只有一个Class对象),这个对象就包含了完整的类的结构信息。我们可 以通过这个对象看到类的结构。这个对象就像一面镜子,透过这个镜子看 到类的结构,所以,我们形象的称之为:反射。
正常方式:引入需要的"包类"名称----->通过New实例化------>取得实例化对象
反射方式:实例化对象------>getClass()方法-------->得到完整"包类"的名称
2.Java反射机制提供的功能:
- 在运行时判断任意一个对象所属的类
- 在运行时构造任意一个类的对象
- 在运行时判断任意一个类所具有的成员变量和方法
- 在运行时获取泛型信息
- 在运行时调用任意一个对象的成员变量和方法
- 在运行时处理注解
- 生成动态代理
3.反射相关的主要API:
- java.lang.Class:代表一个类
- java.lang.reflect.Method:代表类的方法
- java.lang.reflect.Field:代表类的成员变量
- ava.lang.reflect.Constructor:代表类的构造器
4.对象在反射和没有反射的情况对比:
首先创建一个Person类,类中存在一些基本结构,也有私有的构造器,方法,属性:
public class Person {
private String name;
public int age;
public Person() {
}
public Person(String name, int age) {
this.name = name;
this.age = age;
}
private Person(String name) {
this.name = name;
}
public void show(){
System.out.println("你好");
}
private String showNation(String nation){
System.out.println("我的国籍是:"+nation);
return nation;
}
public String getName() {
return name;
}
public int getAge() {
return age;
}
public void setName(String name) {
this.name = name;
}
public void setAge(int age) {
this.age = age;
}
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
}
接下来对比一下反射与没有反射的区别
public class ReflectionTest {
//反射之前对于Person类的操作
@Test
public void test(){
//1.创建类的对象
Person p1=new Person();
p1.age=10;
//2.通过对象调用其内部的属性方法
System.out.println(p1.toString());
p1.show();
//在Persn类外部是不可以通过对象调用其内部私有结构
//比如,name,showNation()以及私有的构造器
}
//反射之后,对于Person的操作
@Test
public void test1() throws Exception {
Class clazz=Person.class;
//1.通过反射。创建Person类的对象
Constructor constructor = clazz.getConstructor(String.class, int.class);
Object obj = constructor.newInstance("Tom", 20);
Person p=(Person) obj;
System.out.println(p.toString());
//2.通过反射。调用对象指定的属性,方法
//调用属性
Field age = clazz.getDeclaredField("age");
age.set(p,10);
System.out.println(p.toString());
//调用方法
Method show = clazz.getDeclaredMethod("show");
show.invoke(p);
System.out.println("***************************");
//通过反射,可以调用Person类的私有结构。比如:私有的构造器,方法,属性
//调用私有的构造器
Constructor constructor1 = clazz.getDeclaredConstructor(String.class);
constructor1.setAccessible(true);
Person p2 = (Person)constructor1.newInstance("Jerry");
System.out.println(p2);
//调用私有属性
Field name = clazz.getDeclaredField("name");
name.setAccessible(true);
name.set(p2,"XIAOPING");
System.out.println(p2);
//调用私有方法
Method showNation = clazz.getDeclaredMethod("showNation", String.class);
showNation.setAccessible(true);
// showNation.invoke(p2,"中国");//invoke就是调用方法,相当于p2.showNation()
String nation = (String) showNation.invoke(p2, "中国");//invoke的返回值是Object
System.out.println(nation);
}
}
可以总结出来,反射可以调用Person类中的私有的构造器,方法,属性。
疑问:
-
反射机制与面向对象的封装性是不是矛盾的?如何看待两个技术
不矛盾。 -
通过直接new的方式或反射的方式都可以调用公共的结构,开发中到底用哪个?
建议使用new的方式。