news 2026/4/18 10:55:52

类加载器分析(四)启动类加载器 之 字段

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
类加载器分析(四)启动类加载器 之 字段

类加载器分析(四)启动类加载器 之 字段

字段

一个类会有多个字段,字段的内容如下

比如String类的字段可以通过下述Java程序输出

public static void main(String[] args){ Class<?> stringClass = String.class; System.out.println("java.lang.String 字段信息:"); System.out.println("=========================="); // 获取所有声明的字段(包括private) Field[] fields = stringClass.getDeclaredFields(); System.out.println("字段总数: " + fields.length); System.out.println(); // 详细列出每个字段 for (int i = 0; i < fields.length; i++) { Field field = fields[i]; System.out.println((i + 1) + ". " + field.getName()); System.out.println(" 类型: " + field.getType().getSimpleName()); System.out.println(" 修饰符: " + field.getModifiers()); System.out.println(" 是否是合成字段: " + field.isSynthetic()); System.out.println(); } // 按类别统计 long privateFields = Arrays.stream(fields) .filter(f -> java.lang.reflect.Modifier.isPrivate(f.getModifiers())) .count(); long finalFields = Arrays.stream(fields) .filter(f -> java.lang.reflect.Modifier.isFinal(f.getModifiers())) .count(); System.out.println("统计信息:"); System.out.println("- Private字段: " + privateFields); System.out.println("- Final字段: " + finalFields); }

输出结果如下

java.l ang.String 字段信息:==========================字段总数:111. value 类型: byte[]修饰符:18是否是合成字段:false2. coder 类型: byte 修饰符:18是否是合成字段:false3.hash类型: int 修饰符:2是否是合成字段:false4. hashIsZero 类型: boolean 修饰符:2是否是合成字段:false5. serialVersionUID 类型: long 修饰符:26是否是合成字段:false6. COMPACT_STRINGS 类型: boolean 修饰符:24是否是合成字段:false7. serialPersistentFields 类型: ObjectStreamField[]修饰符:26是否是合成字段:false8. REPL 类型: char 修饰符:26是否是合成字段:false9. CASE_INSENSITIVE_ORDER 类型: Comparator 修饰符:25是否是合成字段:false10. LATIN1 类型: byte 修饰符:24是否是合成字段:false11. UTF16 类型: byte 修饰符:24是否是合成字段:false统计信息: - Private字段:7- Final字段:9

修饰符就是访问权限

3.4.1 字段数量

InstanceKlass中用_fields成员变量描述Java类的字段

class InstanceKlass: public Klass { ... u2 _java_fields_count; // The number of declared Java fields Array<u2>* _fields;

获取和设置字段方法如下

Array<u2>* fields() const { return _fields; }//获取字段集合 void set_fields(Array<u2>* f, u2 java_fields_count) {//设置字段集合 guarantee(_fields == NULL || f == NULL, "Just checking"); _fields = f; _java_fields_count = java_fields_count; }
3.4.2 字段获取时机

设置字段时机是在解析字节码文件完成后设置,gdb调试如下

#0 InstanceKlass::set_fields (this=0x100041ca8, f=0x7fffe88057b8, java_fields_count=11)at /home/jx/src/openjdk/src/hotspot/share/oops/instanceKlass.hpp:451#1 0x00007ffff61a9f38 in ClassFileParser::apply_parsed_class_metadata (this=0x7ffff59fddd0,this_klass=0x100041ca8,java_fields_count=11)at /home/jx/src/openjdk/src/hotspot/share/classfile/classFileParser.cpp:3945#2 0x00007ffff61ae843 in ClassFileParser::fill_instance_klass (this=0x7ffff59fddd0, ik=0x100041ca8,changed_by_loadhook=false,cl_inst_info=...,__the_thread__=0x7ffff0028830)at /home/jx/src/openjdk/src/hotspot/share/classfile/classFileParser.cpp:5272#3 0x00007ffff61ae3da in ClassFileParser::create_instance_klass (this=0x7ffff59fddd0, ###在这里可以看到是创建类,步进可见class_parserchanged_by_loadhook=false,cl_inst_info=...,__the_thread__=0x7ffff0028830)at /home/jx/src/openjdk/src/hotspot/share/classfile/classFileParser.cpp:5228

在栈的第3帧的实现可以看到

//在这一步对字节码文件进行解析,获取对应的字段集合和字段总数 ClassFileParser parser(stream, name, loader_data, &cl_info, ClassFileParser::BROADCAST, // publicity level CHECK_NULL); //解析之后将字段集合和字段总数赋值给ClassFileParser类的下面成员变量 //Array<u2>* _fields; //u2 _java_fields_count; ... //下面的函数会将ClassFileParser的_fields和_java_fields_count赋值给InstanceKlass的_fields和_java_fields_count InstanceKlass* result = parser.create_instance_klass(old_stream != stream, *cl_inst_info, CHECK_NULL);

实际上,真正实现字段获取时在字节码解析阶段,通过gdbClassFileParser类的构造函数调试可发现

ClassFileParser::ClassFileParser(ClassFileStream* stream,
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/18 7:42:41

中小企业设备管理系统信息管理系统源码-SpringBoot后端+Vue前端+MySQL【可直接运行】

摘要 随着信息技术的快速发展&#xff0c;中小企业在设备管理方面面临诸多挑战&#xff0c;传统的手工记录和纸质管理方式效率低下且容易出错。设备管理系统的信息化成为提升企业运营效率的关键手段。通过构建一套高效的设备管理系统&#xff0c;企业可以实现设备的全生命周期管…

作者头像 李华
网站建设 2026/4/18 3:32:28

Nginx 配置参数化实践:通过 Docker Run 传递参数实现动态代理配置

概述 在微服务架构中&#xff0c;前端应用通常需要代理多个后端服务的 API 请求。传统的做法是为每个环境&#xff08;开发、测试、生产&#xff09;构建不同的镜像&#xff0c;这会导致镜像管理复杂、部署效率低下。本文将介绍如何通过 Nginx 配置参数化&#xff0c;结合 Doc…

作者头像 李华
网站建设 2026/4/18 3:38:36

Python+Vue的卫生室药店信息管理系 django Pycharm flask

收藏关注不迷路&#xff01;&#xff01;需要的小伙伴可以发链接或者截图给我 项目介绍 随着时代的发展&#xff0c;我国的医疗事业也取得了非常大的成就&#xff0c;很多大型的医院都哦以及实现了现代医疗信息的管理&#xff0c;但是很多卫士室扔采用人工手动的方式对病人和医…

作者头像 李华