要想在IT领域站得住脚,必须得不断地学习来强化自己,但是学过的技术不实践很容易便被遗忘,所以一直都打算开个博客,来记录自己学的知识,另外也可以分享给有需要的人!

  最近在学习反射,为了更好地理解反射,就去查各种资料学习了java类型信息。

  目录

  • 前言
  • java类的加载和初始化
  • Class对象
  • java类型信息

  一、前言

  在了解java类型信息前,需要先了解我们编写的类在Java中是如何加载的,以及Class类的基本概念和作用,以方便我们更好理解Java类型信息。

  二、类的加载和初始化

  2.1 类的加载

  当程序要使用某个类时,如果该类还未被加载到内存中,则系统会通过加载,连接,初始化三步来实现对这个类的初始化。

  • 加载

    • 就是指class文件读入内存,并为之创建一个Class对象(下文会详细介绍)
    • 任何类被使用时系统都会为之建立一个Class对象。
  • 连接
    • 验证 是否有正确的内部结构,并和其他类协调一致
    • 准备 负责为类的静态成员分配内存,并设置默认初始化值
    • 解析 对类中的接口、类、方法、变量的符号引用进行解析并定位,解析成直接引用(符号引用就是编码时用字符串表示某个变量、接口的位置,直接引用就是根据符号引用翻译出来的地址)
  • 初始化
    • 初始化Java代码和静态Java代码块,即在内存中建立java内存模型

  2.2 类的加载时机

  • 第一次创建类的实例
  • 访问类的静态变量,或者为静态变量赋值
  • 调用类的静态方法
  • 使用反射方法强制创建某个类或接口对应的java.lang.Class对象
  • 初始化某个类的子类
  • 直接使用java.exe命令来运行某个类

  2.3类加载器

  • Bootstrap ClassLoader 根类加载器
  • Extension ClassLoader 扩展类加载器
  • System ClassLoader 系统类加载器
  • Bootstrap ClassLoader 根类加载器

    •   也被称为引导类的加载器,负责Java核心类的加载

      •     比如System String等,在JDK中JRE的lib目录下rt.jar文件中
  • Extension ClassLoader扩展类加载器
    •   负责JRE的扩展目录中jar包的加载

      •     在JDK中JRE的lib目录下ext目录
  • System ClassLoader系统类加载器
    •   负责在JVM启动时加载来自java命令的class文件,以及classpath环境变量所指定的jar包和类型

  2.4 双亲委派模型

  类加载器Java类如同其他的Java类一样,也是要由类加载器来加载的。除了启动类根类加载器每个类都有其父类加载器(父子关系有组合(不是继承)来实现)。

  所谓双亲委派模型是指每次收到类加载请求时,先将请求委派给父类加载器完成(所有加载请求最终会委派到顶层的Bootstrap Class Loader加载器中),如果父类加载器无法完成这个加载(该加载器的搜索范围没有找到对应的类),子类尝试自己加载。

      双亲委派好处

      • 避免同一个类被加载多次
      • 每个加载 器只能加载自己范围内的类

三、Class类

  Class类封装一个对象和接口的运行时的状态(即在这个类的对象中记录每个对象所属的类,保存类的类型信息)。操作类对象在使用前会经历上文所说的加载过程(加载、裂解和初始化),并为之创建一个Class类的对象---Class对象,存放在操作类的.class文件中。这个Class类对象与操作类class是一一对应的,即当一个操作类被加载后,就不会重复加载。而通过该class对象,就可以访问对应的class。我们可以把Class理解为一个类的标识对象,它相当于是一个类的铭牌。拿到一个Class,我们就可以找到对应的类。

注意,Class类没有公有(public)构造方法,因此不能显式地声明一个Class对象,只在类加载时创建。

从Class对象中可以获得的信息:

    • 获取到class所包含的构造器
    • 获取到class所包含的方法
    • 获取到class所包含的成员变量
    • 获取到class所包含的Annotation

另外,网上很多帖子说,一旦某个操作类的Class对象被载入内存,就用它来产生该操作类的所有对象!

获取Class对象的三种方式:

    • 调用Object类的getClass()方法来得到Class对象,如:
 Dog dog;
Class class = dog.getClass();
    • 使用Class类中的静态方法forName(string)获得与参数string对象的类的Class对象。例如
 Class class = Class.forName("Dog");
    • 任何类都具备一个静态的属性(class属性),可引用这个属性获得对应的Class对象,如:
 Class class = Dog.class;

Class类常用的方法

    • getName()
    • o 返回由 类对象表示的实体(类,接口,数组类,原始类型或空白)的名称,作为 String 。
    • newInstance()
    • o 创建由此 类对象表示的类的新实例。
    • getClassLoader
    • o 返回该类的类加载器

四、Java类型信息

  我们知道,我们的程序在运行时,某一个操作类可以回被实例化多个对象,一个程序也可能拥有若干个不同操作类的多个对象,那么Java如何在运行时识别对象所属和类的信息的呢?答案是运行时类型信息(Run-Time Type Identification),RTTI使得你可以在程序运行时发现和使用类型信息。

一、RTTI概要

  1. 类型信息RTTI:即对象和类的信息,例如类的名字、继承的基类、实现的接口等。
  2. 类型信息的作用:程序员可以在程序运行时发现和使用类型信息。
  3. RTTI真正含义:运行时,识别一个对象的类型。

二、RTTI的实现方式

  1. 传统RTTI:即在编译时已知道了所有的类型
  2. 反射机制:在程序运行时发现和使用类的信息

三、传统RTTI的使用

 abstract class Shapes{
void draw(){
System.out.println(this+".draw()");
}
abstract public String toString();
} class Circle extends Shapes{
public String toString(){
return "Circle";
}
}
class Triangle extends Shapes{
public String toString(){
return "Triangle";
}
}
class Square extends Shapes{
public String toString(){
return "Square";
}
}
class Test{
public static List<Shapes>getList(){
List<Shapes> list_aShapes = Arrays.asList(new Circle,new Square,new Triangle);
}
} public class Shape{
public static void main(String args[]){
List<Shapes> list_aShapes = Test.getList();
for(Shapes shape:list_aShapes){
shape.draw();
}
}
}

  运行结果:Circle.draw()

  Square.draw()

  Triangle.draw()

  结果分析:

  上面代码中,shape对象只是一个泛化引用,显然一开始系统并不知道泛华引用的确切类型,我们希望使用的是shape对象对应的确切类型,这是,就需要系统使用RTTI。上例,只是打印出泛化引用的所有类型。进一步解析,Circle,Square,Triangle三个类都继承了抽象类Shape,现有一个List<Shape>的数组,存的是Circle,Square.Triangle的对象,当你拿出一个对象时,你只知道她是Shape类,但不知道它的具体类型。使用RTTI,可以查询到某个shape引用所指向对象的具体类型。

四、反射机制

  java反射的官方介绍是 反射机制是在运行中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性。

  反射机制能做什么?

    1. 在运行时判断任意一个类所属的类
    2. 在运行时构造任意一个类的对象
    3. 在运行时判断任意一个所具有的成员变量和方法
    4. 在运行时调用任意一个对象的方法
    5. 生成动态代理  

  其实,我们在学习框架时能够发现反射的广泛应用,通过反射和一个未知类型的对象打交道,Class类和java.lang.reflection类库一起对反射作技术支撑,该类库包含Field,Method和Constructor类,这些类的对象由JVM启动时创建,用以表示未知类里对应的成员。

  关于反射的详细内容之后的文章会介绍!

五、传统RTTI和反射的共同点和区别

  区别:

    一、如果该类在编译前就已知。也就是该类在classPath路径下。这就是传统RTTI。

    二、如果该类编译器未知,也就是在程序运行时才知道的。这就是反射

  所以RTTI和反射的本质区别只是检查一个类的.class文件的时机不同,反射:.class 文件是在编译时不可获得的,所以在运行时打开和检查未知类的.class文件从而变已知。RTTI:  .class 文件是在编译时打开和检查。

  相同点:都是基于Class类来执行的,都可以获取类的类型信息。

六、总述

  其实大家会发现,相比于反射,我们基本不会接触到传统RTTI机制,这是为何呢?其实它的名字已经给出了答案,这是一个机制---传统RTTI机制,其实在Java很多地方都依赖于这个机制来实现,比如说多态(当我们把子类对象的引用赋给父类对象的情况时,需要使用RTTI去识别该对象所属的类),所以说这并不是可有可有的。我们也可以在网上的很多文章看到,很多文章都是把RTTI分为传统RTTI和反射,以阐述java类型信息的详情,这样我们即可以很好地学习反射,也搞清楚了Java在运行时对于对象的识别方法。

Java类型信息(RTTI和反射)的更多相关文章

  1. Thinking in Java -- 类型信息RTTI

    Thinking in Java – 类型信息 个人感觉 java 中的比較难的部分了,在看了些netty源代码发现事实上这块很实用. 这章重点是RTTI和反射.先说下自己的理解 RTTI是执行时识别 ...

  2. JAVA类型信息——反射机制

    JAVA类型信息——反射机制 一.反射机制概述 1.反射机制:就是java语言在运行时拥有的一项自我观察的能力,java通过这种能力彻底了解程序自身的情况,并为下一步的动作做准备. 2.反射机制的功能 ...

  3. Java基础 -- 深入理解Java类型信息(Class对象)与反射机制

    一 RTTI概念 认识Claa对象之前,先来了解一个概念,RTTI(Run-Time Type Identification)运行时类型识别,对于这个词一直是 C++ 中的概念,至于Java中出现RT ...

  4. JAVA类型信息——Class对象

    JAVA类型信息——Class对象 一.RTTI概要 1.类型信息RTTI :即对象和类的信息,例如类的名字.继承的基类.实现的接口等. 2.类型信息的作用:程序员可以在程序运行时发现和使用类型信息. ...

  5. Java 类型信息

    <Thinking in Java 4th>第14章 类型信息 运行时类型信息(Run-Time Type Identification)使得你可以在程序运行时发现和使用类型信息. 14. ...

  6. Java类型信息

    一.引言 最近在阅读<Java编程思想>,学习一下java类型信息,现在做一下总结.Java如何让我们在运行时识别对象和类的信息的.主要有两种方式:一种是传统的“RTTI”,它假定我们在编 ...

  7. JAVA类型信息——Class对象(转载)

    JAVA类型信息--Class对象 一.RTTI概要 1.类型信息RTTI :即对象和类的信息,例如类的名字.继承的基类.实现的接口等. 2.类型信息的作用:程序员可以在程序运行时发现和使用类型信息. ...

  8. 【Java核心技术】类型信息(Class对象 反射 动态代理)

    1 Class对象 理解RTTI在Java中的工作原理,首先需要知道类型信息在运行时是如何表示的,这是由Class对象来完成的,它包含了与类有关的信息.Class对象就是用来创建所有“常规”对象的,J ...

  9. Java 类型信息详解和反射机制

    本文部分摘自 On Java 8 RTTI RTTI(RunTime Type Information)运行时类型信息,能够在程序运行时发现和使用类型信息,把我们从只能在编译期知晓类型信息并操作的局限 ...

随机推荐

  1. 快速获取dom到body左侧和顶部的距离,简单粗暴无bug-getBoundingClientRect

    获取dom到body左侧和顶部的距离-getBoundingClientRect 平时在写js的时候,偶尔会需要用js来获取当前div到 body 左侧.顶部的距离.网上查一查,有很多都是通过offs ...

  2. Docker学习之docker-compose

    docker-compose 安装 1.Mac/Windows: 安装docker的时候附带安装了. 2.Linux: curl https://github.com/docker/compose L ...

  3. php 上传word文件 Word文件转 PDF

    说一下,之前的时候做上传word文档转pdf的项目时候, 通过安装open office 连接服务组件, 遇到种种问题, 例如上传的word文档版本,, word文档中编辑的样式复杂会出现转pdf失败 ...

  4. Windows认证 | Windows本地认证

    Windows的登陆密码是储存在系统本地的SAM文件中的,在登陆Windows的时候,系统会将用户输入的密码与SAM文件中的密码进行对比,如果相同,则认证成功. SAM文件是位于%SystemRoot ...

  5. Django REST Framework之频率限制

    开放平台的API接口调用需要限制其频率,以节约服务器资源和避免恶意的频繁调用 使用 自定义频率限制组件:utils/thottle.py class MyThrottle(BaseThrottle): ...

  6. Scrapy项目 - 实现百度贴吧帖子主题及图片爬取的爬虫设计

    要求编写的程序可获取任一贴吧页面中的帖子链接,并爬取贴子中用户发表的图片,在此过程中使用user agent 伪装和轮换,解决爬虫ip被目标网站封禁的问题.熟悉掌握基本的网页和url分析,同时能灵活使 ...

  7. Java 代理模式 (二) 动态代理

    代理模式 代理(Proxy)是一种设计模式, 提供了对目标对象另外的访问方式:即通过代理访问目标对象. 这样好处: 可以在目标对象实现的基础上,增强额外的功能操作.(扩展目标对象的功能). 代理模式的 ...

  8. 利用Travis CI+GitHub实现持续集成和自动部署

    前言 如果你手动部署过项目,一定会深感持续集成的必要性,因为手动部署实在又繁琐又耗时,虽然部署流程基本固定,依然容易出错. 如果你很熟悉持续集成,一定会同意这样的观点:"使用它已经成为一种标 ...

  9. MongoDB 学习笔记之 mongo-connector安装

    mongo-connector安装: https://github.com/mongodb-labs/mongo-connector 1. yum -y install openssl* 2.编译安装 ...

  10. 【linux】jdk安装及环境变量配置

    登录linux后,切换目录到 /usr/local cd /user/local 在/usr/local目录新建文件夹java用于存放jdk文件 mkdir java 在文件夹java中下载jdk文件 ...