类 Class 

每个类被加载之后,系统就会为该类生成一个对应的Class对象,通过该Class对象就可以访问到JVM中的这个类。
我们在Java中获取Class对象一般有三种方式:
(1), 使用Class类的forName(String className)静态方法。该方法需要传入字符串参数,该字符串参数的值是某个类的全限定类名(必须添加完整包名)。
(2), 调用某个类的class属性来获取该类对应的Class对象 例如,Person.class 将会返回Person类对应的class对象
(3), 调用某个对象的getClass()方法。该方法是 java.lang.Object 类中的一个方法,所以所有Java对象都可以调用该方法,该方法将会返回该对象所属类对应的 Class对象
public class SimpleClassTest {
public static void main(String[] args) throws ClassNotFoundException {
Class testcls1 = Class.forName("com.vgxit.testclass.Test");
System.out.println(testcls1); Class testcls2 = Test.class;
System.out.println(testcls2); Test test = new Test();
System.out.println(test.getClass());
}
}

Class对象中的信息 (反射)

1,反射构造方法:Connstructor

  • Connstructor<T> getConstructor(Class... paraeterTypes): 返回此Class象对应类的、带指定形参列表的public构造器
  • Constructor<T> getDeclaredConstructor(Class... parameterTypes): 同上 ,可返回private类型的构造器

  • Constructor>[] getConstructors(): 返回此Class对象对应类的所public构造器
  • Constructor>[] getDeclaredConstructors(): 返回此 Class 对象对应类的所有构造器(含private的)

测试用User类

package com.zmd.classTest;
public class User {
private String name;
private int age; public User(String name, int age) {
this.name = name;
this.age = age;
} @Override
public String toString() {
return "User{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
}

测试使用getConstructor:

public class ConstructorTest {
public static void main(String[] args) throws IllegalAccessException, InvocationTargetException, InstantiationException, NoSuchMethodException {
//获取对应的Class
Class<User> userClass = User.class;
//获取对应参数的构造器
Constructor<User> userConstructor = userClass.getConstructor(String.class, int.class);
//使用构造器.newInstance创建对象
User user = userConstructor.newInstance("V哥",18);
System.out.println(user);
}
}

上面的User对象创建成功,它的这个构造方法是public的。果把这个构造方法修改成了private的,就需要使用getDeclaredConstructor获取 所有构造器(含私有的)

测试使用getDeclaredConstructor:

使用private的构造器,需要设置为可访问的setAccessible(true)

public class ConstructorTest {
public static void main(String[] args) throws IllegalAccessException, InvocationTargetException, InstantiationException, NoSuchMethodException {
//获取对应的Class
Class<User> userClass = User.class;
//获取对应参数的构造器
Constructor<User> userConstructor = userClass.getDeclaredConstructor(String.class, int.class);
//禁止检查java的访问控制
userConstructor.setAccessible(true);
//使用构造器.newInstance创建对象
User user = userConstructor.newInstance("V哥",18);
System.out.println(user);
}
}
获取所有的构造方法的代码:
        Constructor<?>[] constructors = userClass.getDeclaredConstructors();
for (Constructor<?> constructor:constructors){
System.out.println(constructor);
}

2、反射方法Method :

  • Method getMethod(String name, Class... parameterTypes): 返回此 Class对象对应的带指定形参列表的public修饰的方法
  • Method[] getMethods(): 返回此Class对象所表示的类的所有public方法。
  • Method getDeclaredMethod(String name, Class... parameterTypes): 返回此Class对象对应类的、带指定形参列表的所有方法(含private的)
  • Method[] getDeclaredMethods(): 返回此Class象对应类的全部方法,

示例User类

package com.zmd.classTest;

import javax.sound.midi.Soundbank;

/**
* @ClassName User
* @projectName: object1
* @author: Zhangmingda
* @description: XXX
* date: 2021/5/16.
*/
public class User {
private static String name;
private int age; private User(String name, int age) {
this.name = name;
this.age = age;
} @Override
public String toString() {
return "User{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
private void sayHello(String toName){
System.out.println(this.name + "说:Hello " + toName);
}
public static void run(){
System.out.println(name + "在疯跑...");
}
}

执行反射得到的方法invoke

获取到Method方法实例.invoke(实例对象,方法args...) 执行对象的对应方法

package com.zmd.classTest;

import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method; /**
* @ClassName ConstructorMethod
* @projectName: object1
* @author: Zhangmingda
* @description: XXX
* date: 2021/5/16.
*/
public class ConstructorMethod {
public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException { Constructor<User> userConstructor = User.class.getDeclaredConstructor(String.class, int.class);
userConstructor.setAccessible(true);
User vGe = userConstructor.newInstance("V哥",22);
// Method sayHelloMethod = User.class.getMethod("sayHello", String.class);//sayHello 为public
Method sayHelloMethod = User.class.getDeclaredMethod("sayHello", String.class);//sayHello 为private
sayHelloMethod.setAccessible(true); //sayHello 为private
sayHelloMethod.invoke(vGe,"美女"); //V哥说:Hello 美女 }
}

反射静态方法,执行的时候对象传入null

package com.zmd.classTest;

import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method; /**
* @ClassName ConstructorStaticMethod
* @projectName: object1
* @author: Zhangmingda
* @description: XXX
* date: 2021/5/16.
*/
public class ConstructorStaticMethod {
public static void main(String[] args) throws NoSuchMethodException, InvocationTargetException, IllegalAccessException, InstantiationException {
Method runMethod = User.class.getDeclaredMethod("run");
runMethod.invoke(null); //null在疯跑... Constructor<User> constructor = User.class.getDeclaredConstructor(String.class,int.class);
constructor.setAccessible(true);
constructor.newInstance("v哥",22);
runMethod.invoke(null); //v哥在疯跑... 所有类共享static 静态变量,创建一个类后User类的name类变量变为了v哥
}
}

3、获取对应的成员变量:Field

  • Field getField(String name): 返回此Classd对象对应类的、指定名称 public成员变量。
  • Field[] getFields(): 返回此Class对象对应类的所有public成员变量。
  • Field getDeclaredField(String name): 返回此Class对象对应类的、指定名称的成员变,与成员变的访问权限无关
  • Field[] getDeclaredFields(): 返回此Class对象对应类的全部成员变量,与成员变量的访问权限无关。
package com.zmd.classTest;

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException; /**
* @ClassName ConstructorVar
* @projectName: object1
* @author: Zhangmingda
* @description: XXX
* date: 2021/5/16.
*/
public class ConstructorVar {
public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException, NoSuchFieldException {
Constructor<User> constructor = User.class.getDeclaredConstructor(String.class, int.class);
constructor.setAccessible(true);
User vGe = constructor.newInstance("V哥",22); Field nameField = User.class.getDeclaredField("name");
Field ageField = User.class.getDeclaredField("age"); nameField.setAccessible(true);
ageField.setAccessible(true); System.out.println(nameField.get(vGe));
System.out.println(ageField.get(vGe));
}
}

获取静态变量,实例传入null即可

public class User {
private String name;
private int age;
private static final String type = "中国人";
}
package com.zmd.classTest;

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException; /**
* @ClassName ConstructorVar
* @projectName: object1
* @author: Zhangmingda
* @description: XXX
* date: 2021/5/16.
*/
public class ConstructorVar {
public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException, NoSuchFieldException { Field typeField = User.class.getDeclaredField("type");
typeField.setAccessible(true);
System.out.println(typeField.get(null)); //输出:中国人 //静态变量获取,对象为null即可
}
}

改变实例变量:

        //改变实例变量
System.out.println(vGe); //User{name='V哥', age=22}
ageField.set(vGe,18);
System.out.println(vGe); //User{name='V哥', age=18}

4,获取内部类:

Class[] getDeclaredClasses(): 返回该Class对象对应类里包含的全部内部类。
public class User {
public static class Student{
private int grade;
public void showGrade(){
System.out.println("年级是");
}
}
}
package com.zmd.classTest;

import java.lang.reflect.InvocationTargetException;

/**
* @ClassName GetInnerClass
* @projectName: object1
* @author: Zhangmingda
* @description: XXX
* date: 2021/5/16.
*/
public class GetInnerClass {
public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
//获取所有内部类
Class<?>[] innerClasses = User.class.getDeclaredClasses();
for (Class<?> cls :innerClasses){
if (cls.getSimpleName().equals("Student")){
User.Student student = (User.Student) cls.getConstructor().newInstance();
student.showGrade(); //调用内部类对象的方法
cls.getMethod("showGrade").invoke(student); //反射方式调用内部类的方法
}
}
}
}

5,通过内部类Class获取外部类:

        //通过内部类获取外部类
Class<User.Student> studentClass = User.Student.class;
Class<User> userClass = (Class<User>)studentClass.getDeclaringClass();
System.out.println(userClass); //class com.zmd.classTest.User

6,获取一个类实现了那些接口

Class[] getInterfaces(): 返回该 Class 对象对应类所实现的全部接口

        //通过内查看实现了哪些接口
Class<?>[] interfaces = User.class.getInterfaces();
for (Class<?> interfaceClass : interfaces) {
System.out.println(interfaceClass);
}

7,获取一个类对应的父类的Class

Class getSuperclass():返回该Class对象对应类的超类的Class对象
        //获取父类的Class
Class<?> parentClass = studentClass.getSuperclass();
System.out.println(parentClass); //class java.lang.Object

8,获取对应类的修饰符、所在包、类名等基本信息。

  • (1), int getModifiers(): 返回此类或接口的所有修饰符。修饰符由 public protected private final static abstract 等对应的常量组成。返回的整数应使用Modifier工具类的方法来解码,才可以获取真实的修饰符
  • (2),Package getPackage(): 获取此类的包。
  • (3),String getName(): 以字符串形式返回此 Class 对象所表示的类的名称。
  • (4),String getSimpleName(): 以字符串形式返回此 Class 对象所表示的类的简称
获取修饰符:

        Method runMethod = User.class.getDeclaredMethod("run");
//获取方法修饰符代号
int modifies = runMethod.getModifiers();
System.out.println(Modifier.isPublic(modifies)); //是否是public的 //true
System.out.println(Modifier.isStatic(modifies)); //是否是static静态的//true
System.out.println(Modifier.isAbstract(modifies));//是否是抽象的//false
System.out.println(Modifier.isFinal(modifies)); //是否是final的//false

9,判断该类是否为接口、枚举、注解类型等。

  • (1), boolean isAnnotation(): 返回此Class对象是否表示一个注解类型。
  • (2), boolean isAnnotationPresent(Class annotationC ass) 判断此Class对象是否使用了某个Annotation修饰。
  • (3), boolean isAnonymousClass(): 返回此Class对象是否是一个匿名类
  • (4), boolean isArray(): 返回此Class对象是否表示一个数组类。
  • (5), boolean isEnum(): 返回此Class对象是否表示一个枚举(由enum关键宇定义)。
  • (6), boolean isInterface(): 返回此Class对象是否表示一个接口〈使用interface定义)。
  • (7), boolean isInstance(Object obj): 判断obj是否是此Class对象的实例,该方法可以完全代替instanceof操作符。

java 编程基础 Class对象 反射 :获取类的构造方法,方法,成员变量,内部类,外部类,父类,实现的接口,修饰符等...的更多相关文章

  1. java 编程基础 Class对象 反射 :参数反射

    方法参数反射 Java8在java.lang.reflect包下新增了Executable抽象基类,该对象代表可执行的类成员,该类派生了Constructor和Method两个子类.Executabl ...

  2. java 编程基础 Class对象 反射:动态代理 和AOP:java.lang.reflect.Proxy:(Proxy.newProxyInstance(newProxyInstance​(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h))

    为什么我们使用动态代理 静态代理会让类变多了,多了代理类,工作量变大了,且不易扩展.比如我们上节课的例子,要实现不同的扩展方法就要编写不同的代理类,非常麻烦.   Proxy类的使用规则 Proxy提 ...

  3. java 编程基础 Class对象 反射 :数组操作java.lang.reflect.Array类

    java.lang.reflect包下还提供了Array类 java.lang.reflect包下还提供了Array类,Array对象可以代表所有的数组.程序可以通过使 Array 来动态地创建数组, ...

  4. java 编程基础 Class对象 反射:代理模式和静态代理

    生活中的代理 类(对象)代理模式 代理模式是面向对象编程中比较常见的设计模式. 1. 用户只关心接口功能,而不在乎谁提供了功能.上图中接口是 Subject 2. 接口真正实现者是上图的 RealSu ...

  5. java中的反射机制,以及如何通过反射获取一个类的构造方法 ,成员变量,方法,详细。。

    首先先说一下类的加载,流程.只有明确了类这个对象的存在才可以更好的理解反射的原因,以及反射的机制. 一.  类的加载 当程序要使用某个类时,如果该类还未被加载到内存中,则系统会通过加载,连接,初始化三 ...

  6. C#通过反射获取类中的方法和参数个数,反射调用方法带参数

    using System; using System.Reflection; namespace ConsoleApp2 { class Program { static void Main(stri ...

  7. Java反射学习-1 - 反射获取类的属性,方法,构造器

    新建一个Person类 package cn.tx.reflect; /** * 注解初步了解 * @author Administrator * */ public class Person { p ...

  8. Java反射获取类和对象信息全解析

    反射可以解决在编译时无法预知对象和类是属于那个类的,要根据程序运行时的信息才能知道该对象和类的信息的问题. 在两个人协作开发时,你只要知道对方的类名就可以进行初步的开发了. 获取类对象 Class.f ...

  9. java 编程基础 反射方式获取泛型的类型Fileld.getGenericType() 或Method.getGenericParameterTypes(); (ParameterizedType) ;getActualTypeArguments()

    引言 自从JDK5以后,Java Class类增加了泛型功能,从而允许使用泛型来限制Class类,例如,String.class的类型实际上是 Class 如果 Class 对应的类暂时未知,则使 C ...

随机推荐

  1. ICCV2021 | Swin Transformer: 使用移位窗口的分层视觉Transformer

    ​  前言  本文解读的论文是ICCV2021中的最佳论文,在短短几个月内,google scholar上有388引用次数,github上有6.1k star. 本文来自公众号CV技术指南的论文分享系 ...

  2. 深入理解Redis 数据结构—简单动态字符串sds

    Redis是用ANSI C语言编写的,它是一个高性能的key-value数据库,它可以作用在数据库.缓存和消息中间件.其中 Redis 键值对中的键都是 string 类型,而键值对中的值也是有 st ...

  3. 华为云企业级Redis评测第一期:稳定性与扩容表现

    摘要:采用Redis Labs推出的多线程压测工具memtier_benchmark对比测试下GaussDB(for Redis) 和原生Redis的特性差异. 本文分享自华为云社区<华为云企业 ...

  4. Linux 使用wpa_supplicant手动配置连接wifi

    Linux 使用wpa_supplicant手动配置连接wifi wpa_supplicant 简介 wpa_supplicant是Linux BSD, Mac OSX和Windows的WPA的服务, ...

  5. FFT/NTT复习笔记&多项式&生成函数学习笔记Ⅱ

    因为垃圾电脑太卡了就重开了一个... 前传:多项式Ⅰ u1s1 我预感还会有Ⅲ 多项式基础操作: 例题: 26. CF438E The Child and Binary Tree 感觉这题作为第一题还 ...

  6. 【2020五校联考NOIP #3】序列

    题面传送门 原题题号:Codeforces Gym 101821B 题意: 给出一个排列 \(p\),要你找出一个最长上升子序列(LIS)和一个最长下降子序列(LDS),满足它们没有公共元素.或告知无 ...

  7. P5509 派遣

    题面传送门. 数论小杂烩( 由题意,对于每个士兵 \(i\),要么选,对答案产生 \(a_i(\frac{x}{i-x})\) 倍的贡献,要么不选,对答案产生 \(1\) 倍的贡献. 由此可知每个士兵 ...

  8. GWAS初探

    原理 GWAS 的主要方法学依据是归纳法中的共变法,是探究复杂因果关系最主要的科学思维和方法.所谓共变法,是通过考察被研究现象发生变化的若干场合中,确定是否只有一个情况发生相应变化,如果是,那么这个发 ...

  9. 【R】表达矩阵指定绘制两样本的相关性散点图?

    表达矩阵 要做两两样本的相关性散点图,并计算标明相关系数. 编写函数要点: 直接在aes中传参是不行的 线性回归表达式 函数 方法1:用!!ensym myplot <- function(in ...

  10. 深入浅出KMP

    前言:曾经有次在阿里的面试中遇到这个基础的问题,当时知道有这么回事,可是时间久了便 想不起来,可能是不怎么用到,基本调用库什么的,还有个是理解不深刻,不能得到show me the code 的程度, ...