1. class 类的使用

  • 万事万物皆对象 (基本数据类型, 静态成员不是面向对象), 所以我们创建的每一个类都是对象, 即类本身是java.lang.Class类的实例对象, 但是这些对象不需要 new 出来, 因为java.lang.Class类的构造方法是私有的;
  • 任何一个类都是Class类的实例对象.这个实例对象有_三种表达方式_: (我们新建一个Student类)
Class c1 = Student.class; // 实际告诉我们任何一个类都有一个隐含的静态成员变量class(知道类名时用)
Student stu = new Student();
Class c2 = stu.getClass(); // 已知该类的对象通过getClass方法(知道对象时用)
try {
Class c3 = Class.forName("com.cnblogs.reflect.Student");// 会有一个ClassNotFoundException异常
} catch (ClassNotFoundException e) {
e.printStackTrace();
}

ps: 官网解释说, c1, c2表示 Student类的类类型(class type), 类也是对象, 是Class类的实例对象, 这个对象我们称之为该类的_类类型_

这里有一点值得注意的是: 当我们执行System.out.println(c1 == c2) 时, 结果是true, 这是为什么? 原因是不管c1, c2都代表了Student类的类类型, 一个类可能是Class类的一个实例对象.

我们完全可以通过类的类类型创建该类的对象实例, 即通过c1或者c2创建Student的实例;

Student stu = (Student) c1.newInstance(); // 前提是Student里必须要有无参构造方法, 否则会报异常

2. 动态加载类

  • 编译时加载类是静态加载类

    new 创建对象是静态加载类, 在编译时刻就需要加载所有可能使用到的类, 如果有一个缺失, 那么整个文件都无法通过编译;
  • 运行时加载类是动态加载类

3. 获取方法信息

  • 基本的数据类型, void关键字都存在类类型
Class c1 = int.class; // int的类类型
Class c2 = String.class; // String的类类型, 可以理解为编译生成生成的那个String.class字节码文件
Class c3 = double.class;
Class c4 = Double.class;
Class c5 = void.class;
  • Class类的基本API操作
public static void printClassMessage(Object obj) {
Class c = obj.getClass();
// 获取类的类名称;
System.out.println("类的名称是: " + c.getName());
System.out.println("---------------------------------"); Method[] methods = c.getMethods();
for (Method method : methods) {
// 得到方法返回值类型的类类型
Class returnType = method.getReturnType();
System.out.println("returnType = " + returnType.getName());
// 得到方法的名称
System.out.println("method = " + method.getName());
// 获取参数类型 -> 得到的是参数列表的类类型
Class<?>[] parameterTypes = method.getParameterTypes();
for (Class parameterClass : parameterTypes) {
System.out.println("parameterClass = " + parameterClass.getName());
}
System.out.println("---------------------------------");
}
}

4. 获取成员变量构造函数信息

public static void printFieldMessgae(Object obj) {

        Class<?> c = obj.getClass();
Field[] fields = c.getDeclaredFields();
for (Field field : fields) {
// 得到成员变量的类型的类类型
Class<?> fieldType = field.getType();
String typeName = fieldType.getName();
System.out.println("typeName = " + typeName);
// 得到成员变量的名称
String fieldName = field.getName();
System.out.println("fieldName = " + fieldName);
}
/**
* 获取构造函数
*/
Constructor<?>[] constructors = c.getDeclaredConstructors();
for (Constructor<?> constructor : constructors) {
System.out.println("constructor = " + constructor.getName());
/**
* 获取构造函数的参数的类类型
*/
Class<?>[] parameterTypes = constructor.getParameterTypes();
for (Class<?> parameterType : parameterTypes) {
System.out.println("parameterType = " + parameterType.getName());
}
}
}

5. 方法反射的基本操作

  • 如何获取某个方法

    方法的名称和方法的参数列表才能决定某个方法

    Method m = c.getDeclaredMethod("方法名", 可变参数列表(参数类型.class))
  • 方法的反射操作

    m.invoke(对象, 参数列表)

    方法如果没有返回值, 返回null, 如果有返回值, 返回Object,

    然后再强转为原函数的返回值类型;

6. 通过反射了解集合泛型的本质

ArrayList list1 = new ArrayList();
ArrayList<String> list2 = new ArrayList(); Class c1 = list1.getClass();
Class c2 = list2.getClass(); System.out.println(c1 == c2); // -> true

Ps: 因为反射的操作都是编译之后的操作, 也就是运行时的操作, c1 == c2返回true, 说明编译之后集合的泛型是去泛型化的.

那么我们可以理解为, Java集合中的泛型, 是用于泛指错误类型元素输入的, 比如在list2中我们add一个int, add(10)就会编译保错, 那么这个泛型就可以只在编译阶段有效, 通过了编译阶段, 泛型就不存在了. 可以验证, 我们绕过编译, 用反射动态的在list2中add一个int是可以成功的. 只是这时因为list2中存储了多个不同类型的数据(String和int), 就不能用for-each来遍历了, 会抛出类型转换错误.

Java reflect 反射学习笔记的更多相关文章

  1. Java虚拟机内存溢出异常--《深入理解Java虚拟机》学习笔记及个人理解(三)

    Java虚拟机内存溢出异常--<深入理解Java虚拟机>学习笔记及个人理解(三) 书上P39 1. 堆内存溢出 不断地创建对象, 而且保证创建的这些对象不会被回收即可(让GC Root可达 ...

  2. 《深入理解Java虚拟机》学习笔记

    <深入理解Java虚拟机>学习笔记 一.走近Java JDK(Java Development Kit):包含Java程序设计语言,Java虚拟机,JavaAPI,是用于支持 Java 程 ...

  3. JAVA GUI编程学习笔记目录

    2014年暑假JAVA GUI编程学习笔记目录 1.JAVA之GUI编程概述 2.JAVA之GUI编程布局 3.JAVA之GUI编程Frame窗口 4.JAVA之GUI编程事件监听机制 5.JAVA之 ...

  4. Java多线程技术学习笔记(二)

    目录: 线程间的通信示例 等待唤醒机制 等待唤醒机制的优化 线程间通信经典问题:多生产者多消费者问题 多生产多消费问题的解决 JDK1.5之后的新加锁方式 多生产多消费问题的新解决办法 sleep和w ...

  5. Java安全防御学习笔记V1.0

    Java安全防御学习笔记V1.0http://www.docin.com/p-766808938.html

  6. java之jvm学习笔记六-十二(实践写自己的安全管理器)(jar包的代码认证和签名) (实践对jar包的代码签名) (策略文件)(策略和保护域) (访问控制器) (访问控制器的栈校验机制) (jvm基本结构)

    java之jvm学习笔记六(实践写自己的安全管理器) 安全管理器SecurityManager里设计的内容实在是非常的庞大,它的核心方法就是checkPerssiom这个方法里又调用 AccessCo ...

  7. java之jvm学习笔记三(Class文件检验器)

    java之jvm学习笔记三(Class文件检验器) 前面的学习我们知道了class文件被类装载器所装载,但是在装载class文件之前或之后,class文件实际上还需要被校验,这就是今天的学习主题,cl ...

  8. java之jvm学习笔记五(实践写自己的类装载器)

    java之jvm学习笔记五(实践写自己的类装载器) 课程源码:http://download.csdn.net/detail/yfqnihao/4866501 前面第三和第四节我们一直在强调一句话,类 ...

  9. java之jvm学习笔记四(安全管理器)

    java之jvm学习笔记四(安全管理器) 前面已经简述了java的安全模型的两个组成部分(类装载器,class文件校验器),接下来学习的是java安全模型的另外一个重要组成部分安全管理器. 安全管理器 ...

随机推荐

  1. cyclone IV中DDR的一个报错{Too many output and bidirectional pins per VCCIO and ground pair in I/O bank 8 }

    Error (169224): Too many output and bidirectional pins per VCCIO and ground pair in I/O bank 8 when ...

  2. AngularJS监听数组变化

    我们在使用angualr的监听时候,业务的需要我们会去监听一个数组的某一个值得变化,再写逻辑代码.然而我们在使用$scope.$watch("",function(){ })时候会 ...

  3. C#-VS发布网站-摘

    在vs生成发布文件 现在已经有了网站,可以发布了.可以将网站发布到您可以使用 Visual Studio 支持的任何连接协议访问的任何位置.复制网站有下面几种方式可选: 复制到本地计算机上的文件夹. ...

  4. 《it项目管理那些事》学习笔记

    此书适合:计算及相关专业的学生,想成为测试工程师.软件工程师.进入项目经理的人,或者经验丰富的it经理人. 之所以称为学习笔记,是加上我从百度搜到一些在看书过程中不明白的it语,作为菜鸟的我,得多看看 ...

  5. Eclipse运行wordcount步骤

    Eclipse运行wordcount步骤 第一步:建立工程,导入代码. 第二步:建立文件写入数据(以空格分开),并上传到hdfs上. 1.创建文件并写入数据: 2.上传hdfs 在hadoop权限下就 ...

  6. Scala_模式匹配

    模式匹配 简单匹配 Scala的模式匹配最常用于match语句中.下面是一个简单的整型值的匹配实例 object TestMatch {  def main(args: Array[String]): ...

  7. shell 命令 grep -v

    grep -v shell命令中,grep命令,是对文本行的搜索命令.grep -v就是反向文本行搜索. 当控制台输出很多时,有很多是我们不想看到的,就可以用到grep -v命令 举个栗子:ls -l ...

  8. (25)uniGUI for C++ builder之UniHTMLMemo初使用及uniGUI如何调用javaScript

    (25)uniGUI for C++ builder之UniHTMLMemo初使用及uniGUI如何调用javaScript 2018年09月29日 22:58:20 中国银行之路在脚下 阅读数:11 ...

  9. (原创)用c++11打造好用的any

    上一篇博文用c++11实现了variant,有童鞋说何不把any也实现一把,我正有此意,它的兄弟variant已经实现了,any也顺便打包实现了吧.其实boost.any已经挺好了,就是转换异常时,看 ...

  10. oracle 中top-n的使用

    对于ms sqlserver数据库中可以直接使用top(n)提取前N 个结果,而oracle中并不能直接使用的.oracle中提供了对于提取前N 条的结果的方法  那就是用行编号 例如:select ...