一、Java反射定义
反射即反向探知,有点像考古学家根据发掘的物品来探知以前的事情
指在Java程序运行状态中,
- 对于给定的一个类(Class)对象,可以获得这个类(Class)对象的所有属性和方法;
- 对于给定的一个对象(new XXXClassName<? extends Object>),都能够调用它的任意一个属性和方法.
这种动态获取类的内容以及动态调用对象的方法和获取属性的机制.就叫做JAVA的反射机制 如下案例
package com.example.demo;
public class Person {
public void say(){
System.out.println("Person中的say方法");
};
}
public static void main(String[] args) throws Exception {
Class clazz = Person.class;
Person person = (Person) clazz.newInstance();
System.out.println(clazz.getName());
System.out.println(clazz.getPackage());
System.out.println(clazz.getClassLoader());
System.out.println(clazz.getSuperclass());
Method method = clazz.getDeclaredMethod("say");
method.invoke(person);
}
通过反射对象得到构造方法\成员变量\成员方法
Constructor<?>[] | getConstructors() 得到构造方法 |
---|
Field[] | getDeclaredFields() 得到成员变量 | Method[] | getDeclaredMethods() 得到成员方法 | Class<?>[] | getInterfaces() 得到接口 | Class<? super T> | getSuperclass() 得到父类 | Package | getPackage() 得到包对象 | int | getModifiers() Java语言修饰符 Modifier的toString(int mod) | String | getName() 得到类名称 |
二、Java反射的优缺点
优点 增加程序的灵活性,避免将固有的逻辑程序写死到代码里 代码简洁,可读性强,可提高代码的复用率 缺点 相较直接调用在量大的情景下反射性能下降 内部暴露和安全隐患
public interface Office {
void toPDF();
}
public class Word implements Office {
@Override
public void toPDF() {
System.out.println("Word 2 PDF ");
}
}
public class Excel implements Office {
@Override
public void toPDF() {
System.out.println(" Excel 2 PDF");
}
}
package com.example.demo.fashe1;
public class Main {
public static void main(String[] args) {
String key = "word";
}
public static Office getInstanceByKey(String key){
if("word".equals(key)){
return new Word();
}
if("excel".equals(key)){
return new Excel();
}
return null;
}
public static Office getInstanceReflectByKey(String key){
String packageName = "com.example.demo.fashe1";
Office office = null;
try {
Class clazz = Class.forName(packageName+"."+key);
office = (Office)clazz.newInstance();
}catch (Exception e){
e.printStackTrace();
}
return office;
}
}
三、反射到底慢在哪些地方
寻找类Class字节码的过程 安全管理机制的权限验证等等 若需要调用native方法调用时JNI接口的使用
public static void main(String[] args) {
String key = "word";
long startTime = System.currentTimeMillis();
for (int i = 0; i < 1000000 ; i++) {
getInstanceByKey(key);
}
long endTime = System.currentTimeMillis();
System.out.println("总计花费时间:" + (endTime - startTime));
}
通过new的方式创建1000000个对象只需要花费9毫秒
如果通过反射的方式的话
public static void main(String[] args) {
String key = "Word";
long startTime = System.currentTimeMillis();
for (int i = 0; i < 1000000 ; i++) {
getInstanceReflectByKey(key);
}
long endTime = System.currentTimeMillis();
System.out.println("总计花费时间:" + (endTime - startTime));
}
大家能发现差距还是非常大的~
源码层面
@CallerSensitive
public static Class<?> forName(String className)
throws ClassNotFoundException {
Class<?> caller = Reflection.getCallerClass();
return forName0(className, true, ClassLoader.getClassLoader(caller), caller);
}
private static native Class<?> forName0(String name, boolean initialize,
ClassLoader loader,
Class<?> caller)
throws ClassNotFoundException;
四、Class内部
在编写每个类实例中,都会定义这个类的包名,类名,访问域,特征符,构造器,字段,函数,父类,接口等等内容.
这些内容在我们的Class类中都提供了对应的获取方法进行获取.
//clazz 代表对应的Class类实例
五、反射的基本操作
5.1 获取类对象的四种方式 Class clazz = Person.class;
Class clazz2 = new Person().getClass();
Class clazz3 = Class.forName("com.example.demo.fashe.Person");
Class clazz4 = Demo02.class.getClassLoader().loadClass("com.example.demo.fashe.Person");
5.1.2 获取类对象的四种方式
int modifier = clazz.getModifiers();
Package aPackage = clazz.getPackage();
String fullClassName = clazz.getName();
String simpleName = clazz.getSimpleName();
ClassLoader classLoader = clazz.getClassLoader();
Class[] interfaces = clazz.getInterfaces();
Class superclass = clazz.getSuperclass();
Annotation[] annotations = clazz.getAnnotations();
5.2 类的属性操作 Person person = (Person) clazz.newInstance();
Field[] fields = clazz.getFields();
Field[] declaredFields = clazz.getDeclaredFields();
Field nameField = clazz.getDeclaredField("name");
int modifiers = nameField.getModifiers();
nameField.setAccessible(true);
nameField.set(person,"平安");
nameField.set(null,"静态字段赋值");
5.3 类的方法操作
Method[] methods = clazz.getMethods();
Method[] declaredMethods = clazz.getDeclaredMethods();
Method say = clazz.getMethod("say", String.class);
Method say1 = clazz.getDeclaredMethod("say");
int modifiers1 = say.getModifiers();
Object 小米666 = say.invoke(person, "小米666");
say.setAccessible(true);
say.invoke(null);
5.4 构造器的操作 Constructor[] cons = clazz.getConstructors();
Constructor[] cons1 = clazz.getDeclaredConstructors();
Constructor conNoParam= clazz.getDeclaredConstructor();
Constructor con= clazz.getDeclaredConstructor(String.class,String.class);
int modifers = con.getModifiers();
conNoParam.newInstance();
con.setAccessible(true);
con.newInstance("abc","bbb");
Person.class.newInstance();
newInstance();方法的本质
六、反射破坏了单例模式
定义一个简单的单例模式
public class PersonSingle {
private static PersonSingle instance;
private PersonSingle(){
}
public static synchronized PersonSingle getInstance(){
if(instance == null){
instance = new PersonSingle();
}
return instance;
}
}
通过反射可以创建多个实例,从而破坏单例的设计
public static void main(String[] args) throws Exception{
PersonSingle s1 = PersonSingle.getInstance();
System.out.println(s1);
Class<? extends PersonSingle> aClass = s1.getClass();
Constructor<? extends PersonSingle> declaredConstructor = aClass.getDeclaredConstructor();
declaredConstructor.setAccessible(true);
Thread.sleep(1000);
PersonSingle s2 = declaredConstructor.newInstance(null);
System.out.println(s2);
}
那么如何防止这种破坏呢,其实很简单我们只需要在私有构造中加个判断就可以了,如下
private PersonSingle(){
if(PersonSingle.instance != null){
throw new RuntimeException("实例已经创建,不允许再创建了...");
}
}
|