Java反射API研究(3)——java.lang.Class<T>

mac2022-06-30  24

  对于反射来说,Class是核心,任何反射的对象都需要通过Class来获得。

  Class 类的实例表示正在运行的 Java 应用程序中的类和接口。枚举是一种类,注释是一种接口。每个数组属于被映射为 Class 对象的一个类,所有具有相同元素类型和维数的数组都共享该 Class 对象。基本的 Java 类型(boolean、byte、char、short、int、long、float 和 double)和关键字 void 也表示为 Class 对象。 

  Class 没有公共构造方法。Class 对象是在加载类时由 Java 虚拟机以及通过调用类加载器中的 defineClass 方法自动构造的。 

  Class<T>的T - 由此 Class 对象建模的类的类型。例如,String.class 的类型是 Class<String>。如果将被建模的类未知,则使用 Class<?>。

  下面来看看如何获得一个类的Class对象。

  有以下几种方式:

  1、调用一个实例的getClass()方法,这个方法是从Object继承下来的,任何一个对象的实例都可以通过getClass()方法来获取当前实例的类的Class对象,如果声明时是父类,则获取的是真实实例的Class,即子类的Class。

  2、通过一个Class对象的Class.getSuperclass(),获取其父类的Class对象

  3、Class的静态方法forName(String class),这个最常用,通过类名包括包名获取一个类的class对象,还有个重载方法forName(String name, boolean initialize, ClassLoader loader),第二个参数表示是否初始化该类,即是否执行该类的静态代码段,forName(String class)调用的这个方法,默认为true,初始化该类的静态方法。这在注册jdbc驱动时有用到。

  4、通过一个类的静态成员获取class对象,类名.class即可。这个成员对象是jvm加上去的。

  5、primitive wrapper classes的TYPE 语法,包装类可以通过包装类的TYPE静态变量来获取。这里要注意一下:

    int.class == Integer.TYPE,但是int.class != Integer.class。经过包装后,包装类和原始类的class不是一个对象。

 

  Class对象的所有方法:

<U> Class<? extends U>asSubclass(Class<U> clazz)强制转换该 Class 对象,以表示指定的 class 对象所表示的类的一个子类。Tcast(Object obj)将一个对象强制转换成此 Class 对象所表示的类或接口。booleandesiredAssertionStatus()如果要在调用此方法时将要初始化该类,则返回将分配给该类的断言状态。static Class<?>forName(String className)返回与带有给定字符串名的类或接口相关联的 Class 对象。static Class<?>forName(String name, boolean initialize, ClassLoader loader)使用给定的类加载器,返回与带有给定字符串名的类或接口相关联的 Class 对象。AnnotatedType[]getAnnotatedInterfaces()返回注释的AnnotatedTypeAnnotatedTypegetAnnotatedSuperclass()返回父类的注解的AnnotatedType<A extends Annotation> AgetAnnotation(Class<A> annotationClass)如果存在该元素的指定类型的注释,则返回这些注释,否则返回 null。Annotation[]getAnnotations()返回此元素上存在的所有注释。<A extends Annotation> A[]getAnnotationsByType(Class<A> annotationClass)返回一个该类型的注解数组StringgetCanonicalName()返回 Java Language Specification 中所定义的底层类的规范化名称。Class<?>[]getClasses()返回类定义的公共的内部类,以及从父类、父接口那里继承来的内部类ClassLoadergetClassLoader()返回该类的类加载器。Class<?>getComponentType()返回表示数组组件类型的 Class。如果此类表示数组类,返回表示此类组件类型的 Class。否则返回null。 Constructor<T>getConstructor(Class<?>... parameterTypes)返回一个 Constructor 对象,它反映此 Class 对象所表示的类的指定公共构造方法。Constructor<?>[]getConstructors()返回一个包含某些 Constructor 对象的数组,这些对象反映此 Class 对象所表示的类的所有公共构造方法。<A extends Annotation> AgetDeclaredAnnotation(Class<A> annotationClass)返回直接存在于此元素上的该类型注释。Annotation[]getDeclaredAnnotations()返回直接存在于此元素上的所有注释。<A extends Annotation> A[]getDeclaredAnnotationsByType(Class<A> annotationClass)返回直接存在于此元素上的该类型的注解数组。Class<?>[]getDeclaredClasses()返回类定义的公共的内部类,不包括继承的。Constructor<T>getDeclaredConstructor(Class<?>... parameterTypes)根据指定参数类型顺序返回一个 Constructor 对象,该对象反映此 Class 对象所表示的类或接口的指定构造方法。可获取非公共构造方法。Constructor<?>[]getDeclaredConstructors()返回 Constructor 对象的一个数组,这些对象反映此 Class 对象表示的类声明的所有构造方法。不仅是公共的。FieldgetDeclaredField(String name)返回一个 Field 对象,该对象反映此 Class 对象所表示的类或接口的指定已声明字段。Field[]getDeclaredFields()返回 Field 对象的一个数组,这些对象反映此 Class 对象所表示的类或接口所声明的所有字段。MethodgetDeclaredMethod(String name, Class<?>... parameterTypes)返回一个 Method 对象,该对象反映此 Class 对象所表示的类或接口的指定已声明方法。Method[]getDeclaredMethods()返回 Method 对象的一个数组,这些对象反映此 Class 对象表示的类或接口声明的所有方法,包括公共、保护、默认(包)访问和私有方法,但不包括继承的方法。Class<?>getDeclaringClass()如果此 Class 对象所表示的类或接口是另一个类的成员,则返回的 Class 对象表示该对象的声明类。即返回所在类的class对象,对成员class对象有效。Class<?>getEnclosingClass()返回底层类的立即封闭类。Constructor<?>getEnclosingConstructor()如果该 Class 对象表示构造方法中的一个本地或匿名类,则返回 Constructor 对象,它表示底层类的立即封闭构造方法。MethodgetEnclosingMethod()如果此 Class 对象表示某一方法中的一个本地或匿名类,则返回 Method 对象,它表示底层类的立即封闭方法。T[]getEnumConstants()如果此 Class 对象不表示枚举类型,则返回枚举类的元素或 null。FieldgetField(String name)返回一个 Field 对象,它反映此 Class 对象所表示的类或接口的指定公共成员字段。Field[]getFields()返回一个包含某些 Field 对象的数组,这些对象反映此 Class 对象所表示的类或接口的所有可访问公共字段。Type[]getGenericInterfaces()返回表示某些接口的 Type,这些接口由此对象所表示的类或接口直接实现。TypegetGenericSuperclass()返回表示此 Class 所表示的实体(类、接口、基本类型或 void)的直接超类的 Type。Class<?>[]getInterfaces()确定此对象所表示的类或接口实现的接口。MethodgetMethod(String name, Class<?>... parameterTypes)返回一个 Method 对象,它反映此 Class 对象所表示的类或接口的指定公共成员方法。Method[]getMethods()返回一个包含某些 Method 对象的数组,这些对象反映此 Class 对象所表示的类或接口(包括那些由该类或接口声明的以及从超类和超接口继承的那些的类或接口)的公共 member 方法。intgetModifiers()返回此类或接口以整数编码的 Java 语言修饰符。StringgetName()以 String 的形式返回此 Class 对象所表示的实体(类、接口、数组类、基本类型或 void)名称。PackagegetPackage()获取此类的包。ProtectionDomaingetProtectionDomain()返回该类的 ProtectionDomain。URLgetResource(String name)查找带有给定名称的资源。InputStreamgetResourceAsStream(String name)查找具有给定名称的资源。Object[]getSigners()获取此类的标记。StringgetSimpleName()返回源代码中给出的底层类的简称。Class<? super T>getSuperclass()返回表示此 Class 所表示的实体(类、接口、基本类型或 void)的超类的 Class。StringgetTypeName()返回该类型的名称TypeVariable<Class<T>>[]getTypeParameters()按声明顺序返回 TypeVariable 对象的一个数组,这些对象表示用此 GenericDeclaration 对象所表示的常规声明来声明的类型变量。booleanisAnnotation()如果此 Class 对象表示一个注释类型则返回 true。booleanisAnnotationPresent(Class<? extends Annotation> annotationClass)如果指定类型的注释存在于此元素上,则返回 true,否则返回 false。booleanisAnonymousClass()当且仅当底层类是匿名类时返回 true。booleanisArray()判定此 Class 对象是否表示一个数组类。booleanisAssignableFrom(Class<?> cls)判定此 Class 对象所表示的类或接口与指定的 Class 参数所表示的类或接口是否相同,或是否是其超类或超接口。booleanisEnum()当且仅当该类声明为源代码中的枚举时返回 true。booleanisInstance(Object obj)判定指定的 Object 是否与此 Class 所表示的对象赋值兼容。booleanisInterface()判定指定的 Class 对象是否表示一个接口类型。booleanisLocalClass()当且仅当底层类是本地类时返回 true。booleanisMemberClass()当且仅当底层类是成员类时返回 true。booleanisPrimitive()判定指定的 Class 对象是否表示一个基本类型。booleanisSynthetic()如果此类是复合类,则返回 true,否则 false。有匿名类部类的类可以称作为复合类。TnewInstance()调用默认构造方法创建此 Class 对象所表示的类的一个新实例。StringtoGenericString()返回该对象的描述,包含标识符等。

  比较重要的方法:newInstance(),创建一个该类的实例,调用默认构造方法。

  特殊的:

  1、获取一个类声明时的确切泛型类型:

class Parameterized<T,V,Z>{ } Type classType = abstractTransfer.getClass().getGenericSuperclass(); Type[] ret = ((ParameterizedType)classType).getActualTypeArguments(); T = (Class) ret[0]; V = (Class) ret[1]; Z = (Class) ret[2];

  2、运行时获取一个被代理的类的真实类对象

TrueClass proxyClass = proxy; Class trueClass = proxyClass.getClass().getSuperclass();

 

二、ClassLoader对象

  参考http://blog.chinaunix.net/uid-21227800-id-65879.html

  类加载器是负责加载类的对象。ClassLoader 类是一个抽象类。如果给定类的二进制名称,那么类加载器会试图查找或生成构成类定义的数据。一般策略是将名称转换为某个文件名,然后从文件系统读取该名称的“类文件”。 

  每个 Class 对象都包含一个对定义它的 ClassLoader 的引用。 

  数组类的 Class 对象不是由类加载器创建的,而是由 Java 运行时根据需要自动创建。数组类的类加载器由 Class.getClassLoader() 返回,该加载器与其元素类型的类加载器是相同的;如果该元素类型是基本类型,则该数组类没有类加载器。

  获取ClassLoader对象的几种方法:

  1、通过Thread对象的getContextClassLoader()获取类加载器。一般为Thread.currentThread().getContextClassLoader()。  当前类的ClassLoader  

  2、通过Class对象的getClassLoader()方法获取类加载器。一般为this.getClass().getClassLoader()。  当前线程的ClassLoader  

  3、通过ClassLoader类的静态方法ClassLoader.getSystemClassLoader()方法。  系统ClassLoader,即系统的入口点所使用的ClassLoader。(注意,system ClassLoader与根ClassLoader并不一样。JVM下system ClassLoader通常为App ClassLoader)

  这三个是否引用的同一对象呢?一般情况下是的,因为都返回True,其他高级应用中可能会不一样。

  这个对象中还有很多操作资源的方法比较有用,自行查阅。

 

三、Package对象

  Package 对象包含有关 Java 包的实现和规范的版本信息。通过用于加载类的 ClassLoader 实例,可以获取并获得此版本信息。通常,此信息存储在与类一起分发的清单中。 

  可以读Package的注解。

 

四、扫描一个包中的所有类

转载于:https://www.cnblogs.com/guangshan/p/4891715.html

最新回复(0)