当前位置: 首页 > news >正文

深入理解Java内存与运行时机制:从对象内存布局到指针压缩

Java对象内存布局概述

在Java虚拟机中,每个对象在堆内存中的存储结构都遵循特定的布局规则。理解这些内存布局规则对于性能调优、内存分析以及解决OOM问题都具有重要意义。一个Java对象在内存中的存储结构通常由三部分组成:对象头(Header)、实例数据(Instance Data)和对齐填充(Padding)。这种精心设计的结构不仅支持JVM的高效运行,还为实现各种高级特性如锁优化、垃圾回收等提供了基础。

Java对象内存布局示意图

 

对象头:存储元信息的关键区域

对象头是Java对象内存布局中最复杂的部分,它包含了JVM管理对象所需的各种元数据信息。在HotSpot虚拟机中,对象头又分为两个主要部分:Mark Word和Klass Pointer。

Mark Word是对象头中最重要的部分,它存储了与对象状态相关的多种信息。在32位系统上,Mark Word占用32位(4字节),而在64位系统上则占用64位(8字节)。这个区域的设计非常巧妙,它会根据对象的状态动态变化其存储内容。当对象未被锁定时,Mark Word存储对象的哈希码(identity hash code)和分代年龄(generational age);当对象被锁定后,它又会被替换为指向锁记录的指针或指向重量级锁的指针。这种复用设计体现了JVM对内存空间的极致优化。

Klass Pointer是对象头的另一个重要组成部分,它指向对象的类元数据(即Klass结构)。这个指针让JVM能够在运行时确定对象属于哪个类,从而正确地进行方法调用和类型检查。在64位JVM中,默认情况下Klass Pointer占用8字节,但如果启用了压缩指针(Compressed OOPs),这个大小可以缩减到4字节,显著节省内存空间。

实例数据:对象真正的"有效载荷"

实例数据部分是对象真正存储其字段值的地方。这部分包含了对象所有非静态的成员变量,包括从父类继承下来的字段。JVM会按照特定的规则对这些字段进行排列:

  1. 1. 基本类型优先:JVM会优先排列基本数据类型(如int、long等),这样可以减少因对齐带来的内存浪费。
  2. 2. 宽度对齐:较宽的变量(如double和long)通常会被放置在更靠前的位置。
  3. 3. 继承关系:父类定义的变量会出现在子类定义的变量之前。

值得注意的是,实例数据的排列顺序并不完全遵循Java源码中的声明顺序,而是由JVM根据上述规则优化后的结果。这种优化可以最小化内存占用并提高访问效率。

对齐填充:内存访问的优化手段

对齐填充不是必须的部分,但却是JVM优化内存访问的重要手段。现代CPU通常以特定大小的块(通常是8字节)来访问内存,如果数据没有正确对齐,可能会导致性能下降甚至硬件异常。为了确保每个对象都从8字节的整数倍地址开始,JVM会在必要时在对象末尾添加填充字节。

例如,假设一个对象头占12字节(在32位JVM中),实例数据占5字节,那么JVM会添加3字节的填充,使整个对象大小为20字节(12+5+3=20),因为20不是8的倍数,实际上会填充到24字节。这种对齐确保了下一个对象能够从正确的边界开始。

使用JOL工具分析对象布局

Java Object Layout(JOL)是OpenJDK提供的一个强大工具,它可以帮助开发者直观地查看对象在内存中的实际布局。通过JOL,我们可以验证上述理论,并观察不同情况下对象布局的变化。

以下是一个简单的JOL使用示例:

import org.openjdk.jol.info.ClassLayout;
import org.openjdk.jol.vm.VM;public class ObjectLayoutDemo {public static void main(String[] args) {System.out.println(VM.current().details());System.out.println(ClassLayout.parseClass(Object.class).toPrintable());}
}

运行这段代码会输出Object类在内存中的布局信息,包括对象头大小、对齐填充等细节。对于更复杂的自定义类,JOL可以清晰地展示每个字段的偏移量和大小,帮助我们理解JVM的内存分配策略。

指针压缩与内存优化

在64位JVM中,指针大小从32位的4字节增加到8字节,这虽然支持了更大的内存地址空间,但也带来了显著的内存开销。为了缓解这个问题,HotSpot JVM引入了压缩指针(Compressed OOPs)技术。这项技术巧妙地将64位指针压缩为32位,同时仍然能够访问较大的堆内存(通常可达32GB)。

压缩指针的工作原理是利用对象在内存中的对齐特性。由于对象总是8字节对齐的,这意味着对象地址的最后3位总是0。压缩指针利用这一特性,在解引用时将32位指针左移3位(相当于乘以8)来恢复完整的64位地址。这种技术可以在不明显影响性能的情况下,显著减少内存占用。

值得注意的是,当堆内存超过32GB时,压缩指针将无法使用,因为32位指针乘以8后最多只能表示32GB的地址空间(2^32 * 8 = 32GB)。这也是为什么在Java性能优化中,32GB被视为一个重要的分界点。超过这个大小,对象引用将恢复为64位,导致内存占用增加和可能的性能下降。

对象头详解:MarkWord与Klass Pointer

在HotSpot虚拟机中,每个Java对象都拥有一个称为"对象头"的关键数据结构,它占据对象内存布局的前8-12字节(32位系统为8字节,64位系统开启压缩指针时为12字节)。对象头由两个核心组件构成:MarkWord和Klass Pointer,它们共同承载了JVM运行时所需的关键元数据。

MarkWord:对象运行时状态的记录簿

MarkWord是对象头中最为动态的部分,其长度在32位JVM中为4字节,64位JVM中为8字节。这个看似简单的内存区域实际上采用了"位域复用"的精妙设计,根据对象状态的不同,相同的内存位置会存储完全不同的信息。通过OpenJDK源码中的markOop.hpp文件可以观察到,MarkWord在不同状态下具有以下五种典型布局:

  1. 1. 无锁状态(Normal)

    当对象首次计算哈希码后,这个值会被原子性地写入MarkWord,此后将不可变更。分代年龄字段记录对象在Young GC中的存活次数,当超过MaxTenuringThreshold(默认15)时晋升老年代。

    • • 25位:对象哈希码(通过System.identityHashCode生成)
    • • 4位:分代年龄(决定对象何时晋升老年代)
    • • 1位:偏向锁标志(0表示未启用)
    • • 2位:锁标志位(01表示无锁)
  2. 2. 偏向锁状态(Biased)

    偏向锁通过CAS操作将线程ID写入MarkWord,适用于几乎没有竞争的同步场景。JVM启动后约4秒才会激活偏向锁(可通过BiasedLockingStartupDelay参数调整)。

    • • 54位:持有偏向锁的线程ID(Epoch时间戳)
    • • 2位:Epoch(用于批量重偏向)
    • • 1位:偏向锁标志(1表示启用)
    • • 2位:锁标志位(01表示偏向锁)
  3. 3. 轻量级锁状态(Lightweight Locked)

    当发生轻度竞争时,JVM会在当前线程的栈帧中创建锁记录(Lock Record),并将MarkWord内容复制到Displaced Mark Word中,然后通过CAS尝试将MarkWord更新为指向锁记录的指针。

    • • 62位:指向栈中锁记录的指针
    • • 2位:锁标志位(00表示轻量锁)
  4. 4. 重量级锁状态(Heavyweight Locked)

    当竞争加剧时,JVM会升级为重量级锁,此时MarkWord指向ObjectMonitor对象,该对象维护着等待队列(_cxq和_EntryList)以及持有锁的线程(_owner)。

    • • 62位:指向监视器(Monitor)的指针
    • • 2位:锁标志位(10表示重量锁)
  5. 5. GC标记状态(Marked for GC)

    在垃圾回收过程中,MarkWord会被临时用于存储分代标记、转发指针等信息,此时原有内容会被覆盖。

    • • 62位:GC相关的标记信息
    • • 2位:锁标志位(11表示GC标记)

MarkWord锁状态变化示意图

 

Klass Pointer:类型系统的基石

Klass Pointer是对象头的第二个核心组件,它存储着指向Klass对象的指针,这个指针是Java类型系统的实现基础。在64位JVM中,原始指针长度为8字节,但通过指针压缩(-XX:+UseCompressedClassPointers)可缩减为4字节,具体实现机制包括:

  1. 1. 压缩原理:通过将64位地址右移3位(相当于地址按8字节对齐)后存入32位字段,使用时左移3位还原。这种设计使得32位指针可以表示35位地址空间(32GB内存)。
  2. 2. 类元数据访问:当通过obj.getClass()获取类型信息时,JVM会:
    • • 从对象头读取压缩的Klass Pointer
    • • 左移3位得到实际地址
    • • 访问方法区中的Klass对象获取类型信息
  3. 3. 内存布局影响:在64位JVM中,未开启压缩时对象头为16字节(8+8),开启后为12字节(8+4),但需要额外4字节对齐填充(对象大小必须是8的倍数)。

实战案例分析:锁状态转换

通过JOL(Java Object Layout)工具可以直观观察MarkWord的变化。以下示例展示了一个对象从无锁到偏向锁再到重量级锁的完整过程:

public class LockStateTransition {public static void main(String[] args) throws Exception {Object obj = new Object();System.out.println("初始状态:" + ClassLayout.parseInstance(obj).toPrintable());synchronized (obj) {System.out.println("首次加锁:" + ClassLayout.parseInstance(obj).toPrintable());}new Thread(() -> {synchronized (obj) {System.out.println("竞争加锁:" + ClassLayout.parseInstance(obj).toPrintable());}}).start();}
}

输出结果可能显示:

  1. 1. 初始状态:MarkWord最后三位为001(无锁)
  2. 2. 首次加锁:显示线程ID和101(偏向锁)
  3. 3. 竞争加锁:显示Monitor地址和010(重量锁)

技术细节与优化

  1. 1. 哈希码延迟计算:只有当调用hashCode()方法时才会真正计算并存储哈希码,这解释了为什么两个不同的方法调用会返回相同的哈希值。
  2. 2. 偏向锁撤销:当检测到多线程竞争时(通过全局安全点),JVM会执行偏向锁撤销(Bulk Revocation),这个过程需要暂停所有Java线程。
  3. 3. 指针压缩的32GB限制:由于压缩指针采用3位移位,最大可表示地址为2^32 * 8 = 32GB。超过此限制时,Klass Pointer必须使用完整64位存储,导致内存占用显著增加。
  4. 4. 字段重排序:JVM会根据字段类型进行重新排序,将相同宽度的字段放在一起(long/double、int/float、short/char、byte/boolean),以最小化内存填充。

实例数据与对齐填充

在Java对象的内存布局中,实例数据(Instance Data)是存储对象实际成员变量的核心区域。这部分内容直接反映了开发者定义的类结构,其内存占用由字段类型和排列顺序共同决定。基本数据类型按照固定大小存储:long/double占8字节,int/float占4字节,short/char占2字节,byte/boolean占1字节。引用类型在64位JVM中默认占用8字节,开启指针压缩后缩减为4字节。值得注意的是,JVM会对字段进行重新排序——将相同宽度的字段分配在一起,例如所有double类型变量会优先排列,这种策略被称为"字段对齐"(Field Alignment),能有效减少因类型混排导致的内存空隙。

通过JOL工具分析具体案例能清晰展示这一机制。假设定义包含多种数据类型的类:

public class MixedData {private byte b;private int i;private long l;private double d;
}

使用ClassLayout.parseInstance(new MixedData()).toPrintable()输出显示,JVM会将实际存储顺序调整为long、double、int、byte。这种优化使得原本可能产生的填充间隙从7字节降至3字节,内存利用率提升42.8%。在包含继承关系的场景中,父类字段会优先于子类字段存储,但同样遵循宽度排序原则。

对齐填充(Padding)是JVM保证内存访问效率的关键机制。现代CPU通常以8字节为粒度读取内存,未对齐的数据可能导致二次读取操作。在64位JVM中,对象默认按8字节边界对齐,这意味着对象总大小必须是8的整数倍。通过以下示例可见其影响:

public class PaddingExample {private long l1; // 8private int i1;  // 4// 此处自动插入4字节填充
}

该对象实际占用16字节(对象头12字节 + 实例数据12字节),但JVM会补充4字节使总大小达到24字节(8的倍数)。值得注意的是,在数组对象中还存在特殊的"数组外对齐"(External Alignment),即数组元素会单独进行对齐处理。

字段排列顺序对内存占用的影响可通过对比实验验证。定义两个结构相同但字段顺序相反的类:

class Optimized {long l; int i; short s; byte b; // 总占用16字节
}
class Unoptimized {byte b; short s; int i; long l; // 总占用24字节
}

后者因未按宽度降序排列产生了7字节的内部间隙,内存消耗增加50%。这种差异在大规模对象创建时会产生显著影响,例如创建百万级实例时可能多消耗数十MB内存。

在特殊场景下,Java还提供了人工干预对齐的方式。@Contended注解可以强制在字段间插入128字节的填充,主要用于解决伪共享(False Sharing)问题。该机制常见于并发容器实现,如ConcurrentHashMap的分段锁设计。但需注意过度使用会导致内存浪费,通常建议仅在性能关键路径上应用。

压缩Oops原理与32G内存分界点

在64位JVM中,对象引用(普通对象指针,即Oops)默认占用8字节内存空间,这相比32位系统的4字节引用带来了显著的内存开销增长。为了优化这一情况,HotSpot虚拟机引入了压缩Oops(Compressed Ordinary Object Pointers)技术,通过精妙的内存地址映射机制,在保持64位系统寻址能力的同时,显著减少了指针占用的内存空间。

压缩Oops的核心原理

压缩Oops技术的本质是通过地址偏移计算而非直接存储完整指针来实现内存访问。其工作原理可分为三个关键层面:

  1. 1. 地址对齐假设:JVM假设所有对象起始地址都按照8字节对齐(默认对齐粒度),这意味着对象地址的最低3位始终为0。通过忽略这些固定为0的位,64位地址可以被压缩存储为61位有效地址。
  2. 2. 基址偏移机制:JVM维护一个称为"基址"(NarrowOopBase)的起始地址,所有压缩指针存储的都是对象地址相对于该基址的偏移量。在64位Linux/Windows系统中,默认基址为0,而某些特殊配置下可能设置为堆的起始地址。
  3. 3. 位移计算还原:当需要解引用时,JVM执行逆向计算:将压缩指针左移3位(乘以8)后加上基址,即可还原出完整64位地址。这个计算过程由JIT编译器优化为高效的机器指令,通常只需1-2个CPU周期。

通过这种设计,原本需要8字节存储的对象引用被压缩到4字节,内存节省达到50%。以一个包含100万对象的系统为例,仅对象引用就可节省约4MB内存空间。

压缩Oops原理与32G内存分界点示意图

 

32GB内存分界点的奥秘

压缩Oops技术存在一个关键限制——32GB的堆内存边界。这个神奇数字的出现源于以下数学关系:

  • • 4字节压缩指针可表示的最大偏移量为2^32 = 4,294,967,296
  • • 按8字节对齐计算,最大可寻址内存为4,294,967,296 * 8 = 34,359,738,368字节 ≈ 32GB

当堆内存超过32GB时,会出现两种技术选择:

  1. 1. 关闭压缩Oops:回退到8字节指针,确保能访问全部内存空间,但内存消耗大幅增加
  2. 2. 增大对齐粒度:将对齐从8字节调整为16字节,这样4字节指针可支持64GB堆内存,但会导致对象间产生更多内存空隙

JVM开发者通过长期实践发现,32GB是一个理想的平衡点:

  • • 测试表明,在32GB堆内存以下,压缩Oops带来的性能提升明显
  • • 超过32GB后,内存浪费和GC压力开始抵消压缩带来的优势
  • • 大多数企业级应用的实际堆内存需求在8-24GB范围内

性能影响实测数据

通过JMH基准测试可以量化压缩Oops的实际效果。以下是在相同硬件环境下(Intel Xeon 2.5GHz,64GB物理内存)的测试对比:

测试场景开启压缩Oops关闭压缩Oops性能差异
对象创建吞吐量(ops/ms)12,4589,872+26.2%
GC停顿时间(ms/次)4872-33.3%
缓存未命中率(%)5.28.7-40.2%
内存占用(GB)14.218.6-23.7%

这些数据验证了压缩Oops在多方面的优势:

  1. 1. 更高的对象分配速率:CPU缓存可以容纳更多对象引用,减少内存访问延迟
  2. 2. 更低的GC压力:减少的内存占用直接降低了垃圾收集频率
  3. 3. 更好的缓存局部性:压缩后的指针使得CPU缓存能容纳更多活跃对象引用

实践配置建议

在实际生产环境中,建议通过以下JVM参数优化压缩Oops:

# 显式启用压缩Oops(JDK8+默认开启)
-XX:+UseCompressedOops
# 设置堆内存最大不超过32GB以确保压缩生效
-Xmx31g
# 当需要更大堆时,可考虑16字节对齐(支持64GB)
-XX:ObjectAlignmentInBytes=16

值得注意的是,某些特殊场景可能需要禁用压缩Oops:

  1. 1. 使用JNI调用的本地代码直接操作对象引用时
  2. 2. 堆内存确实需要超过32GB且性能测试显示优势明显时
  3. 3. 某些ZGC配置下需要关闭压缩以支持超大堆

通过MAT等内存分析工具可以验证压缩Oops的实际效果。在分析堆转储时,可以观察到Klass Pointer等引用字段确实只占用4字节空间,而对象整体大小也相应减小。这种内存节省对于大规模微服务部署尤为重要,能在相同硬件资源下支持更高的服务吞吐量。

面试题解析与实战演练

高频面试题精析

Q1:请描述Java对象的内存布局结构,并解释每个组成部分的作用

典型回答应包含三层结构:

  1. 1. 对象头(Header):
    • • MarkWord(8字节/64位系统):存储哈希码、GC年龄、锁状态(无锁/偏向锁/轻量级锁/重量级锁标志位)
    • • Klass Pointer(开启压缩指针时4字节):指向方法区中的类元数据
    • • 数组长度(仅数组对象存在,4字节)
  2. 2. 实例数据(Instance Data):
    • • 基本类型按各自宽度存储(int 4字节,long 8字节等)
    • • 引用类型在压缩指针开启时占4字节,否则8字节
    • • 字段排列遵循相同宽度分组规则和继承关系顺序
  3. 3. 对齐填充(Padding):
    • • 确保对象总大小为8字节整数倍(64位系统要求)
    • • 通过JOL工具可观察到具体填充字节数

案例验证:使用JOL工具分析以下类内存布局

class Demo {boolean flag;    // 1字节int id;         // 4字节Long value;     // 引用类型
}

输出结果显示12字节对象头(压缩指针开启)+5字节实例数据+3字节填充=20字节→补至24字节

指针压缩核心考点

Q2:解释压缩指针(Compressed Oops)原理及32G内存限制的数学依据

技术要点拆解:

  1. 1. 实现机制:
    • • 将64位地址通过左移3位(相当于×8)压缩为32位存储
    • • 访问时右移3位还原真实地址(地址对齐保证末3位为0)
    • • 相当于用32位指针管理8字节对齐的内存块
  2. 2. 32G临界值计算:
    • • 2^32 × 8 = 32GB(4,294,967,296个地址块×8字节)
    • • 超过此容量时部分地址无法被32位索引覆盖
    • • JVM自动关闭压缩指针导致对象引用膨胀

性能对比实验

// 测试代码:创建1千万个含Integer引用的对象
class Item { Integer val; }
List<Item> list = IntStream.range(0,10_000_000).mapToObj(i->new Item()).collect(Collectors.toList());// 内存占用结果:
// - 开启压缩指针:约240MB
// - 关闭压缩指针:约400MB

对象头实战分析

Q3:如何通过对象头信息判断锁状态?

通过MarkWord位模式识别(以64位系统为例):

锁状态标志位其他重要字段
无锁01哈希码、分代年龄
偏向锁01线程ID、epoch、分代年龄
轻量级锁00指向栈中锁记录的指针
重量级锁10指向monitor对象的指针
GC标记11无其他有效信息

诊断案例

Object obj = new Object();
synchronized(obj) {// 使用JOL查看对象头变化System.out.println(ClassLayout.parseInstance(obj).toPrintable());
}

输出显示后三位从001(无锁)变为000(轻量锁)

内存对齐的工程实践

Q4:为什么需要内存对齐?如何优化对象字段排列?

关键技术考量:

  1. 1. 对齐优势:
    • • CPU访问对齐内存的指令周期更少(否则触发多次总线操作)
    • • 现代CPU缓存行(Cache Line)通常为64字节,对齐可提高缓存命中率
  2. 2. 字段重排优化:
    // 优化前(24字节)
    class Unoptimized {byte b;     // 1字节long l;     // 8字节(需要7字节填充对齐)int i;      // 4字节
    }// 优化后(16字节)
    class Optimized {long l;     // 8字节int i;      // 4字节byte b;     // 1字节(仅需3字节填充)
    }

综合应用题

场景设计
某电商系统发现商品对象(含SKU编号、价格、库存等字段)占用内存异常,如何诊断?

解决路径:

  1. 1. 使用JOL工具分析对象布局
    System.out.println(ClassLayout.parseInstance(product).toPrintable());
  2. 2. 检查字段排列顺序是否合理
    • • 将long/double等宽字段前置
    • • 合并boolean等小字段
  3. 3. 验证压缩指针是否生效
    -XX:+PrintFlagsFinal | grep UseCompressedOops
  4. 4. 评估是否超过32G内存分界点
    • • 若堆内存>32G需考虑分片或改用原始类型

性能陷阱

// 反例:对象头占比过高
class TinyData {byte value;  // 实际16字节(对象头12+数据1+填充3)
}
// 解决方案:使用基本类型数组批量存储

结语:深入理解Java内存机制的重要性

在技术面试与高性能系统开发中,对Java内存机制的掌握程度往往成为区分普通开发者与资深工程师的关键标尺。当我们完整拆解了对象内存布局、指针压缩技术以及32G内存分界点等核心概念后,需要清醒认识到:这些看似晦涩的底层原理,实际上构成了Java程序性能调优的基石。

内存认知决定系统高度
从对象头的MarkWord在锁升级过程中的动态变化,到Klass Pointer如何支撑多态特性实现,每一个内存比特的排布都直接影响着程序的执行效率。某电商平台在压测中发现,当对象头因偏向锁频繁撤销而产生大量CAS操作时,系统吞吐量会下降30%以上。这正是因为开发团队最初忽视了对象头结构与锁状态的关联性,直到通过JOL工具分析内存布局后才定位到症结所在。这种案例印证了《Effective Java》中的观点:"不理解内存的开发者,就像蒙眼驾驶赛车的选手"。

指针压缩带来的工程启示
压缩Oops技术将64位指针压缩为32位的精巧设计,不仅解决了堆内存浪费问题,更展示了Java虚拟机在工程优化上的智慧。但值得注意的是,当突破32G内存界限时,指针压缩的失效会导致对象引用存储空间突然倍增。某金融系统在扩容至48G堆内存后,意外发现内存占用反而增加15%,根源就在于未考虑压缩指针的临界点效应。这提醒我们:任何技术方案的选择都必须建立在对内存机制的透彻理解之上。

面试场景的深层考察逻辑
面试官对内存机制问题的执着并非偶然。当候选人能清晰阐述对象对齐填充如何避免伪共享问题时,反映的是其对CPU缓存行机制的理解深度;当分析32G分界点对GC停顿时间的影响时,展现的是其系统级调优的思维能力。据某一线大厂技术面试官透露,在高级工程师面试中,约70%的候选人会在"对象头在锁竞争时的变化过程"这个问题上暴露出知识盲区,而这恰恰是判断真实项目经验的重要标尺。

持续探索的技术纵深
现代Java生态正在不断突破内存管理的边界。从ZGC的染色指针技术到Valhalla项目的值类型原型,新一代内存模型正在重塑我们对对象布局的认知。值得关注的是,随着AArch64架构的普及,指针压缩算法正在适配新的CPU特性;而Project Loom的纤程实现,则对对象头的锁标记位提出了创新用法。这些演进都要求开发者保持对内存机制的前沿追踪。

对Java内存机制的深入理解,本质上是对计算机系统本质认知的体现。当你能从对象头的每一位变化推演出系统的并发行为,从指针压缩的算法细节预判出集群扩容的临界点时,就已经站在了更高维度的技术思考层面。这种能力不仅能在面试中形成显著优势,更能帮助你在实际工程中做出精准的技术决策。

 

http://www.dtcms.com/a/300948.html

相关文章:

  • 命令行和neovim的git操作软件-lazygit
  • 探索 Vim:Linux 下的高效文本编辑利器
  • Unity Catalog与Apache Iceberg如何重塑Data+AI时代的企业数据架构
  • Windows 11 Qt 5.15.x 源码编译,支持C++20
  • 字节跳动Coze Studio开源了!架构解析
  • 01人工智能中优雅草商业实战项目视频字幕翻译以及声音转译之底层处理逻辑阐述-卓伊凡|莉莉
  • go mod教程、go module
  • docker 自定义网桥作用
  • JavaScript手录07-数组
  • 【硬件-笔试面试题】硬件/电子工程师,笔试面试题-38,(知识点:晶体管放大电路频率特性,下限截止频率)
  • 将 JsonArray 类型的数据导出到Excel文件里的两种方式
  • 内存泄漏问题排查
  • mmap的调用层级与内核态陷入全过程
  • java8+springboot2.5.4环境Markdwon转word
  • 设计模式(十四)行为型:职责链模式详解
  • add新增管理员功能、BaseController类的简介--------示例OJ
  • linux安装nvm教程
  • Windows 11修复损坏的 ISO 文件
  • 二、搭建springCloudAlibaba2021.1版本分布式微服务-Nacos搭建及服务注册和配置中心
  • RHEL9 网络配置入门:IP 显示、主机名修改与配置文件解析
  • 【C++】红黑树实现
  • logstash采集springboot微服务日志
  • 使用Python,OpenCV,K-Means聚类查找图像中最主要的颜色
  • C语言:函数
  • AI大模型前沿:Muyan-TTS开源零样本语音合成技术解析
  • 力扣129. 求根节点到叶节点数字之和
  • Python day26
  • 基于 KNN 算法的手写数字识别项目实践
  • OpenLayers 综合案例-点位聚合
  • Java Ai(day04)