Java笔记(十九) 反射
反射
反射是在运行时获取类型的信息,再根据这些信息进行操作。
一、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笔记(十九) 反射的更多相关文章
- Java笔记(十九)……多线程
概述 进程: 是一个正在执行中的程序 每一个进程执行都有一个执行顺序,该执行顺序是一个执行路径,或者叫一个控制单元 线程: 就是进程中的一个独立的控制单元,线程在控制着进程的执行 一个进程中至少有一个 ...
- python3.4学习笔记(十九) 同一台机器同时安装 python2.7 和 python3.4的解决方法
python3.4学习笔记(十九) 同一台机器同时安装 python2.7 和 python3.4的解决方法 同一台机器同时安装 python2.7 和 python3.4不会冲突.安装在不同目录,然 ...
- “全栈2019”Java第九十九章:局部内部类与继承详解
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- “全栈2019”Java第二十九章:数组详解(中篇)
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- “全栈2019”Java第十九章:关系运算符、条件运算符和三元运算符
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- (C/C++学习笔记) 十九. 模板
十九. 模板 ● 模板的基本概念 模板(template) 函数模板:可以用来创建一个通用功能的函数,以支持多种不同形参,进一步简化重载函数的函数体设计. 语法: template <<模 ...
- java笔记十:java中的反射
Java中,反射是一种强大的工具.它使您能够创建灵活的代码,这些代码可以在运行时装配,无需在组件之间进行源代表链接.反射允许我们在编写与执行时,使我们的程序代码能够接入装载到JVM中的类的内部信息,而 ...
- Java基础学习笔记十九 IO
File IO概述 回想之前写过的程序,数据都是在内存中,一旦程序运行结束,这些数据都没有了,等下次再想使用这些数据,可是已经没有了.那怎么办呢?能不能把运算完的数据都保存下来,下次程序启动的时候,再 ...
- Java基础学习笔记十九 File
IO概述 回想之前写过的程序,数据都是在内存中,一旦程序运行结束,这些数据都没有了,等下次再想使用这些数据,可是已经没有了.那怎么办呢?能不能把运算完的数据都保存下来,下次程序启动的时候,再把这些数据 ...
- Java学习笔记十九:Java中的访问控制修饰符
Java中的访问控制修饰符 一:Java修饰符的种类: 访问修饰符 非访问修饰符 修饰符用来定义类.方法或者变量,通常放在语句的最前端.我们通过下面的例子来说明: public class Hello ...
随机推荐
- Vue-cli 创建的项目配置跨域请求(通过反向代理)---配置多个代理--axios请求
问题描述: 使用 Vue-cli 创建的项目,开发地址是 localhost:8080,需要访问 localhost:9000 或https://m.maoyan.com或http://image.b ...
- MySQL数据库权限分类
一.权限表 mysql数据库中的3个权限表:user .db. host 权限表的存取过程是: 1)先从user表中的host. user. password这3个字段中判断连接的IP.用户名.密码是 ...
- Exception in thread "main" java.lang.UnsatisfiedLinkError: org.apache.hadoop.io.nativeio.NativeIO$Windows.access0(Ljava/lang/String;I)Z
1.window操作系统的eclipse运行wordcount程序出现如下所示的错误: Exception in thread "main" java.lang.Unsatisfi ...
- C# 会话,进程,线程,线程安全
会话->进程->线程 b/s网站中,每个用户的访问为一次会话,会话中包含CPU为用户在内存中开辟空间存储的会话信息, 如Session,进程,会话拥有一个进程,同一进程下可以拥有多个线程. ...
- java传值和传引用区别
1. 在java中所有的参数都是传值的,引用符号&的传递是C++中才有的:2. 在java传参中,基本类型(byte--short--int--long--float--double--boo ...
- la 3938(未完成)
题意:给出一个长度为n的整数序列D,你的任务是对m个询问作出回答.对于询问(a,b), 需要找到两个下标x和y,使得a≤x≤y≤b,并且Dx+Dx+1+...+Dy尽量大. 如果有多组满足条件的x和y ...
- file标签 - 图片上传前预览 - FileReader & 网络图片转base64和文件流
记得以前做网站时,曾经需要实现一个图片上传到服务器前,先预览的功能.当时用html的<input type="file"/>标签一直实现不了,最后舍弃了这个标签,使用了 ...
- mysql配置为半同步复制
mysql 半同步插件是由谷歌提供,具体位置/usr/local/mysql/lib/plugin/下,一个是 master用的 semisync_master.so,一个是 slave 用的 sem ...
- Python题目练习(一)
1.使用while循环输入 1 2 3 4 5 6 8 9 10 i = 1 while i <=10 : if i != 7: print(i) else: print(' ') i += ...
- Codeforces 844F Anti-Palindromize 最小费用流
Anti-Palindromize 想到网络流就差不多了, 拆拆点, 建建边. #include<bits/stdc++.h> #define LL long long #define f ...