反射

反射是在运行时获取类型的信息,再根据这些信息进行操作。

一、Class类

每个已加载的类在内存中都有一份类信息,每个对象都有指向它的类信息的引用。

在Java中,类信息对应的类就是java.lang.Class(注意不是小写的class),Object方法:

public final native Class<?> getClass()

Class是泛型类,还有一种获取Class方法:

Class<Date> cls = Date.class

接口也有Class对象:

Class<Comparable> cls = Comparable.class;

基本类型没有getClass方法,但也都有对应的Class对象,类型参数为相应的包装类型:

Class<Integer> intCls = int.class;
Class<Byte> byteCls = byte.class;
Class<Character> charCls = char.class;
Class<Double> doubleCls = double.class;

void也有:

Class<Void> voidCls = void.class;

对于数组每个维度都有一个:

String[] strArr = new String[10];
int[][] twoDimArr = new int[3][2];
int[] oneDimArr = new int[10];
Class<? extends String[]> strArrCls = strArr.getClass();
Class<? extends int[][]> twoDimArrCls = twoDimArr.getClass();
Class<? extends int[]> oneDimArrCls = oneDimArr.getClass();

根据类名加载Class:

Class<?> cls = Class.forName("java.util.HashMap");

下面介绍Class的一些方法。

1.名称信息

public String getName()
public String getSimpleName()
public String getCanonicalName()
public Package getPackage()

2.字段信息

类中定义的静态和实例变量被称为字段,在Java中用Field表示,位于包java.lang.reflect。Class中获取字段信息的方法:

//返回所有的public字段,包括其父类的,如果没有字段,返回空数组
public Field[] getFields()
//返回本类声明的所有字段,包括非public的,但不包括父类的
public Field[] getDeclaredFields()
//返回本类或者父类中指定名字的public字段,找不到抛异常
public Field getField(String name)
//根据名字找本类的字段
public Field getDeclaredField(String name)

Field也有很多方法获取字段信息:

//获取字段名称
public String getName()
//判断当前程序是否有该字段的访问权限
public boolean isAccessible()
//flag设为true表示允许读写非public的字段
public void setAccessible(boolean flag)
//获取指定对象obj中该字段的值
public Object get(Object obj)
//将指定对象obj中该字段的值设为value
public void set(Object obj, Object value)

在上面的set/get方法中,对于静态变量,obj被忽略,设置为null。

其他方法:

public int getModifiers()
public Class<?> getType()
public void setBoolean(Object obj, boolean z)
public boolean getBoolean(Object obj)
public void setDouble(Object obj, double d)
public double getDouble(Object obj)
public <T extends Annotation> T getAnnotation(Class<T> annotationClass)
public Annotation[] getDeclaredAnnotations()
        try {
Field f = Test.class.getDeclaredField("MAX_COUNT");
int mod = f.getModifiers();
System.out.println(Modifier.toString(mod)); //private static final
System.out.println(Modifier.isPublic(mod));//false
} catch (NoSuchFieldException e) {
e.printStackTrace();
}

3.方法信息

用类Method表示,Class有如下方法:

public Method[] getMethods()
public Method[] getDeclaredMethods()
public Method getMethod(String name, Class<?>... parameterTypes)
public Method getDeclaredMethod(String name, Class<?>... parameterTypes)

Method类中的方法有:

public String getName()
public void setAccessible(boolean flag)
public Object invoke(Object obj, Object... args) throws
IllegalAccessException, Illegal-ArgumentException, InvocationTargetException

对于invoke方法,如果Method为静态方法,obj被忽略,传入null。args可以为null,

或者为空数组。方法的返回值被包装为Object返回,如果实际方法调用抛出异常,异常

被包装为InvocationTargetException重新抛出,可以通过getCause方法得到原异常。

4.创建对象和构造方法

Class有一个可以创建对象的方法:

public T newInstance() throws InstantiationException, IllegalAccessException

它会调用默认的构造方法,如果没有,抛出异常。

newInstance()方法只能使用默认的构造方法。Class还有一些获取其他构造方法的方法:

public Constructor<?>[] getConstructors()
public Constructor<?>[] getDeclaredConstructors()
public Constructor<T> getConstructor(Class<?>... parameterTypes)
public Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes)

类Constructor表示构造方法,通过它可以创建对象,方法为:

public T newInstance(Object ... initargs) throws InstantiationException,
IllegalAccessException, IllegalArgumentException, InvocationTargetException

例子:

Constructor<StringBuilder> contructor= StringBuilder.class
.getConstructor(new Class[]{int.class});
StringBuilder sb = contructor.newInstance(100);

5.类型检查和转换

如果检查的类型是动态的,可以使用Class类的如下方法:

public native boolean isInstance(Object obj);
Class cls = Class.forName("java.util.ArrayList");
if(cls.isInstance(list)){
System.out.println("array list");
}

动态的强制类型转换,可以使用Class方法:

public T cast(Object obj)

判断Class之间的关系

//检查参数类型cls能否赋值给当前class类型的变量
public native boolean isAssignableFrom(Class<?> cls);

6.Class的类型信息

public native boolean isArray()
public native boolean isPrimitive() //是否是基础类型
public native boolean isInterface()
public boolean isEnum()
public boolean isAnnotation()
public boolean isAnonymousClass() //是否是匿名类
public boolean isMemberClass() //是否是成员类,成员类定义在方法外,不是匿名类
public boolean isLocalClass() //是否是本地类,本地类定义在方法内,不是匿名类

7.类的声明信息

Class的其他方法:

public native int getModifiers()
public native Class<? super T> getSuperclass()
//对于类,为自己声明实现的所有接口,对于接口为直接扩展的接口,不包括父类继承的
public native Class<?>[] getInterfaces();
//自己声明的注解
public Annotation[] getDeclaredAnnotations()
//所有注解包括继承得到的
public Annotation[] getAnnotations()
//获取或者检查指定类型的注解
public <A extends Annotation> A getAnnotation(Class<A> annotationClass)
public boolean isAnnotationPresent(
Class<? extends Annotation> annotationClass)

8.类的加载

Class有两个静态方法,可以根据类名加载类:

public static Class<?> forName(String className) //这里的className与Class.getName()返回的值一致
public static Class<?> forName(String name, boolean initialize, ClassLoader loader) //inntialize表示加载后,是否执行类的初始化代码(如static语句块)

第一个方法相当于调用:

Class.forName(className, true, currentLoader)

9.反射与数组

对于数组类型,有一个专门的方法,可以获取它的元素类型:

public native Class<?> getComponentType()

例如:

String[] arr = new String[]{};
System.out.println(arr.getClass().getComponentType());//class java.lang.String

java.lang.reflect包中有一个针对数组的专门类Array,提供了对于数组的一些反射支持,主要方法有:

//创建指定元素类型、长度的数组
public static Object newInstance(Class<?> componentType, int length)
//创建多维数组
public static Object newInstance(Class<?> componentType, int... dimensions)
//获取数组array指定索引位置index处的值
public static native Object get(Object array, int index)
//修改数组array指定的索引位置的index处的值为value
public static native void set(Object array, int index, Object value)
//返回数组的长度
public static native int getLength(Object array)

Array也支持各种基本类型操作数组元素

public static native double getDouble(Object array, int index)
public static native void setDouble(Object array, int index, double d)
public static native void setLong(Object array, int index, long l)
public static native long getLong(Object array, int index)

10.反射与枚举

枚举类型也有一个专门的方法,可以获取所有的枚举常量:

public T[] getEnumConstants()

二、反射与泛型

我们曾经说过,泛型参数在运行时会被擦除,其实在类信息

Class中仍然有关于泛型的一些信息,可以通过反射得到。

获取类的泛型参数的Class实例方法:

public TypeVariable<Class<T>>[] getTypeParameters()

Field有如下方法:

public Type getGenericType()

Method有如下方法:

public Type getGenericReturnType()
public Type[] getGenericParameterTypes()
public Type[] getGenericExceptionTypes()

Constructor有如下方法:

public Type[] getGenericParameterTypes()

其中Type是一个接口,Class实现了Type,Type的其他子接口还有:

TypeVariable:类型参数,可以有上界,比如T extends Number

ParameterizedType :参数化类型,有原始类型和具体的类型参数比如,List

WildcardType :通配符类型,比如?、?extends Number、 ? super Integer

三、总结

不建议使用反射,理由如下:

1)没有编译器检测,容易出错

2)性能相对低下

所以,如果能用接口实现同样的灵活性,就不要使用反射。

Java笔记(十九) 反射的更多相关文章

  1. Java笔记(十九)……多线程

    概述 进程: 是一个正在执行中的程序 每一个进程执行都有一个执行顺序,该执行顺序是一个执行路径,或者叫一个控制单元 线程: 就是进程中的一个独立的控制单元,线程在控制着进程的执行 一个进程中至少有一个 ...

  2. python3.4学习笔记(十九) 同一台机器同时安装 python2.7 和 python3.4的解决方法

    python3.4学习笔记(十九) 同一台机器同时安装 python2.7 和 python3.4的解决方法 同一台机器同时安装 python2.7 和 python3.4不会冲突.安装在不同目录,然 ...

  3. “全栈2019”Java第九十九章:局部内部类与继承详解

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...

  4. “全栈2019”Java第二十九章:数组详解(中篇)

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...

  5. “全栈2019”Java第十九章:关系运算符、条件运算符和三元运算符

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...

  6. (C/C++学习笔记) 十九. 模板

    十九. 模板 ● 模板的基本概念 模板(template) 函数模板:可以用来创建一个通用功能的函数,以支持多种不同形参,进一步简化重载函数的函数体设计. 语法: template <<模 ...

  7. java笔记十:java中的反射

    Java中,反射是一种强大的工具.它使您能够创建灵活的代码,这些代码可以在运行时装配,无需在组件之间进行源代表链接.反射允许我们在编写与执行时,使我们的程序代码能够接入装载到JVM中的类的内部信息,而 ...

  8. Java基础学习笔记十九 IO

    File IO概述 回想之前写过的程序,数据都是在内存中,一旦程序运行结束,这些数据都没有了,等下次再想使用这些数据,可是已经没有了.那怎么办呢?能不能把运算完的数据都保存下来,下次程序启动的时候,再 ...

  9. Java基础学习笔记十九 File

    IO概述 回想之前写过的程序,数据都是在内存中,一旦程序运行结束,这些数据都没有了,等下次再想使用这些数据,可是已经没有了.那怎么办呢?能不能把运算完的数据都保存下来,下次程序启动的时候,再把这些数据 ...

  10. Java学习笔记十九:Java中的访问控制修饰符

    Java中的访问控制修饰符 一:Java修饰符的种类: 访问修饰符 非访问修饰符 修饰符用来定义类.方法或者变量,通常放在语句的最前端.我们通过下面的例子来说明: public class Hello ...

随机推荐

  1. json数据转为Map集合

    我是利用Gson来弄得Gson gson = new Gson();//显得出HashMap的类型Type type = new TypeToken<HashMap<String, Str ...

  2. shell 脚本加密

    日常编写shell脚本时会写一些账号和密码写入脚本内,但是不希望泄露账号密码,所以对shell脚本进行加密变成可执行文件. 主要使用 shc 对 Linux shell 脚本加密,shc是一个专业的加 ...

  3. Python杂写1

    一:编程及编程语言介绍 编程的目的:人把自己的思想流程表达出来,让计算机按照这种思想去做事,把人给解放出来. 编程语言:简单的说就是一种语言,是人和计算机沟通的语言. 编程:例如Python,利用Py ...

  4. JavaScript 高级程序设计第二版

    20.4 部署 20.4.1 构建 构建过程始于在源控制中定义用于存储文件的逻辑结构.最好避免使用一个文件存放所有的JavaScript,遵循以下面向对象语言中的典型模式:将每个对象或自定义了类别分别 ...

  5. Python知乎热门话题数据的爬取实战

    import requestsfrom pyquery import PyQuery as pq url = 'https://www.zhihu.com/explore'headers = { 'u ...

  6. Django认证系统auth认证

    使用Django认证系统auth认证 auth认证系统可以处理范围非常广泛的任务,且具有一套细致的密码和权限实现.对于需要与默认配置不同需求的项目,Django支持扩展和自定义认证;会将用户信息写入到 ...

  7. 使用Docker方式运行Mysql(MariaDB)

    两者差不多.我使用的是MariaDB. 下面的docker命令,挂了数据,配置,映射了端口,指定了root密码,服务端编码. 蛮快的! docker run \ --name mariadb \ -v ...

  8. OpenJDK-study-001 windows上安装Mercurial 4.4.1 克隆OPENJDK版本库

     下载安装 1.下载Mercurial 进入https://www.mercurial-scm.org/wiki/Mercurial下载,windows上傻瓜式安装的,安装好之后,命令行进入安装目录, ...

  9. RN错误随笔 - Unable to resolve module 'AccessibilityInfo'

    错误信息:.React Native 运行报错:Unable to resolve module 'AccessibilityInfo' 可以看到在 异常的返回的JSON 结构中给出了推荐的解决方法 ...

  10. Netty断线重连

    Netty断线重连 最近使用Netty开发一个中转服务,需要一直保持与Server端的连接,网络中断后需要可以自动重连,查询官网资料,实现方案很简单,核心思想是在channelUnregistered ...