JVM要点

本文转载自:JVM 看这一篇就够了

一、JVM概述

  • JVM:Java Virtual Machine,也就是Java虚拟机
  • 所谓虚拟机是指:通过软件模拟的具有完整硬件系统功能的、运行在一个完全隔离环境中的计算机系统
  • JVM是通过软件来模拟Java字节码的指令集,是Java程序的运行环境

二、JVM主要功能

  • 通过 ClassLoader 寻找和装载 class 文件
  • 解释字节码成为指令并执行,提供 class 文件的运行环境
  • 进行运行期间的内存分配和垃圾回收
  • 提供与硬件交互的平台

三、虚拟机是Java平台无关的保障

1

四、JVM规范作用及其核

4.1 JVM规范作用
  • Java 虚拟机规范为不同的硬件提供了一种编译Java技术代码的规范
  • 该规范使Java软件独立于平台,因为编译时针对作为虚拟机的“一般机器”而做
  • 这个“一般机器”可用软件模拟并运行于各种现存的计算机系统,也可用硬件来实现
4.2 JVM规范定义的主要内容
  • 字节码指令集
  • Class文件的格式
  • 数据类型和值
  • 运行时数据区
  • 栈帧
  • 特殊方法
  • 类库
  • 异常
  • 虚拟机的启动、加载、链接和初始化

六、Class字节码解析

6.1 Class文件格式概述
  • Class文件是JVM的输入,Java虚拟机规范中定义了Class文件的结构,Class文件是JVM实现平台无关、技术无关的基础
  • 无符号数:基本数据类型,以u1、u2、u4、u8来代表几个字节的无符号数
  • 表:由多个无符号和其他表构成的符合数据类型,通常以 "_info"结尾
  • Class文件是一组以8字节为单位的字节流,各个数据项目按序紧凑排列
  • 对于占用空间大于8字节的数据项,按照高位在前的方式分割成多个8字节进行存储
  • Class文件格式里面只有两种类型:无符号数、表
6.2 Class文件的格式
  • javap工具生成非正式的 ”虚拟机汇编语言“,格式如下:

  • [[]…]] [comment]

  • 是指令操作码在数组中的下标,该数组以字节形式来存储当前方法的Java虚拟机代码;也可以是相当于方法起始处的字节偏移量

  • 是指令的助记码、是操作数、是行尾的注释

6.3 Class文件格式说明
  • constant_pool_count:是从1开始的

  • 不同的常量类型,用tag来区分,它后面对应的 info 结构是不一样的

  • L表示对象,[ 表示数组、V表示void

  • stack:方法执行时,操作栈的深度

  • Locals:局部变量所需的储存空间,单位是slot

  • slot是虚拟机为局部变量分配内存所使用的最小单位

  • args_size:参数个数,为1的话,因实例方法默认会传入this,locals也会预留一个slot来存放

七、ASM

7.1 ASM概述
  • ASM是一个Java字节码操纵框架,它能被用来动态生成类或者增强既有类的功能
  • ASM可以直接产生二进制class文件,也可以在类被加载入虚拟机之前动态改变类行为,ASM从类文件中读入信息后,能够改变类行为,分析类信息,甚至能根据要求生成新类
  • 目前许多框架如 cglib、Hibernate、spring 都直接或间接地使用ASM操作字节码
7.2 ASM编程模型
  • Core API:提供了基于事件形式的编程模型。该模型不需要一次性将整个类的结构读取到内存中,因此这种方式更快,需要的内存更少,但这种编程方式难度较大
  • Tree API:提供了基于树型的编程模型。该模型需要一次性将一个类的完整结构全部读取到内存中,所以这种方法需要更多的内存,这种编程方式较简单
7.3 ASM的Core API
  • ASM Core ApI 中操纵字节码的功能基于 ClassVisitor 接口。这个接口中的每个方法对应了 class 文件中的每一项

  • ASM 提供了三个基于 ClassVisitor 接口的类来实现 class 文件的生成和转换

  • ClassReader:ClassReader 解析一个类的 class 字节码

  • ClassAdapter:ClassAdapter 是 ClassVisitor 的实现类,实现要变化的功能

  • ClassWriter:ClassWriter 也是 ClassVisitro 的实现类,可以用来输出变化后的字节码

  • ASM 给我们提供了 ASMifier 工具来帮助开发,可使用ASMifier 工具生成 ASM 结构来对比

八、类加载、连接和初始化

8.1 类加载和类加载器
  • 类被加载到 JVM 开始,到卸载出内存,整个生命周期如图:
2
  • 加载:查找并加载类文件的二进制数据

  • 连接:就是将已经读入内存的类的二进制数据合并到 JVM 运行时环境中去,包含以下步骤:

  • 验证:确保被加载类的正确性

  • 准备:为类的 静态变量 分配内存,并初始化

  • 解析:把常量池中的符号引用转换成直接引用

  • 初始化:为类的静态变量赋初始值

8.2 类加载要完成的功能
  • 通过类的全限定名来获取该类的二进制字节流
  • 把二进制字节流转化为方法区的运行时数据结构
  • 在堆上创建一个 java.lang.Class 对象,用来封装类在方法区内的数据结构,并向外提供了访问方法区内数据结构的接口
8.3 加载类的方式
  • 最常见的方式:本地文件系统中加载、从jar等归档文件中加载
  • 动态的方式:将 java 源文件动态编译成 class
  • 其他方式:网络下载、从专有数据库中加载等等
8.4 类加载器
  • Java 虚拟机自带的加载器包括以下几种:
  • 启动类加载器(BootstrapClassLoader)
  • 平台类加载器(PlatformClassLoader) JDK8:扩展类加载器(ExtensionClassLoader)
  • 应用程序类加载器(AppClassLoader)
  • 用户自定义的加载器:是 java.lang.ClassLoader 的子类,用户可以定制类的加载方式;只不过自定义类加载器其加载的顺序是在所有系统类加载器的最后
8.5 类加载器的关系
3

九、类加载器使用

9.1 类加载器说明
  • 启动类加载器:用于加载启动的基础模块类,比如:java.base、java.management、java.xml等
  • 平台类加载器:用于加载一些平台相关的模块,比如:java.scripting、java.compiler *、java.corba *等
  • 应用程序类加载器:用于加载应用级别的模块,比如:jak.compiler、jdk.jartool、jdk.jshell 等等;还加载 classpath 路径中的所有类库
  • JDK8:启动类加载器:负责将<JAVA_HOME>/lib,或者 -Xbootclasspath 参数指定的路径中的,且是虚拟机识别的类库加载到内存中(按照名字识别,比如 rt.jar,对于不能识别的文件不予装载)
  • JDK8:扩展类加载器:负责加载 <JRE_HOME>/lib/ext,或者 java.ext.dirs 系统变量所指定路径中的所有类库
  • JDK8:应用程序类加载器:负责加载 classpath 路径中的所有类库
  • Java 程序不能直接引用启动类加载器,直接设置 classLoader 为 null,默认就使用启动类加载器
  • 类加载器并不需要等到某个类“首次主动使用”的时候才加载它,JVM规范允许类加载器在预料到某个类将要被使用的时候就预先加载它
  • 如果在加载的时候 .class 文件缺失,会在该类首次主动使用时报告 LinkageError 错误,如果一直没有被使用,就不会报错
9.2 双亲委派模型
  • JVM中的 ClassLoader 通常采用双亲委派模型,要求除了启动类加载器外,其余的类加载器都应该有自己的父级加载器。这里的父子关系是组合而不是继承,工作过程如下:
  • 一个类加载器接收到类加载请求后,首先搜索它的内建加载器定义的所有“具名模块”
  • 如果找到了合适的模块定义,将会使用该加载器来加载
  • 如果 class 没有在这些加载器定义的具名模块中找到,那么将委托给父级加载器,直到启动类加载器
  • 如果父级加载器反馈它不能完成加载请求,比如在它的搜索路径下找不到这个类,那子类加载器才自己来加载
  • 在类路径下找到的类将成为这些加载器的无名模块
  • 双亲委派模型对于保证 Java 程序的稳定运作很重要,可以避免一个类被加载多次
  • 实现双亲委派的代码在 java.lang.ClassLoader 的 loadClass() 方法中,如果自定义类加载器的话,推荐覆盖实现 findClass() 方法
  • 如果有一个类加载器能加载某个类,称为 定义类加载器,所有能成功返回该类的 Class 的类加载器 都被称为初始类加载器
9.3 双亲委派模型的说明
  • 双亲委派模型对于保证 Java 程序的稳定运作很重要
  • 实现双亲委派的代码 java.lang.ClassLoader 的 loadClass() 方法中,如果自定义类加载器的话,推荐覆盖实现 findClass() 方法
  • 如果有一个类加载器能加载某个类,称为 定义类加载器,所有能成功返回该类的 Class 的类加载器 都被称为 初始化加载器
  • 如果没有指定父加载器,默认就是启动类加载器
  • 每个类加载器都有自己的命名空间,命名空间由该类加载器及其所有父加载器所加载的类构成,不同的命名空间,可以出现类的全路径名 相同的情况
  • 运行时包由同一个类加载器的类构成,决定两个类是否属于同一个运行时包,不仅要看全路径名是否一样,还要看定义类加载器是否相同。只有属于同一个运行时包的类才能实现相互包内可见
9.4 破坏双亲委派模型
  • 双亲委派模型有一个问题:父加载器无法向下识别子加载器加载的资源
  • 为了解决这个问题,引入了线程上下文类加载器,可以通过 Thread 的 setContextClassLoader() 进行设置
  • 实现热部署时,比如 OSGI 的模块化热部署,它的类加载器就不再是严格按照双亲委派模型,很多可能就在平级的类加载器中执行了
9.5 双亲委派加载顺序

Java的双亲委派机制是Java类加载器的一种工作模式。它是为解决类加载冲突和安全性问题而设计的。

当一个类需要被加载时,Java虚拟机会按照特定的顺序尝试加载这个类。首先,它会将这个请求交给最顶层的类加载器——启动类加载器(Bootstrap Classloader)。如果启动类加载器无法完成加载任务(因为启动类加载器只加载核心类库),那么它会请求它的父类加载器(根据双亲委派模型,通常是扩展类加载器)来加载该类。如果父类加载器仍然无法加载这个类,那么它会再请求它的父类加载器的父类加载器,直到达到顶层的启动类加载器。

这种层层委派的机制能够保证在一个Java应用程序中,同一个类只会被加载一次,并且由同一个类加载器加载。这样可以避免不同的类加载器对同一个类进行加载,造成类的冗余或者不一致。同时,双亲委派机制还可以有效地提高类的安全性,防止恶意代码替换核心类库。

总结起来,双亲委派机制的步骤如下:

  1. 当一个类需要被加载时,首先询问最顶层的启动类加载器。
  2. 如果启动类加载器找不到该类,请求父类加载器加载。
  3. 父类加载器再将加载请求往上委派,直到找到能加载该类的加载器或者无法再往上委派为止。
  4. 如果所有的父类加载器都无法加载该类,则由最底层的加载器尝试加载。

通过双亲委派机制,Java保证了类的唯一性和安全性。

十、类连接和初始化

10.1 类连接主要验证的内容
  • 类文件结构检查:按照 JVM 规范规定的类文件结构进行
  • 元数据验证:对字节码描述的信息进行语义分析,保证其符合 Java 语言规范要求
  • 字节码验证:通过对数据流和控制流进行分析,确保程序语义是合法和符合逻辑的。这里主要对方法体进行校验
  • 符号引用验证:对类自身以外的信息,也就是常量池中的各种符号引用,进行匹配校验
10.2 类连接中的准备
  • 为类的 静态变量 分配内存,并初始化
10.3 类连接中的解析
  • 解析就是把常量中的符号引用转换成直接引用的过程,包括:符号引用:以一组无歧义(唯一)的符号来描述所引用的目标,与虚拟机的失效无关
  • 直接引用:直接执行目标的指针、相对偏移量、或是能间接定位到目标的句柄,是和虚拟机实现相关的
  • 主要针对:类、接口、字段、类方法、接口方法、方法类型、方法句柄、调用点限定符
10.4 类的初始化
  • 类的初始化就是为类的静态变量赋初始值,或者说是执行类构造器 方法的过程
  • 初始化一个类的时候,并不会先初始化它实现的接口
  • 初始化一个接口的时候,并不会先初始化它的父接口
  • 只有当程序首次使用接口里面的变量或者是调用接口方法的时候,才导致接口初始化
  • 如果类还没有加载和连接,就先加载和连接
  • 如果类存在父类,且父类没有初始化,就先初始化父类
  • 如果类中存在初始化语句,就依次执行这些初始化语句
  • 如果是接口的话:
  • 调用 Classloader 类的 loadClass 方法类装载一个类,并不会初始化这个类,不是对类的主动使用

十一、类的主动初始化

11.1 类的初始化时机
  • Java 程序对类的使用方式分成:主动使用和被动使用,JVM 必须在每个类或接口 ”首次主动使用“ 时才初始化它们;被动使用类不会导致类的初始化,主动使用的情况:
  • 创建类实例
  • 访问某个类或接口的静态变量
  • 调用类的静态方法
  • 反射某个类
  • 初始化某个类的子类,而父类还没有初始化
  • JVM 启动的时候运行的主类
  • 定义了 default 方法的接口,当接口实现类初始化时
11.2 类的卸载
  • 当代表一个类的 Class 对象不再被引用,那么 Class 对象的生命周期就结束了,对应的在方法区中的数据也会被卸载
  • JVM 自带的类加载器装载的类,是不会卸载的,由用户自定义的类加载器的加载的类是可以卸载的
11.3 运行时数据区
  • PC 寄存器、Java虚拟机栈、Java堆、方法区、运行时常量池、本地方法栈等
11.4 PC 寄存器
  • 每个线程都拥有一个PC寄存器,是线程私有的,用来存储指向下一条指令的地址
  • 在创建线程的时候,创建相应的PC寄存器
  • 执行本地方法时,PC寄存器的值为 undefined
  • 是一块比较小的内存空间,是唯一一个在JVM规范中没有规定 OutOfMemoryError 的内存区域
11.4 Java栈
  • 栈由一系列帧(栈帧)(Frame)组成(因此Java栈也叫做帧栈),是线程私有的
  • 栈帧用来保存一个方法的局部变量、操作数栈(Java没有寄存器,所有参数传递使用操作数栈)、常量池指针、动态链接、方法返回等
  • 每一次方法调用创建一个帧,并压栈,退出方法的时候,修改栈顶指针就可以把栈帧中的内容销毁
  • 局部变量表存放了编译期可知的各种基本数据类型和引用类型,每个 slot 存放32位的数据,long、double、占两个槽位
  • 栈的优点:存取速度比堆块,仅次于寄存器
  • 栈的缺点:存在栈中的数据大小、生存区是在编译器决定的,缺乏灵活性
11.4 Java堆
  • 用来存放应用系统创建的对象和数组,所有线程共享 Java 堆
  • GC主要管理堆空间,对分代GC来说,堆也是分代的
  • 堆的优点:运行期动态分配内存大小,自动进行垃圾回收;
  • 堆的缺点:效率相对较慢
11.5 方法区
  • 方法区是线程共享的,通常用来保存装载的类的结构信息
  • 通常和元空间关联在一起,但具体的跟JVM实现和版本有关
  • JVM规范把方法区描述为堆的一个逻辑部分,但它有一个别名称为 Non-heap(非堆),应是为了与 Java 堆分开
11.6 运行时常量池
  • 是Class文件中每个类或接口的常量池表,在运行期间的表示形式,通常包括:类的版、字段、方法、接口等信息
  • 在方法区中分配
  • 通常在加载类和接口到JVM后,就创建相应的运行时常量池
11.7 本地方法栈
  • 在 JVM 中用来支持 native 方法执行的栈就是本地方法栈

十二、Java堆内存模型和分配

12.1 Java堆内存概述
  • Java 堆用来存放应用系统创建的对象和数组,所有线程共享Java堆
  • Java堆是在运行期动态分配内存大小,自动进行垃圾回收
  • Java垃圾回收(GC)主要就是回收堆内存,对分代GC来说,堆也是分代的
12.2 Java堆的结构
4
  • 新生代用来存放新分配的对象;新生代中经过垃圾回收,没有回收掉的对象,被复制到老年代
  • 老年代存储对象比新生代存储对象的年龄大得多
  • 老年代存储一些大对象
  • 整个堆大小 = 新生代 + 老年代
  • 新生代 = Eden + 存活区
  • 从前的持久代,用来存放Class、Method 等元信息的区域,从 JDK8 开始去掉了,取而代之的是元空间(MetaSpace),元空间并不在虚拟机里面,而是直接使用本地内存
12.3 对象内存布局
  • 对象在内存中储存的布局(这里以Hotspot虚拟机为例来说明),分为:对象头、实例数据和对齐填充
  • 对象头,包含两部分:
    1、Mark Word:存储对象自身的运行数据,如:HashCode、GC分代年龄,锁状态标志等
    2、类型指针:对象指向它的类元数据的指针
  • 实例数据:真正存放对象实例数据的地方
  • 对齐填充:这部分不一定存在,也没有什么特别含义,仅仅是占位符。因为 HotSpot 要求对象起始地址都是8字节的整数倍,如果不是,就对齐
12.4 对象的访问定位
  • 使用句柄:Java堆中会划分出一块内存来作为句柄池,reference 中存储句柄的地址,句柄中存储对象的实例数据和类元数据的地址,如图
5
  • 使用指针:Java堆中会存放访问类元数据的地址,reference存储的就直接是对象的地址,如图:
6

十三、Trace跟踪和Java堆的参数配置

13.1 Trace跟踪参数
  • 可以打印GC的简要信息:-Xlog:gc
  • 打印GC详细信息:-Xlog:gc*
  • 指定GC log的位置,以文件输出:-Xlog:gc:garbage-collection.log
  • 每一次GC后,都打印堆信息:-xlog:gc+heap = debug
13.2 GC 日志格式
  • GC发生的时间,也就是 JVM 从启动以来经过的秒数
  • 日志级别信息,和日志类型标记
  • GC 识别号
  • GC 的类型和说明 GC 的原因
  • 容量:GC 前容量 -> GC后容量(该区域总容量)
  • GC 持续时间,单位秒。有的收集器会有详细的描述,比如:user表示应用程序消耗的时间,sys表示系统内核消耗的时间,real 表示操作从开始到结束的时间
13.3 Java堆的参数
  • -Xms:初始化堆大小,默认物理内存的 1/64
  • -Xmx:最大堆大小,默认物理内存的 1/4
  • -Xmn:新生代大小,默认整个堆的 3/8
  • -XX:+HeapDumpOnOutOfMemoryError:OOM时导出堆到文件
  • -XX:+HeapDumpPath:导出 OOM 的路径
  • -XX:NewRatio:老年代与新生代的比值,如果 xms=xmx,且设置了 xmn 的情况,该参数不用设置
  • -XX:SurvivorRatio:Eden区和Survivor区的大小比值,设置为8,则两个 Survivor 区与一个Eden区的比值为 2:8,一个 Survivor 占整个新生的 1/10
  • -XX:OnOutOfMemoryError:在OOM时,执行一个脚本
  • -Xss:通常只有几百k,决定了函数调用的深度
13.4 元空间的参数
  • -XX:MetaspaceSize:初始空间大小
  • -XX:MaxMetaspaceSize:最大空间,默认是没有限制的
  • -XX:MinMetaspaceFreeRatio:在GC之后,最小的Metaspace 剩余空间容量的百分比
  • -XX:MaxMetaspaceFreeRatio:在GC之后,最大的Metaspace剩余空间容量的百分比
13.5 字节码执行引擎
  • JVM 的字节码执行引擎,功能基本就是输入字节码文件,然后对字节码进行解析并处理,最后输出执行的结果
  • 实现方式可能有通过解释器直接解释执行字节码,或者通过即时编译器产生本地代码,也就是编译执行,当然也可能两者都有
13.6 栈帧
  • 栈帧是用于支持JVM进行方法调用和方法执行的数据结构
  • 栈帧随着方法调用而创建,随着方法结束而销毁
  • 栈帧里面存储了方法的局部变量表、操作数栈、动态链接、方法返回地址等信息
13.7 局部变量表
  • 局部变量表:用来存放方法参数和方法内部定义的局部变量的存储空间
  • 以变量槽 slot 为单位,目前一个 slot 存放32位以内的数据类型
  • 对于64位的数据占2个slot
  • 对于实例方法,第0位 slot 存放的是 this,然后从1到n,依次分配给参数列表
  • 然后根据方法体内部定义的变量顺序和作用域来分配 slot
  • slot 是复用的,以节省栈帧的空间,这种设计可能会影响系统的垃圾收集行为
13.7 操作数栈
  • 操作数栈:用来存放方法运行期间,各个指令操作的数据
  • 操作数栈中元素的数据类型必须和字节码指令的顺序严格匹配
  • 虚拟机在实现栈帧的时候可能会做一些优化,让两个栈帧出现部分重叠区域,以存放公用的数据
13.8 动态链接
  • 动态链接:每一个栈帧持有一个指向运行时常量池中该栈帧所属方法的引用,以支持方法调用过程的动态链接
  • 静态解析:类加载的时候,符号引用就转化为直接引用
  • 动态链接:运行期间转化为直接引用
13.9 方法返回地址
  • 方法返回地址:方法执行后返回的地址
13.10 方法调用
  • 方法调用:就是确定具体调用哪一个方法,并不涉及方法内部的执行过程
  • 部分方法是直接在类加载的解析阶段,就确定了直接引用关系
  • 但是对于实例方法,也称虚方法,因为多重和多态,需要运行期动态分派
13.11 分派
  • 静态分派:所有依赖静态类型来定位方法执行版本的分派方式,比如:重载方法
  • 动态分派:根据运行期的实际类型来定位方法执行版本的分派方式,比如:覆盖方法
  • 单分派和多分派:就是按照分派思考的维度,多于一个的就算多分配,只有一个的称为单分派

十四、垃圾回收

14.1 垃圾回收概述
  • 什么是垃圾:简单说就是内存中已经不再被使用到的内存空间就是垃圾

  • 垃圾回收算法:

  • 可作为GC Roots的对象包括:虚拟机栈(栈帧局部变量)中引用的对象、方法区类静态属性引用的对象、方法区中常量引用的对象、本地方法栈中JNI引用的对象

  • HotSpot 使用了一组叫做 OopMap 的数据结构达到准确式GC的目的

  • 在OopMap的协助下,JVM可以很快的做完GC Roots 枚举。但是JVM并没有为每一条指令生成一个OopMap

  • 记录OopMap 的这些“特定位置”被称为安全点,即当前线程执行到安全点后才允许暂停进行GC

  • 如果一段代码中,对象引用关系不会发生变化,这个区域中任何地方开始GC都是安全的,那么这个区域称为安全区域

  • 优点:失效简单、效率高

  • 缺点:不能解决对象之间循环引用的问题

  • 引用计数法:给对象添加一个引用计数器,有访问就加1,引用失效就减1

  • 根搜索算法(可达性分析法):从根(GC Roots)节点向下搜索对象节点,搜索走过的路径称为引用链,当一个对象到根之间没有连通的话,则该对象不可用

14.2 跨代引用
  • 跨代引用:也就是一个代中的对象引用另一个代中的对象
  • 跨代引用假说:跨代引用相对于同代引用来说只是极少数
  • 隐含推论:存在相互引用关系的两个对象,是应该倾向于同时生存或同时消亡的
14.3 记忆集
  • 记忆集(Remembered Set):一种用于记录从非收集区域指向收集区域的指针集合的抽象数据结构
  • 字长精度:每个记录精确到一个机器字长,该子包含跨代指针
  • 对象精度:每个记录精确到一个对象,该对象里有字段含有跨代指针
  • 卡精度:每个记录精确到一块内存区域,该区域内有对象含有跨代指针
  • 卡表(Card Table):是记忆集的一种具体实现,定义了记忆集的记录精度和与堆内存的映射关系等
  • 卡表的每个元素都对应着其标识的内存区域中一块特定大小的内存块,这个内存块称为 卡页(Card Page)
14.4 写屏障
  • 写屏障可以看成是JVM对 ”引用类型字段赋值“ 这个动作的AOP
  • 通过写屏障来实现当对象状态改变后,维护卡表状态
14.5 判断是否垃圾的步骤
  • 跟搜索算法判断不可用
  • 看是否有必要执行 finalize 方法
  • 两个步骤走完后对象仍然没有人使用,那就属于垃圾
14.6 GC 类型
  • MinorGC / YoungGC:发生在新生代的收集动作
  • MajorGC / OldGC:发生在老年代的GC,目前只有CMS收集器会有单独收集老年代的行为
  • MixedGC:收集整个新生代以及部分老年代,目前只有G1收集器会有这种行为
  • FullGC:收集整个Java堆和方法区的GC
14.7 Stop-The-World
  • STW是Java中一种全局暂停的现象,多半由于GC引起。所谓全局停顿,就是所有Java代码停止运行,native代码可以执行,但不能和JVM交互
  • 其危害是长时间服务停止,没有响应;对于HA系统,可能引起主备切换,严重危害生产环境
14.8 垃圾收集类型
  • 串行收集:GC单线程内存回收、会暂停所有的用户线程,如:Serial
  • 并行收集:多个GC线程并发工作,此时用户线程是暂停的,如:Parallel
  • 并发收集:用户线程和GC线程同时执行(不一定是并行,可能交替执行),不需要停顿用户线程,如:CMS
14.9 判断类无用的条件
  • JVM 中该类的所有实例都已经被回收
  • 加载该类的 ClassLoader 已经被回收
  • 没有任何地方引用该类的 Class 对象
  • 无法在任何地方通过反射访问这个类
14.10 垃圾回收算法
14.10.1 标记清除算法
  • 标记清除算法(Mark-Sweep):分为标记和清除两个阶段,先标记出要回收的对象,然后统一回收这些对象

  • 在这里插入图片描述

  • 优点:简单

  • 缺点:

  • 效率不高,标记和清除的效率都不高

  • 产生大量不连续的内存碎片,从而导致在分配大对象时触发GC

14.10.2 复制算法
  • 复制算法(Copying):把内存分成两块完全相同的区域,每次使用其中一块,当一块使用完了,就把这块上还存活的对象拷贝到另外一块,然后把这块清除掉
  • 在这里插入图片描述
  • 优点:实现简单,运行高效,不用考虑内存碎片问题
  • 缺点:内存有些浪费
  • JVM实际实现中,是将内存分为一块较大的Eden区和两块较小的 Survivor 空间,每次使用Eden和一块 Survivor,回收时,把存活的对象复制到另一块 Survivor
  • HotSpot 默认的 Eden 和 Survivor 比是 8:1,也就是每次能用 90% 的新生代空间
  • 如果 Survivor 空间不够,就要依赖老年代进行分配担保,把放不下的对象直接进入老年代分配担保:当新生代进行垃圾回收后,新生代的存活区放置不下,那么需要把这些对象放置到老年代去的策略,也就是老年代为新生代的GC做空间分配担保,步骤如下:
  • 在发生 MinorGC 前,JVM会检查老年代的最大可用的连续空间,是否大于新生代所有对象的总空间,如果大于,可以确保 MinorGC 是安全的
  • 如果小于,那么JVM会检查是否设置了允许担保失败,如果允许,则继续检查老年代最大可用的连续空间,是否大于历次晋升到老年代对象的平均大小
  • 如果大于,则尝试进行一次 MinorGC
  • 如果不大于,则改做一次 Full GC
14.10.3 标记整理算法
  • 标记整理算法(Mark-Compact):由于复制算法在存活对象比较多的时候,效率较低,且有空间浪费,因此老年代一般不会选用复制算法,老年代多选用标记整理算法
  • 标记过程跟标记清除算法一样,但后续不是直接清除可回收对象,而是让所有存活对象都向一端移动,然后直接清除边界以外的内存
  • 在这里插入图片描述

垃圾收集器

  • 串行收集器、并行收集器、新生代Parallel、Scavenge收集器、CMS、G1
  • 在这里插入图片描述

串行收集器

  • Serial(串行)收集器 / Serial Old 收集器,是一个单线程的收集器,在垃圾收集时,会 Stop-the-World
  • 在这里插入图片描述
  • 优点:简单,对于单cpu,由于没有多线程的交互开销,可能更高效,是默认的 Client 模式下的新生代收集器
  • 使用 -XX:+UseSerialGC 来开启,会使用:Serial + SerialOld 的收集器组合
  • 新生代使用复制算法,老年代使用标记-整理算法

并行收集器****ParNew收集器

  • ParNew(并行)收集器:使用多线程进行垃圾回收,在垃圾收集时,会Stop-the-World
  • 在这里插入图片描述
  • 在并发能力好的 CPU 环境里,它停顿的时间要比串行收集器短;但对于单 CPU 或并发能力较弱的CPU,由于多线程的交互开销,可能比串行回收器更差
  • 是 Server 模式下首选的新生代收集器,且能和 CMS 收集器配合使用
  • 不再使用 -XX:+UseParNewGC来单独开启
  • -XX:ParallelGCThreads:指定线程数,最好与 cpu 数量一致

新生代Parallel Scavenge 收集器

  • 新生代 Parallel Scavenge 收集器 / Parallel Old 收集器:是一个应用于新生代的,使用复制算法的、并行的收集器
  • 与 ParNew 很类似,但更关注吞吐量,能最高效率的利用 CPU,适合运行后台应用
  • 在这里插入图片描述
  • 使用 -XX:+UseParallelGC 来开启
  • 使用 -XX:+UseParallelOldGC 来开启老年代使用 ParallelOld收集器,使用 Parallel Scavenge + Parallel Old 的收集器组合
  • -XX:MaxGCPauseMillis:设置GC 的最大停顿时间
  • 新生代使用复制算法,老年代使用标记-整理算法

CMS收集器

  • CMS(Concurrent Mark and Sweep 并发标记清除)收集器分为:初始标记:只标记GC Roots 能直接关联到的对象;并发标记:进行GC Roots Tracing 的过程

  • 重新标记:修正并发标记期间,因程序运行导致标记发生变化的那一部分对象

  • 并发清除:并发回收垃圾对象

  • 在这里插入图片描述

  • 在初始化标记和重新标记两个阶段还是会发生 Stop-the-World

  • 使用标记清除算法,多线程并发收集的垃圾收集器

  • 最后的重置线程,指的是清空跟收集相关的数据并重置,为下次收集做准备

  • 优点:低停顿,并发执行

  • 缺点:

  • 并发执行,对 CPU 资源压力大

  • 无法处理 在处理过程中 产生的垃圾(浮动垃圾),可能导致 FullGC

  • 采用的标记清除算法会导致大量碎片,从而在分配大对象可能触发 FullGC

  • 开启:-XX:UseConcMarkSweepGC:使用 ParNew + CMS + Serial Old 的收集器组合,Serial Old 将作为 CMS 出错的后备收集器

  • -XX:CMSInitiatingOccupancyFraction:设置 CMS 收集器在老年代空间被使用多少后触发回收,默认 80%

G1收集器

  • G1(Garbage-First)收集器:是一款面向服务应用的收集器,与其他收集器相比,具有以下特点:

  • G1 把内存划分成多个独立的区域(Region)

  • G1 仍采用分代思想,保留了新生代和老年代,但它们不再是物理隔离的,而是一部分Region的集合,且不需要 Region 是连续的

  • 在这里插入图片描述

  • G1 能充分利用多 CPU 、多核环境硬件优势,尽量缩短 STW

  • G1 整体上采用标记-整理算法,局部是通过复制算法,不会产生内存碎片

  • G1 的停顿可预测,能明确指定在一个时间段内,消耗在垃圾收集上的时间不能超过多长时间

  • G1 跟踪各个 Region 里面垃圾堆的价值大小,在后台维护一个优先列表,每次根据允许的时间来回收价值最大的区域,从而保证在有限时间内的高效收集

  • 垃圾收集:

  • 初始标记:只标记GC Roots 能直接关联到的对象

  • 并发标记:进行 GC Roots Tracing 的过程

  • 最终标记:修正并发标记期间,因程序运行导致标记发生变化的那一部分对象

  • 筛选回收:根据时间来进行价值最大化的回收

  • 使用和配置G1:-XX:+UseG1GC:开启G1,默认就是G1

  • -XX:MaxGCPauseMillis = n :最大GC停顿时间,这是个软目标,JVM将尽可能(但不保证)停顿小于这个时间

  • -XX:InitiatingHeapOccupancyPercent = n:堆占用了多少的时候就触发GC,默认为45

  • -XX:NewRatio = n:默认为2

  • -XX:SurvivorRatio = n:默认为8

  • -XX:MaxTenuringThreshold = n:新生代到老年代岁数,默认是15

  • -XX:ParallelGCThreads = n:并行GC的线程数,默认值会根据平台不同而不同

  • -XX:ConcGCThreads = n:并发 GC 使用的线程数

  • -XX:G1ReservePercent = n:设置作为空闲空间的预留内存百分比,以降低目标空间溢出的风险,默认值是 10%

  • -XX:G1HeapRegionSize = n:设置的 G1 区域的大小。值是2的幂,范围是1MB到32MB,目标是根据最小的Java堆大小划分出约2048个区域

14.11 ZGC收集器(了解)
  • ZGC收集器:JDK11加入的具有实验性质的低延迟收集器

  • ZGC的设计目标是:支持TB级内存容量,暂停时间低(<10ms),对整个程序吞吐量的影响小于15%

  • ZGC里面的新技术:着色指针 和 读屏障

  • GC性能指标:

  • 吞吐量 = 应用代码执行的时间 / 运行的总时间

  • GC负荷,与吞吐量相反,是 GC 时间 / 运行的总时间

  • 暂停时间,就是发生 Stop-the-World 的总时间

  • GC 频率,就是GC在一个时间段发生的次数

  • 反应速度:就是从对象成为垃圾开始到被回收的时间

  • 交互式应用通常希望暂停时间越少越好

  • JVM内存配置原则:

  • 新生代尽可能设置大点,如果太小会导致:

  • 对于老年代,针对响应时间优先的应用:由于老年代通常采用并发收集器,因此其大小要综合考虑并发量和并发持续时间等参数

  • 对于老年代,针对吞吐量优先的应用:通常设置较大的新生代和较小的老年代,这样可以尽可能回收大部分短期对象,减少中期对象,而老年代尽量存放长期存活的对象

  • 依据对象的存活周期进行分类,对象优先在新生代分配,长时间存活的对象进入老年代

  • 根据不同代的特点,选取合适的收集算法:少量对象存活,适合复制算法;大量对象存活,适合标记清除或标记整理

  • 如果设置小了,可能会造成内存碎片,高回收频率会导致应用暂停

  • 如果设置大了,会需要较长的回收时间

  • YGC 次数更加频繁

  • 可能导致 YGC 后的对象进入老年代,如果此时老年代满了,会触发FGC

十五、高效并发

15.1 Java内存模型
  • JCP 定义了一种 Java 内存模型,以前是在 JVM 规范中,后来独立出来成为JSR-133(Java内存模型和线程规范修订)
  • 内存模型:在特定的操作协议下,对特定的内存或高速缓存进行读写访问的过程抽象
  • Java 内存模型主要关注 JVM 中把变量值存储到内存和从内存中取出变量值这样的底层细节
  • 在这里插入图片描述
  • 所有变量(共享的)都存储在主内存中,每个线程都有自己的工作内存;工作内存中保存该线程使用到的变量的主内存副本拷贝
  • 线程对变量的所有操作(读、写)都应该在工作内存中完成
  • 不同线程不能相互访问工作内存,交互数据要通过主内存

内存间的交互操作

  • Java内存模型规定了一些操作来实现内存间交互,JVM会保存它们是原子的
  • lock:锁定,把变量标识为线程独占,作用于主内存变量
  • unlock:解锁,把锁定的变量释放,别的线程才能使用,作用于主内存变量
  • read:读取,把变量从主内存读取到工作内存
  • load:载入,把read读取到的值放入工作内存的变量副本中
  • use:使用,把工作内存中一个变量的值传递给执行引擎
  • assign:赋值,把从执行引擎接收到的值赋给工作内存里面的变量
  • store:存储,把工作内存中一个变量的值传递到主内存中
  • wirte:写入,把 store 进来的数据存放如主内存的变量中
  • 在这里插入图片描述
15.2 内存间的交互操作的规则
  • 不允许 read 和 load 、store 和 write 操作之一单独出现,以上两个操作必须按照顺序执行,但不保证连续执行,也就是说,read 和 load 之间、store 与 write 之间是可插入其他指令的
  • 不允许一个线程丢弃它的最近的 assign 操作,即变量在工作内存中改变了之后必须把该变化同步回主内存
  • 不允许一个线程无原因地(没有发生过任何 assign 操作)把数据从线程的工作内存同步回主内存中
  • 一个新的变量只能从主内存中 ”诞生“,不允许在工作内存中直接使用一个未被初始化的变量,也就是对一个变量实施 use 和 store 操作之前,必须先执行过了 assign 和 load 操作
  • 一个变量在同一个时刻只允许一条线程对其执行 lock 操作,但 lock 操作可以被同一条线程重复执行多次,多次执行lock后,只有执行相同次数的 unlock 操作,变量才会被解锁
  • 如果对一个变量执行 lock 操作,将会清空工作内存中此变量的值,在执行引擎使用这个变量前,需要重新执行 load 或 assign 操作初始化变量的值
  • 如果一个变量没有被 lock 操作锁定,则不允许对它执行 unlock 操作,也不能 unlock 一个被其他线程锁定的变量
  • 对一个变量执行 unlock 操作之前,必须先把此变量同步回主内存(执行 store 和 write 操作)
15.3 volatile特性
15.3.1多线程中的可见性
  • 可见性:就是一个线程修改了变量,其他线程可以知道
  • 保证可见性的常见方法:volatile、synchronized、final(一旦初始化完成,其他线程就可见)
15.3.2 volatile
  • volatile 基本上是 JVM 提供的最轻量级的同步机制,用 volatile 修饰的变量,对所有的线程可见,即对 volatile 变量所做的写操作能立即反映到其他线程中

  • 用 volatile 修饰的变量,在多线程环境下仍然是不安全的

  • volatile 修饰的变量,是禁止指令重排优化的

  • 适合使用 valatile 的场景

  • 运算结果不依赖变量的当前值

  • 确保只有一个线程修改变量的值

15.3.3 指令重排
  • 指令重排:指的是 JVM 为了优化,在条件允许的情况下,对指令进行一定的重新排列,直接运行当前能够立即执行的后序指令,避开获取下一条指令所需数据造成的等待

  • 线程内串行语义,不考虑多线程间的语义

  • 不是所有的指令都能重排,比如:

  • 写后读 a = 1;b = a;写一个变量之后,再读这个位置

  • 写后写 a = 1;a = 2;写一个变量之后,再写这个变量

  • 读后写 a = b;b = 1;读一个变量之后,再写这个变量

  • 以上语句不可重排,但是 a = 1;b = 2;是可以重排的

  • 程序顺序原则:一个线程内保证语义的串行性

  • volatile规则:volatile 变量的写,先发生于读

  • 锁规则:解锁(unlock)必然发生在随后的加锁(lock)前

  • 传递性:A 先于 B,B 先于 C,那么 A 必然先于 C

  • 线程的 start 方法先于它的每一个动作

  • 线程的所有操作先于线程的终结

  • 线程中断(interrupt())先于被中断线程的代码

  • 对象的构造函数执行结束先于 finalize() 方法

15.4 Java线程安全的处理方法
  • 不可变是线程安全的

  • 互斥同步(阻塞同步):synchronized、java.util.concurrent.ReentrantLock。目前这两个方法性能已经差不多了,建议优先选用 synchronized,ReentrantLock 增加了如下特性:

  • 等待可中断:当持有锁的线程长时间不释放锁,正在等待的线程可以选择放弃等待

  • 公平锁:多个线程等待同一个锁时,须严格按照申请锁的时间顺序来获取锁

  • 锁绑定多个条件:一个 ReentrantLock 对象可以绑定多个 condition 对象,而 synchronized 是针对一个条件的,如果要多个,就得有多个锁

  • 非阻塞同步:是一种基于冲突检查的乐观锁策略,通常是先操作,如果没有冲突,操作就成功了,有冲突再采取其他方式进行补偿处理

  • 无同步方案:其实就是在多线程中,方法并不涉及共享数据,自然也就无需同步了

15.5 锁优化
15.5.1 自旋锁与自适应自旋
  • 自旋:如果线程可以很快获得锁,那么可以不再 OS 层挂起线程,而是让线程做几个忙循环,这就是自旋
  • 自适应自旋:自旋的时间不再固定,而是由前一次在同一个锁上的自旋时间和锁的拥有者状态来决定
  • 如果锁被占用时间很短,自旋成功,那么能节省线程挂起、以及切换时间,从而提升系统性能
  • 如果锁被占用时间很长,自旋失败,会白白浪费处理器资源,降低系统性能
15.5.2 锁消除
  • 在编译代码的时候,检测到根本不存在共享数据竞争,自然也就无需同步加锁了;通过 -XX:+EliminateLocks 来开启
  • 同时要使用 -XX:DoEscapeAnalysis 开启逃逸分析逃逸分析:
  • 如果一个方法中定义的一个对象,可能被外部方法引用,称为方法逃逸
  • 如果对象可能被其他外部线程访问,称为线程逃逸,比如赋值给类变量或者可以在其他线程中访问的实例变量
15.5.3 锁粗化
  • 通常我们都要求同步块要小,但一系列连续的操作导致一个对象反复的加锁和解锁,这会导致不必要的性能损耗。这种情况建议把锁同步的范围加大到整个操作序列
15.5.4 轻量级锁
  • 轻量级是相对于传统锁机制而言,本意是没有多线程竞争的情况下,减少传统锁机制使用 OS 实现互斥所产生的性能损耗
  • 其实现原理很简单,就是类似乐观锁的方式
  • 如果轻量级锁失败,表示存在竞争,升级为重量级锁,导致性能下降
15.5.5 偏向锁
  • 偏向锁是在无竞争情况下,直接把整个同步消除了,连乐观锁都不用,从而提高性能;所谓的偏向,就是偏心,即锁会偏向于当前已经占有锁的线程
  • 只要没有竞争,获得偏向锁的线程,在将来进入同步块,也不需要做同步
  • 当有其他线程请求相同的锁时,偏向模式结束
  • 如果程序中大多数锁总是被多个线程访问的时候,也就是竞争比较激烈,偏向锁反而会降低性能
  • 使用 -XX:-UseBiasedLocking 来禁用偏向锁,默认开启
15.6 JVM 中获取锁的步骤
  • 会先尝试偏向锁;然后尝试轻量级锁
  • 再然后尝试自旋锁
  • 最后尝试普通锁,使用 OS 互斥量在操作系统层挂起
15.7 同步代码的基本规则
  • 尽量减少持有锁的时间
  • 尽量减少锁的粒度
15.8 性能监控与故障处理工具
15.8.1 命令行工具
  • 命令行工具:jps、jinfo、jstack、jmap、jstat、jstatd、jcmd
  • 图形化工具:jconsole、jmc、visualvm
  • 两种连接方式:JMX、jstatd
15.8.2 JVM 检测工具的作用
  • 对 jvm 运行期间的内部情况进行监控,比如:对 jvm 参数、CPU、内存、堆等信息的查看
  • 辅助进行性能调优
  • 辅助解决应用运行时的一些问题,比如:OutOfMemoryError、内存泄漏、线程死锁、锁争用、Java进程消耗 CPU 过高 等等
15.8.3 jps
  • jps(JVM Process Status Tool):主要用来输出 JVM 中运行的进程状态信息,语法格式如下:jps [options] [hostid]
  • hostid 字符串的语法与 URI 的语法基本一致:[protocol:] [ [ // ] hostname] [ :port ] [/servername],如果不指定hostid,默认为当前主机或服务器
15.8.4 jinfo
  • 打印给定进程或核心或远程调试服务器的配置信息。语法格式:jinfo [option] pid #指定进程号(pid)的进程
  • jinfo [ option ] <executable #指定核心文件
  • jinfo [option] [server-id@] #指定远程调试服务器
15.8.5 jstack
  • jstack 主要用来查看某个 Java 进程内的线程堆栈信息。语法格式如下:jstack [option] pid
  • jstack [option] executable core
  • jstack [option] [server-id@] remorte-hostname-or-ip
15.8.6 jmap
  • jmap 用来查看堆内存使用情况,语法格式如下:jmap [option] pid
  • jmap [option] executable core
  • jmap [option] [server-id@] remote-hostname-or-ip
15.8.7 jstat
  • JVM 统计监测工具,查看各个区域内存和 GC 的情况
  • 语法格式如下:jstat [generalOption | outputOptions vmid [interval[s|ms] [count]]]
15.8.8 jstated
  • 虚拟机的 jstat 守护进行,主要用于监控 JVM 的创建与终止,并提供一个接口,以有序远程监视工具附加到本地系统上运行的 JVM、
  • 语法格式:jstatd [ options ]
15.8.9 jcmd
  • JVM 诊断工具,将诊断命令请求发送到正在运行的额 Java 虚拟机,比如可以用来导出堆,查看 java 进程,导出线程信息,执行 GC 等
15.9 图形化工具
15.9.1 jconsole
  • 一个用于监视 Java 虚拟机的符合 JMX的图形工具。它可以监视本地和远程 JVM,还可以监视和管理应用程序
15.9.2 jmc
  • jmc(JDK Mission Control)Java 任务控制(JMC)客户端包括用于监视和管理 Java 应用程序的工具,而不是引入通常与这些类型的工具相关联的性能开销
15.9.3 VisualVM
  • 一个图形工具,它提供有关在 Java 虚拟机中运行的基于 Java 技术的应用程序的详细信息
  • Java VisualVM 提供内存和 CPU 分析,堆转储分析,内存泄漏检测,访问 MBean 和垃圾回收
15.10 远程连接
  • JMX 连接可以查看:系统信息、CPU使用情况、线程多少、手动执行垃圾回收等比较偏于系统层面的信息
  • jstatd 连接方式可以提供:JVM 内存分布详细信息、垃圾回收分布图、线程详细信息,甚至可以看到某个对象使用内存的大小
    s|ms] [count]]]
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容