0

0

什么是Java中的字节码校验器_JVM如何保证加载的类文件是安全的

P粉602998670

P粉602998670

发布时间:2026-02-17 01:21:14

|

146人浏览过

|

来源于php中文网

原创

字节码校验器在类加载的链接阶段中的验证子阶段工作,即classloader调用defineclass()后、resolveclass()前触发;它静态扫描字节码,确保类型匹配、栈平衡与控制流可达,失败抛出verifyerror。

什么是java中的字节码校验器_jvm如何保证加载的类文件是安全的

字节码校验器在类加载的哪个阶段工作

它不是独立运行的工具,而是 ClassLoaderdefineClass() 后、resolveClass() 前触发的一道强制检查。JVM 不会跳过它——哪怕你用自定义类加载器,只要走标准流程,verify() 就会自动执行。

常见错误现象:java.lang.VerifyError 通常出现在类第一次被主动使用(比如调用静态方法、实例化)时,而不是 Class.forName() 那一刻。这是因为校验是“懒触发”的,但一旦失败,堆栈里会明确标出 VerifyError 和出问题的字节码偏移量。

  • 如果你用 ASM 或 Javassist 动态生成类,没做栈映射表(StackMapTable)更新,大概率在这里翻车
  • Java 7+ 默认启用更严格的校验(-XX:+UseSplitVerifier 已废弃),旧版生成的 class 文件在新版 JVM 上可能直接拒载
  • 校验不检查业务逻辑安全(比如是否删库),只管“结构上能不能不崩溃”:类型匹配、操作数栈平衡、控制流是否可达

校验器实际检查哪些字节码规则

它不跑代码,只静态扫描指令流,核心目标是防止 JVM 运行时状态错乱。重点盯三类问题:

  • aload_0 后跟 invokevirtual,但栈顶不是对象引用?→ 拒绝
  • 方法返回 int 却在末尾写 areturn?→ 拒绝
  • 跳转指令(如 goto)指向非法偏移,或跳进某个指令中间?→ 拒绝

性能影响很小:校验发生在类首次解析,且只做一次。但如果你频繁 defineClass() 大量动态类(比如某些 RPC 框架),校验开销会叠加。可临时关掉(-Xverify:none),但仅限可信环境——关了等于把门锁焊死再拆掉钥匙孔。

立即学习Java免费学习笔记(深入)”;

为什么有些非法 class 文件能绕过校验

校验器有两套模式:对 bootstrap classloader 加载的类(如 java.lang.*)用“苛刻模式”,对应用类用“宽松模式”。后者允许部分不严谨但实际不会出错的字节码(比如未初始化就读取局部变量),前提是不破坏类型安全。

Eva Design System
Eva Design System

基于深度学习的色彩生成器

下载

容易踩的坑:

  • javac -target 8 编译,却在 Java 17 JVM 上运行——校验器版本升级后可能拒绝旧规范的 class 文件
  • 混淆工具(如 ProGuard)若没配好 -adaptclassstrings,可能改坏常量池引用,导致 VerifyError: Expecting a stackmap frame
  • 反射调用 Unsafe.defineAnonymousClass() 可跳过校验,但这不是正经路子,JDK 16+ 已标记为 deprecated

如何调试 VerifyError 的具体原因

光看错误信息不够,得看到底哪条指令坏了。用 javap -v 是最快路径:

javap -v MyClass.class | grep -A 20 "public static void main"

重点关注输出里的 StackMapTable 属性和每条指令的栈帧变化。如果报 Expecting a stackmap frame at branch target,说明跳转目标处缺少栈映射;如果是 Illegal type at constant pool index,就得查常量池索引是否越界或类型错位。

真正麻烦的是:校验器不告诉你“你少写了一条 iconst_0”,只说“栈不平衡”。所以动态生成字节码时,别靠手算栈高,老实用 ASM 的 COMPUTE_FRAMES 模式让框架自动补全。

复杂点在于,校验逻辑随 JDK 版本演进过多次,Java 6 的“旧校验器”和 Java 9 的“新校验器”对同一段字节码可能给出不同结论。别假设“以前能跑现在就一定行”。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1553

2023.10.24

go语言goto的用法
go语言goto的用法

本专题整合了go语言goto的用法,阅读专题下面的文章了解更多详细内容。

138

2025.09.05

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

750

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

571

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

234

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

209

2025.08.29

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

418

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

592

2023.08.10

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

283

2026.02.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 3.6万人学习

C# 教程
C# 教程

共94课时 | 9.6万人学习

Java 教程
Java 教程

共578课时 | 66.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号