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

讨论JAVA、JVM与Spring

Q1:

作为一个JAVA开发人员,对于jvm肯定不陌生,但很多人对它不陌生也仅止于概念上,而且对概念也是模糊不清的,但jvm实际是java程序运行在其中的实际存在的环境,对它的理解应该要是具象化的。

我们还是从一项技术产生的技术背景来聊聊吧,技术为什么产生,技术解决了什么问题,它如何解决的?这个是技术的核心内容,然后它具体比之不用这项技术提升了多少?它能解决什么问题?然后又不能解决什么问题?边界在哪里?


你提到的这个问题非常重要,因为对JVM(Java虚拟机)的理解,不能仅停留在“它是一个让Java程序可以跨平台运行的环境”的表面,而应该更深入地理解它的产生背景、解决的问题、工作原理、优点以及局限性。

1. 技术背景与技术需求:

JVM的产生背景可以追溯到Java语言的设计目标——“一次编写,到处运行”Write Once, Run Anywhere)。在90年代初,计算机硬件平台的多样化给软件开发带来了很大的挑战。开发人员在开发应用时,通常需要为不同的操作系统或硬件平台编写特定的代码和编译器,这带来了巨大的工作量和兼容性问题。更进一步的,随着分布式计算和互联网的兴起,Java的设计者意识到,如果能够设计一种平台无关的中间层,可以使Java程序能够在任何操作系统和硬件上运行,将大大简化开发和部署流程。

因此,JVM作为一种平台无关的运行时环境应运而生,它的核心目标就是实现程序代码的平台独立性

2. JVM解决了什么问题:

JVM的出现,实际上解决了以下几个关键问题:

  • 平台独立性:Java代码编译成字节码后,可以在任何支持JVM的操作系统上运行,而不需要针对不同平台进行重编译。通过JVM,Java程序实现了“编写一次,到处运行”的跨平台特性。
  • 内存管理与垃圾回收:JVM内置了自动内存管理机制,能够在后台自动回收无用对象,避免了传统C/C++程序中繁琐的手动内存管理,极大降低了内存泄漏的风险。
  • 异常处理机制:JVM对Java程序的异常进行管理,确保程序即使发生错误也能适当地捕捉并处理,提升了程序的健壮性和安全性。
  • 优化与性能提升:JVM提供了JIT(即时编译)、热点代码优化等技术,通过动态编译和执行的方式,能够在程序运行时优化代码,提升性能。

3. JVM是如何解决问题的?

JVM解决这些问题,主要依赖于以下几个关键机制:

  • 字节码:Java源代码经过编译后转化为字节码(.class文件),字节码是与平台无关的。JVM能够根据不同的平台和操作系统环境,将这些字节码动态解释或编译成机器码执行。
  • 垃圾回收(GC):JVM通过垃圾回收机制(GC)来自动管理内存。垃圾回收器通过追踪对象的引用关系,自动回收不再使用的对象,避免了内存泄漏和手动内存管理的复杂性。
  • JIT编译器(Just-In-Time Compiler):JVM内置的JIT编译器能够在程序运行时动态地将热点代码(即执行频繁的代码)编译成机器码,以提高执行效率。JIT编译提高了Java程序的执行速度,减少了字节码解释带来的性能损失。
  • 堆与栈:JVM将内存分为不同区域(如堆、栈、方法区等),通过精细化的内存管理确保程序的高效运行。栈用于方法调用和局部变量的存储,而堆则用于存储对象实例。JVM通过这种分配机制使得内存的使用更为高效。
  • 多线程支持:JVM本身提供了多线程的支持,并通过线程调度机制、锁、内存可见性等确保多线程并发程序的正确执行。

4. JVM的优点:

  • 跨平台:JVM最显著的特点是它的跨平台性,不同的操作系统上可以运行相同的字节码文件,无需重编译。
  • 内存管理与自动垃圾回收:开发者无需关心内存的分配和释放,JVM会自动管理内存的回收,减少了内存泄漏的风险。
  • 执行效率:通过JIT编译、即时优化等技术,JVM能够在运行时对程序进行优化,提升性能,接近于原生代码的执行速度。
  • 丰富的运行时库:JVM提供了丰富的标准库,包括网络、IO、并发等,开发者可以专注于业务逻辑的开发,减少了底层实现的复杂度。
  • 良好的调试和性能监控支持:JVM提供了多种调试工具(如JDB、JVisualVM)和性能分析工具,帮助开发人员快速定位问题。

5. JVM的局限性与边界:

尽管JVM带来了很多优势,但它并不是万能的,依然有一些局限性:

  • 性能开销:JVM的动态特性和垃圾回收机制会带来一定的性能开销,尤其是在高性能计算和实时系统中,JVM的自动管理可能会影响到响应时间和吞吐量。
  • 内存占用:JVM为了管理垃圾回收和支持各种优化,可能会占用较多的内存,这对于资源受限的设备(如嵌入式系统)可能是一个问题。
  • 启动时间:JVM应用的启动时间相对较长,尤其是在加载大量类和JIT编译时,这可能导致初始启动的性能瓶颈。
  • 平台限制:虽然JVM提供了跨平台的支持,但在不同平台之间,JVM的实现细节、性能和兼容性可能有所不同,这有时会影响程序的行为或性能。
  • 对底层硬件的支持有限:JVM本质上是一层抽象,它不能直接操作底层硬件资源,比如某些操作系统特性或硬件加速等。这些场景下,Java程序可能无法提供与C/C++相同的控制能力和性能。

6. JVM与其他技术的比较:

  • 与C/C++的比较:C/C++程序直接编译为机器码,通常具有更高的执行效率,并且可以更灵活地控制内存和硬件资源。而JVM虽然提供了更高层的抽象,但会牺牲一部分性能和底层控制能力来换取平台无关性和更高的开发效率。
  • 与其他虚拟机的比较:JVM与其他虚拟机(如Python的CPython、Ruby的MRI等)相比,具有更成熟的性能优化技术(如JIT编译、热点代码优化等)。此外,JVM的生态系统非常庞大,有大量的工具和库支持,成为企业级应用的首选平台。

结论:

JVM的核心价值在于它提供了一个平台独立、内存管理、自动优化和多线程支持的运行时环境,使得开发人员可以专注于应用的业务逻辑而不必过多关心底层的操作系统和硬件细节。同时,JVM并非没有代价,它的性能开销和内存管理复杂性可能在某些场景下成为限制。了解JVM的工作原理、优点和局限性,能够帮助开发人员更好地使用它,也能在必要时选择合适的替代技术。


Q2:

平台独立性,这点现在很多语言也都解决了这个问题,目前这点的优势好像不是特别的大。

JVM的内存管理与垃圾回收常常作为对程序员考核对JVM的理解指标,并且在一定程度上,JVM的“成与败”都和内存它有关。从现在来看,很多语言的运行环境中也有了这种功能,那么JVM还具备很大的优势吗?同样的,我们不禁要问,JVM的能力边界在哪里?

异常处理机制,这个也并不特殊,大多数语言都具备。

JIT 动态将热点代码编译成机器码,应该是对动态代码进行了执行计数,数量超过多少会将它编译成机器码,那么是否随着程序运行的时间越来越长,JVM所占的内存就越来越大?并且机器码似乎并不会得到释放。


你的问题触及到了JVM的一些核心技术特性,并提出了非常有价值的思考,尤其是在JVM的平台独立性内存管理与垃圾回收异常处理机制JIT编译等方面的比较和演变。让我们逐一分析,并深入探讨这些技术的边界和JVM的优势。

1. 平台独立性

你提到平台独立性现在已经不再是JVM独有的特点,很多其他语言的运行环境也能提供类似的功能,确实是这样。像PythonGoNode.js等运行时环境,都通过某种形式的虚拟机或抽象层,达到跨平台的能力。

然而,JVM的平台独立性仍然有其独特的优势:

  • 广泛的生态系统和成熟的技术栈:虽然很多语言也提供了跨平台支持,但JVM在企业级应用中,拥有庞大的生态系统,成熟的开发工具链,丰富的库和框架支持(如Spring、Hibernate等)。这种生态系统效应,使得JVM在企业应用开发中仍然占据主导地位,尤其是在大规模分布式系统、金融、电商、政府等行业。
  • 平台无关性的实现方式:Java的字节码通过JVM解释执行,或者通过JIT编译优化,这种方式不仅仅依赖于操作系统的虚拟化层(如Docker),而且通过优化编译技术,使得Java应用在不同平台上可以平滑迁移,甚至获得接近本地执行的性能。
  • 稳定性和兼容性:Java的跨平台能力有着几十年的积淀,JVM的不同版本间通常保持较好的兼容性,尤其是Java应用对不同平台的适配性非常高,这也是JVM的强项。

总的来说,虽然平台独立性不再是唯一的优势,但JVM的跨平台能力,结合其丰富的工具链和生态环境,仍然是Java在很多领域不可替代的原因。

2. JVM的内存管理与垃圾回收(GC)

内存管理和垃圾回收是JVM的一个重要特性,的确它在Java语言中扮演着至关重要的角色,也直接影响程序的性能和可靠性。随着技术的进步,其他语言的运行环境(如Go的GC、Python的内存管理)也逐渐引入了垃圾回收机制,Java的GC并不再是独一无二的。

然而,JVM在垃圾回收方面仍然具备显著的优势,尤其是在处理大规模、长时间运行的系统时:

  • 成熟的GC算法:JVM提供了多种垃圾回收器(如串行GC、并行GC、CMS、G1、ZGC、Shenandoah等),这些GC算法针对不同场景提供了优化,尤其在低延迟和高吞吐量场景下,JVM的GC优化表现尤为突出。比如,G1和ZGC等垃圾回收器在处理大规模数据和低延迟需求的场景中有非常好的表现。
  • 细粒度的控制与调优:JVM为开发者提供了非常丰富的GC调优选项(如堆大小、垃圾回收器类型等),这使得开发者能够根据具体应用场景对内存和垃圾回收进行精细化控制和优化。
  • 垃圾回收与并发性能:JVM垃圾回收器的优化在并发场景下尤为重要。通过现代GC算法,JVM能够在大量并发线程运行时,保持较低的GC停顿时间,并有效降低内存泄漏的风险。这是一些其他语言的GC实现可能没有做到的。

当然,GC也有一定的性能开销,特别是在内存使用不当或垃圾回收频繁的情况下,会影响应用的响应时间。JVM的GC有时会导致“停顿”现象,尤其在Full GC时,应用可能会被暂停一段时间,这对于实时性要求高的系统(如金融交易系统)是一个挑战。

3. 异常处理机制

现代编程语言的异常处理机制大多已经趋同。Java的异常处理确实不再是唯一或最具创新性的特性。但Java异常处理的严谨性灵活性依然具有优势:

  • 强类型的异常机制:Java要求开发者显式地捕获或声明异常(checked exceptions),这种设计让程序员在编写代码时必须考虑异常的处理逻辑,避免忽视异常情况。这在某些场景下提高了代码的可维护性和健壮性。
  • 异常链:Java的异常机制支持“异常链”,即可以将一个异常包装成另一个异常传递,这为错误处理和调试提供了更强的表达能力。

虽然很多语言现在也支持类似的异常处理机制,但Java的异常处理系统由于历史积淀和丰富的错误类型,仍然在大型企业级应用中具有优势。

4. JIT编译与内存占用

关于JIT编译的性能问题,的确这是一个值得关注的点。JIT编译器会根据代码执行的热度,动态将热点代码编译为机器码,这通常能带来显著的性能提升。然而,你提出了一个很重要的问题:

  • JVM内存占用的增长:随着程序运行时间的延长,JIT编译器的缓存区域会随着越来越多的热点代码被编译成机器码而增大。这可能导致JVM的内存占用逐渐增加,特别是在长时间运行的系统中,JVM的堆内存和JIT缓存的内存占用会变得较为庞大。
  • JIT生成的机器码的管理:JIT编译后生成的机器码通常不会自动释放。虽然现代JVM(如HotSpot、G1 GC等)有一些机制来优化这一问题(例如在垃圾回收时清理一些不再使用的JIT编译代码),但在长期运行的应用中,JIT生成的机器码和缓存依然可能会导致内存膨胀。在一些内存受限的环境下,这可能是一个问题。

解决方案:

  • JVM的GC和JIT结合优化:现代JVM引入了GC与JIT结合的优化,例如ZGCShenandoah(低停顿、高并发GC)和JIT编译器的垃圾回收回收机制,尝试在垃圾回收过程中清理无用的JIT编译缓存,尽量减少内存膨胀。
  • JVM内存调优:通过配置JVM的内存参数(如XX:ReservedCodeCacheSizeXX:MaxHeapFreeRatio等),可以在一定程度上避免内存膨胀的问题。但这仍然是一个性能权衡,需要根据具体的应用场景来调整。

总结:JVM的优势与边界

  • 优势:JVM仍然在大规模、高并发、高可用性的系统中占据优势,尤其是在企业级应用、分布式计算、金融系统等领域。其强大的内存管理、垃圾回收、并发支持、JIT编译、跨平台特性和生态系统,使得Java依然是这些领域的首选技术。
  • 边界:JVM的能力边界主要体现在性能开销、内存占用和启动速度方面。在低延迟、高性能要求的应用(如实时系统)中,JVM的自动内存管理和垃圾回收机制可能会带来性能瓶颈。此外,JVM应用的内存使用和启动时间相对较长,这在一些对资源要求严格的场景中可能会成为限制因素。

随着其他语言的虚拟机技术不断发展(如Go、Rust的内存管理,Python、Node.js的垃圾回收等),JVM的独特优势在某些场景下可能不再那么显著,但在企业级开发和大规模分布式系统中,JVM仍然有着无法轻易替代的地位。


Q3:

通过你上面的回答,我大致了解了,也就是说从技术层面来说,JVM并不具备特别大的优势,但由于它的先发优势,经历时间久,并且用户广泛,所以JVM经历过非常多次的迭代优化,使得它较为稳定。

JAVA语言目前的最大优势,在于它的生态足够繁荣,各种工具应有尽有,特别适合企业级应用,即便在某些方面它的性能不足,但如果对某些方面的性能要求不高,继承起来也很简单,并且JAVA的开发者依然非常活跃,这本身就是最大的优势。

某些语言甚至比JAVA还要发展的快并且快速成熟,但已经习惯于使用JAVA的人,对于转到其他类似语言来说,并没有多大的兴趣,而且价值不大,即便其他语言当前有着JAVA没有的优势,但依赖于JAVA强大的生态系统,很多特性和工具也会慢慢增加。

怎么说这种情况呢?先发优势,用户优势。


你总结得非常到位,确实从技术层面来说,JVM并不总是比其他语言的运行环境有特别大的技术优势,尤其在一些特定场景下,其他语言的运行时可能具有更好的性能表现或更高效的内存管理。但正如你所说,Java的真正优势并不单纯在于技术,而是在于其生态系统稳定性先发优势

让我们来深入探讨一下这种“先发优势”以及“用户优势”对Java语言的影响。

1. 先发优势与技术积累

Java作为一种语言,经历了多年的发展和演化,它的虚拟机(JVM)、标准库以及各种工具链都经过了无数次的优化和迭代。这种先发优势意味着Java在很多关键技术上已经走在了前面,并且经历了大量的实际使用验证,积累了丰富的经验。

  • 技术稳定性:Java的稳定性不仅体现在语言本身,还体现在其庞大的生态系统中。开发者和企业已经在Java上投入了大量的资源和时间,从而形成了非常完善的技术堆栈。例如,Spring框架、Hibernate、Maven、Jenkins、Eclipse等开发工具,已经成为了行业标准,并广泛被企业所使用。这种长期的积累,使得Java在很多领域成为了“事实上的标准”,即便其他语言出现了性能或功能上的优势,Java依然在许多企业中占据着主导地位。
  • 企业级应用的经验:Java在企业级应用的开发中积累了大量的经验,特别是在大规模分布式系统、金融行业、电商平台、政府应用等领域,Java有着丰富的案例和成熟的解决方案。企业在选择技术栈时,往往会倾向于选择那些已经被验证和广泛使用的技术,而不是冒险尝试一些相对新颖但尚未被广泛验证的语言。
  • 深厚的社区与文档支持:Java作为一种历史悠久的语言,拥有一个极为庞大的开发者社区。无论是技术支持、开源库、开发教程,还是大量的文档和解决方案,Java都有着其他语言无法比拟的丰富资源。这些资源帮助开发者更快速地解决问题,从而提升了开发效率。

2. 用户优势与学习曲线

Java的另一个重要优势在于其用户基础的广泛性,尤其是在企业级应用领域。因为Java的生态系统已经非常成熟,很多大型企业已经在使用Java开发系统,这些企业在技术选型上通常不愿意冒险,更倾向于选择那些“成熟、稳定”的技术栈。

  • 人才优势:Java拥有一个庞大的开发者社区和丰富的人才池。对于企业来说,招聘Java开发者相对容易,因为Java的学习曲线并不陡峭,并且很多计算机专业的学生在学习过程中都会接触到Java。因此,Java开发者的数量庞大,市场上有充足的技术支持。
  • 学习成本与迁移成本:对许多开发者来说,学习Java相对容易,尤其是对于那些有其他面向对象语言经验的人来说,Java的学习曲线并不陡峭。而且,Java与许多其他语言在面向对象编程的思想上有相似之处,这使得Java成为许多人学习编程时的首选语言。对于已经掌握Java的开发者来说,迁移到其他语言(如Go、Rust等)可能并没有很大兴趣,尤其是在Java已经满足大多数需求的情况下。
  • 企业内技术栈的依赖:对于很多使用Java的大型企业来说,迁移到其他语言的成本极高。企业不仅仅是在技术上依赖Java,还在管理、团队、工具链等方面深度依赖Java。因此,即便其他语言可能具备某些优势,企业在考虑到技术转型的成本时,往往会倾向于继续使用Java,并通过现有的工具和框架来弥补Java的不足。

3. Java的持久竞争力:

Java依赖的正是这种强大的生态系统和用户基础,即使在某些方面(比如性能、现代语言特性等)不如新兴语言,也仍然能够保持其在企业级应用中的主导地位。

  • 生态系统的强大效应:正如你所说,生态系统的繁荣是Java最强的竞争力之一。很多新兴语言(如Go、Kotlin、Rust等)虽然技术上可能比Java更为先进,但在工具链、框架、库、开发工具、社区支持等方面,Java依然拥有无法比拟的优势。很多时候,新语言的优势会随着时间的推移逐步在Java中得到补充。比如,Kotlin目前已经成为Java开发的热门替代品,但它的崛起并不是因为Java的彻底失败,而是因为Java社区对现代编程范式的需求,Kotlin能够快速填补这些空缺,而Java则持续通过新版本逐渐引入这些特性。
  • 语言发展和创新:Java本身并未止步不前,虽然它的变化相对保守,但它依然在不断进化。例如,Java 8引入的Lambda表达式、流式API等新特性,提升了语言的现代性;Java 16之后,模块化系统和其他新特性也在不断推进。即便如此,Java也能在向后兼容性方面做得非常好,这使得企业在迁移和使用过程中更加稳定。

4. 先发优势、用户优势的双重作用

正如你所说,先发优势用户优势的结合,是Java能够在激烈的编程语言竞争中长久占据优势的关键。它不仅仅是技术上的领先,更是生态系统的完整性和用户的广泛依赖,使得Java成为了企业级应用的首选。

  • 技术优势的叠加效应:随着时间的推移,Java生态系统的优势逐步积累,形成了一个良性循环。企业对Java的依赖让开发者和工具提供商更愿意投入到Java生态中,从而促进了Java语言和技术的不断进化。这种积累使得Java的生态越发强大,不仅有更好的技术支持,还有更多的社区贡献和开源项目。
  • 持续的技术进步:Java虽然在一些新兴特性上可能不如其他语言,但它仍然在积极改进。新版本的Java不断引入新特性(如记录类、模式匹配等),并且JVM也持续进行优化和创新,这使得Java在很多方面都能与时俱进。

结论:

Java的真正优势并不仅仅在于其技术本身,而是其庞大的生态系统稳定的用户基础。这让Java成为了企业级应用开发的首选技术之一,即便其他语言在某些方面提供了技术上的优势,Java依然通过其稳定性、成熟的工具链和强大的开发者社区占据着领先地位。

先发优势用户优势让Java形成了一个良性循环,促使它在很长一段时间内都能够维持强大的竞争力。即使是当前快速发展的其他语言,想要超越Java的影响力和市场份额,仍然需要付出巨大的努力。而Java的开发者和企业在这种生态系统中找到了一个稳定、可靠的技术栈,即便面对新兴技术,也更倾向于在已有的基础上进行创新和优化,而不是轻易迁移到新的语言和平台。


Q4:

回到JVM,JVM有哪些是需要用户自主去调优的配置?一般来说,是根据对应用使用内存的估计,稍微调整下内存的最大值与最小值吧?

另外,熟练掌握JVM会给写代码带来什么改变?更加注重节省内存?了解对象的生命周期对于编写低配置高效率的程序有着很大的好处,这几乎贯穿于JAVA程序员的全部编码规范。


你的问题非常关键,深入理解JVM不仅仅是调整内存大小,更涉及到对应用性能的全面优化和对代码质量的提升。让我详细回答你的两个问题。


1. JVM需要用户自主调优的配置有哪些?

确实,调整堆内存的最大值和最小值(通过-Xms-Xmx参数)是最常见的JVM调优手段,但除此之外,JVM还有许多配置项可以根据应用的具体需求进行调优。以下是一些常见的需要自主调优的配置:

1.1 垃圾回收器(GC)的选择和参数调优

  • 选择合适的垃圾回收器:JVM提供了多种垃圾回收器,如Serial GC、Parallel GC、CMS GC、G1 GC、ZGC等。不同的垃圾回收器适用于不同的场景。例如,G1 GC适用于大内存、低停顿的应用,而CMS GC适用于需要低延迟的应用。
    • 设置示例
      • XX:+UseG1GC:使用G1垃圾回收器
      • XX:+UseConcMarkSweepGC:使用CMS垃圾回收器
  • 调节垃圾回收参数:根据应用的需求,可以调整垃圾回收器的行为。例如,设置年轻代和老年代的比例、调整GC线程数、设置GC暂停时间目标等。
    • 设置示例
      • XX:MaxGCPauseMillis=200:设置最大GC暂停时间为200毫秒
      • XX:G1HeapRegionSize=32m:设置G1 GC的堆区域大小

1.2 方法区(元空间)大小的设置

  • 元空间(Metaspace)大小:在JDK 8及之后,永久代被元空间取代。元空间用于存储类的元数据,默认情况下大小是有限的。对于动态生成大量类的应用(如大量使用代理、动态生成字节码等),需要调整元空间大小。
    • 设置示例
      • XX:MetaspaceSize=128m:设置初始元空间大小
      • XX:MaxMetaspaceSize=256m:设置元空间最大大小

1.3 线程栈大小

  • 线程栈大小(Stack Size):每个线程都有自己的栈空间,默认大小可能根据平台不同而变化。如果应用创建大量线程,或者线程执行深度递归,需要调整栈大小。
    • 设置示例
      • Xss512k:设置每个线程的栈大小为512KB

1.4 JIT编译器参数

  • 调整即时编译器(JIT)行为:可以通过参数控制JIT编译的层级、优化策略等,以达到性能和启动时间的平衡。
    • 设置示例
      • XX:CompileThreshold=1000:设置方法被调用多少次后进行JIT编译
      • XX:+TieredCompilation:开启分层编译

1.5 类数据共享(Class Data Sharing)

  • 启用类数据共享:通过共享常用类的元数据,可以加快应用的启动速度并减少内存占用。
    • 设置示例
      • Xshare:on:启用类数据共享

1.6 GC日志和监控

  • 启用GC日志:通过查看GC日志,可以分析垃圾回收的频率、停顿时间等信息,有助于进一步调优。
    • 设置示例
      • Xlog:gc*:gc.log:将GC日志输出到gc.log文件
      • XX:+PrintGCDetails:打印GC详细信息

1.7 其他性能相关参数

  • 内存对齐和对象压缩:调整对象指针的压缩,可以节省内存空间,但在大堆内存下可能需要关闭。
    • 设置示例
      • XX:+UseCompressedOops:启用指针压缩(默认开启)
      • XX:-UseCompressedOops:禁用指针压缩
  • 并发参数:调整并行垃圾回收线程数、并行化程度等。
    • 设置示例
      • XX:ParallelGCThreads=4:设置并行GC的线程数为4

1.8 应用特定的配置

  • 针对特定应用场景:例如,对于高并发的Web应用,可能需要调整Tomcat等应用服务器的线程池参数,以及JVM的相关配置。

2. 熟练掌握JVM对代码编写的影响

深入理解JVM的工作原理和特性,可以在代码编写过程中带来以下变化和提升:

2.1 内存管理意识

  • 减少不必要的对象创建:理解对象的创建成本和垃圾回收的代价,编写代码时会更加注意避免不必要的对象创建,减少临时对象,提升内存利用率。
  • 对象生命周期管理:了解对象在JVM内存中的生命周期,尽量缩短对象的存活时间,让对象在年轻代被回收,降低老年代的压力,避免频繁的Full GC。

2.2 性能优化

  • 代码执行路径优化:理解JIT编译器的工作机制,编写代码时可以考虑到热点代码的优化,使关键方法更易于被JIT编译,提高运行效率。
  • 避免性能陷阱:熟悉JVM的性能特性,可以避免一些常见的性能问题,如过度同步、死锁、频繁的异常抛出等。

2.3 并发编程

  • 线程模型理解:深入理解JVM的线程实现和调度,有助于编写高效的并发代码,正确使用线程池、同步机制等。
  • 内存可见性:了解JVM内存模型(Java Memory Model),理解volatile、synchronized的作用,确保并发代码的正确性。

2.4 故障诊断能力

  • 堆栈分析:熟悉JVM内存布局和垃圾回收机制,能够通过内存快照(heap dump)、线程快照(thread dump)等工具,诊断内存泄漏、线程死锁等问题。
  • GC日志分析:能够阅读和分析GC日志,发现内存使用的异常情况,针对性地进行优化。

2.5 代码规范与最佳实践

  • 编码风格优化:基于对JVM的理解,遵循更好的编码规范,如:
    • 尽量使用基本类型:在可能的情况下,使用基本类型(int、long等)而非包装类型,减少内存开销。
    • String优化:合理使用StringBuilder/StringBuffer进行字符串拼接,避免大量的字符串常量池占用。
    • 集合的合理初始化:为集合类指定初始容量,减少扩容带来的性能损耗。

2.6 更好的架构设计

  • 模块化与解耦:理解类加载机制,能够设计出更合理的模块化架构,避免类加载冲突和版本问题。
  • 内存敏感型设计:在设计系统时,考虑内存使用的影响,避免设计导致的内存浪费,如缓存机制的滥用。

2.7 提升代码的可靠性和稳定性

  • 异常处理优化:理解异常处理的开销,避免在性能关键路径中频繁抛出和捕获异常。
  • 资源管理:确保及时释放资源(如IO流、数据库连接等),防止资源泄漏导致的内存问题。

总结

熟练掌握JVM不仅仅是为了调整配置参数,更重要的是深入理解JVM的工作机制,对代码编写和系统设计产生积极的影响。具体而言:

  • 性能优化:通过了解JVM的内部机制,编写出更高效的代码,优化程序的运行性能。
  • 可靠性提升:避免常见的内存和并发问题,提升系统的稳定性和可靠性。
  • 问题排查:具备诊断和解决复杂问题的能力,如内存泄漏、性能瓶颈等。
  • 职业发展:深入理解JVM是高级Java开发者的必备技能,有助于在职业生涯中承担更重要的角色,如性能调优专家、架构师等。

总之,掌握JVM可以让你从一个单纯的代码编写者成长为对系统有全局把控能力的工程师,更好地设计和优化Java应用。


你上面提到了类加载机制,理解类加载机制,能够设计出更合理的模块化架构。这很重要,也很有用,spring应该是这方面做的比较优秀的框架了,但与此同时,spring的类加载机制并不是很好理解,代码阅读难度大。你怎么看待,spring的类加载有没有一条一以贯之的设计理念?它与JVM的类加载机制是否吻合?


你提到了类加载机制在设计合理的模块化架构中所起的重要作用,这确实是一个关键点。深入理解JVM的类加载机制能够帮助开发者更好地设计系统,避免类冲突和依赖问题。接下来,我们将讨论Spring的类加载机制、它的设计理念以及它与JVM的类加载机制之间的关系。


1. JVM的类加载机制概述

在深入讨论Spring之前,先简要回顾一下JVM的类加载机制:

  • 类加载器层次结构:JVM使用双亲委派模型,类加载器按层次结构分为:
    • 启动类加载器(Bootstrap ClassLoader):加载核心类库,如java.lang包。
    • 扩展类加载器(Extension ClassLoader):加载扩展类库。
    • 应用程序类加载器(Application ClassLoader):加载用户类路径上的类。
    • 自定义类加载器:开发者可以自定义类加载器来加载特殊来源的类。
  • 类加载过程
    1. 加载:根据类的全限定名获取字节码并将其转换为Class对象。
    2. 链接
      • 验证:确保字节码符合JVM规范。
      • 准备:为类的静态变量分配内存并设置默认初始值。
      • 解析:将符号引用转换为直接引用。
    3. 初始化:执行类的<clinit>方法,初始化静态变量和静态代码块。
  • 双亲委派模型:类加载器在加载类时,首先委托给父类加载器加载,只有在父加载器无法加载时,子加载器才尝试加载。

理解JVM的类加载机制有助于设计模块化架构,避免类冲突,确保应用的稳定性和可维护性。


2. Spring的类加载机制

Spring框架在JVM的类加载机制之上,提供了自己的抽象和实现,以满足其灵活性和模块化的需求。

2.1 Spring的类加载特点

  • 资源加载抽象:Spring提供了ResourceResourceLoader接口,抽象了资源的加载,无论是文件系统、类路径还是URL,都可以通过统一的方式加载资源。
  • 模块化设计:Spring本身被设计为高度模块化的框架,包含多个独立的模块(如Spring Core、Spring MVC、Spring Data等)。这种设计需要对类的加载和模块之间的依赖关系进行精细的管理。
  • Bean定义和加载:Spring的IoC容器通过解析配置文件或注解,动态地加载和实例化Bean。这涉及到对类的反射和动态代理,以及对类加载器的使用。

2.2 Spring Boot和自定义类加载器

  • 可执行Jar包:Spring Boot允许将应用打包为可执行的Jar包,其中包含所有的依赖。这需要一个特殊的类加载器来加载嵌套在Jar包内的其他Jar包。
  • LaunchedURLClassLoader:Spring Boot使用自定义的LaunchedURLClassLoader,覆盖了JVM默认的类加载器行为,以支持从嵌套的Jar中加载类。
  • 类加载顺序和隔离:为了避免类冲突和确保依赖的正确加载,Spring Boot的类加载器在加载类时,可能会改变默认的双亲委派模型。这使得某些类可以在本地加载,而不是从父加载器加载。

3. Spring的设计理念与类加载机制

3.1 设计理念

  • 灵活性和可扩展性:Spring追求高度的灵活性,允许开发者根据需求进行定制。这包括资源的加载、Bean的定义和初始化过程。
  • 模块化和组件化:Spring的模块化设计需要对不同模块的类和资源进行有效的管理,确保模块之间的解耦和依赖关系的正确处理。
  • 与JVM机制的融合:Spring并未完全脱离JVM的类加载机制,而是在其之上进行了扩展和定制,以满足框架自身的需求。

3.2 与JVM类加载机制的关系

  • 继承和扩展:Spring的类加载器是对JVM类加载器的继承和扩展,它们遵循JVM的类加载规范,但在必要时会进行定制化处理。
  • 双亲委派模型的调整:在某些情况下,Spring会调整双亲委派模型,例如在加载应用自身的类和资源时,可能会优先于父加载器。这种调整是为了满足应用的特殊需求,如热部署、插件机制等。
  • 与反射和动态代理的结合:Spring大量使用反射和动态代理,这些技术需要对类加载器进行深入的理解和控制,以正确地加载和操作类。

4. Spring类加载机制的复杂性和理解难点

4.1 复杂性来源

  • 自定义类加载器的使用:Spring Boot的自定义类加载器改变了默认的类加载行为,增加了理解和调试的难度。
  • 资源抽象的多样性:Spring的资源加载机制支持多种资源类型,开发者需要理解不同资源之间的区别和加载方式。
  • 动态Bean的加载和代理:Spring容器在运行时动态地加载、代理和管理Bean,这涉及到类加载、字节码操作和代理机制。

4.2 理解难度的体现

  • 代码阅读困难:由于Spring的代码高度抽象、模块化,涉及大量的设计模式(如工厂模式、代理模式等),初学者在阅读源码时可能会感到困难。
  • 调试复杂性:在出现类加载相关的问题时,如ClassNotFoundExceptionNoClassDefFoundError等,问题的定位和解决需要对Spring的类加载机制有深入的理解。
  • 文档和资料的不足:尽管Spring有丰富的文档,但对于类加载器的内部实现和机制,相关资料相对较少,需要通过阅读源码和实践来深入理解。

5. 如何看待Spring的类加载机制

5.1 一以贯之的设计理念

  • Spring的核心理念是“简化开发,提供灵活性”。它通过抽象和封装,隐藏了复杂的实现细节,为开发者提供了强大的功能。
  • 类加载机制的设计也是遵循这一理念。Spring通过自定义类加载器和资源加载器,提供了比JVM默认机制更灵活的方式,支持多种应用场景(如嵌套Jar、模块化部署等)。
  • 一致性和可扩展性:虽然Spring的类加载机制复杂,但它的设计是有一致性的,所有的自定义类加载器和资源加载器都是为了满足框架的灵活性和可扩展性需求。

5.2 与JVM类加载机制的吻合

  • 构建在JVM机制之上:Spring的类加载机制并非重新发明轮子,而是建立在JVM的类加载机制之上,遵循JVM的规范。
  • 有选择地调整:在需要满足特定需求时,Spring会调整JVM的默认行为,例如修改双亲委派模型,但这是在充分理解JVM机制的基础上进行的。
  • 兼容性和标准性:Spring的设计始终考虑到与JVM和Java标准的兼容,确保应用能够在不同的环境和平台上运行。

6. 如何更好地理解Spring的类加载机制

6.1 学习JVM的类加载机制

  • 深入理解JVM的类加载流程:掌握类加载的各个阶段和双亲委派模型。
  • 了解类加载器的实现:学习如何自定义类加载器,以及它们在什么情况下会被使用。

6.2 研究Spring的实现

  • 阅读源码:通过阅读Spring的源码,理解其类加载器和资源加载器的实现原理。
  • 实践和调试:在实际项目中,尝试使用Spring的不同功能,观察类加载行为,积累经验。

6.3 参考资料和社区

  • 官方文档和指南:Spring的官方文档提供了大量的参考资料,帮助理解框架的设计和实现。
  • 社区交流:参与社区讨论,向有经验的开发者请教,分享经验和心得。

7. 结论

Spring的类加载机制是对JVM类加载机制的扩展和定制,旨在提供更大的灵活性和功能性。它的设计理念一以贯之,始终围绕着简化开发、提高灵活性和可扩展性。然而,这种灵活性也带来了理解和使用上的复杂性。

对于开发者来说,深入理解JVM的类加载机制,以及Spring如何在其之上构建自己的机制,是非常有价值的。这不仅有助于解决实际开发中的问题,还能提高对框架和系统的整体理解,设计出更合理、更高效的应用架构。


总结

  • Spring的类加载机制建立在JVM机制之上,通过扩展和定制,满足框架的特殊需求。
  • 其设计理念一致,旨在提供灵活性和可扩展性,虽然复杂,但有迹可循。
  • 深入理解JVM和Spring的类加载机制,对设计模块化架构、解决实际问题具有重要意义。

希望以上回答能够帮助你更好地理解Spring的类加载机制,以及它与JVM类加载机制之间的关系。

相关文章:

讨论JAVA、JVM与Spring

Q1: 作为一个JAVA开发人员&#xff0c;对于jvm肯定不陌生&#xff0c;但很多人对它不陌生也仅止于概念上&#xff0c;而且对概念也是模糊不清的&#xff0c;但jvm实际是java程序运行在其中的实际存在的环境&#xff0c;对它的理解应该要是具象化的。 我们还是从一项技术产生的…...

【04】MySQL数据库和数据表的基本操作详解与实例

文章目录 一、连接MySQL服务器二、数据库的基本操作2.1数据库的基本操作1. 创建数据库2. 选择数据库3. 删除数据库4.查询所有数据库5.修改数据库的字符集 2.2 数据表的基本操作1. 创建数据表2. 查看数据表结构3. 删除数据表4. 修改数据表5. 插入数据6. 查询数据7. 更新数据8. 删…...

Spring中实现动态数据源切换,基于AbstractRoutingDataSource

背景 在项目开发过程中&#xff0c;我们可能会遇到一个场景&#xff1a;某个类型数据源有多个数据源实例&#xff0c;需要我们按照不同的请求切换到不同数据源去。 而目前绝大多数java应用都是基于Spring框架来开发&#xff0c;我们很多时候相关的数据源连接都是交给了Spring框…...

StarRocks-join优化

1、背景 有两个大表&#xff0c;都是6kw级别上下的&#xff0c;通过SR然后包装了一个接口对外提供查询&#xff0c;当前的问题是&#xff0c;这样大的join查询会导致BE直接宕机。并且这个sql很有代表性&#xff0c;我截图如下&#xff1a; 这个表是个单分区&#xff0c;所以直接…...

js 高亮文本中包含的关键词标红

在开发中&#xff0c;遇到需要将文本中包含的关键字高亮的情况&#xff0c;可以做以下处理。 <div class"title"v-html"highlightKeywords(item.title, state1.tags1.concat(state2.tags2).concat(state3.tags3))"> </div> ...... ...... con…...

DVWA靶场——File Inclusion

File Inclusion&#xff08;文件包含&#xff09;漏洞 指攻击者通过恶意构造输入&#xff0c;利用应用程序错误的文件包含机制&#xff0c;导致程序包含并执行未经授权的本地或远程文件。这类漏洞广泛存在于Web应用程序中&#xff0c;尤其是在那些允许用户提供文件路径或URL的地…...

Android Framework禁止弹出当前VOLTE不可用的提示窗口

文章目录 VoLTE简介VoLTE 的优势 当前VOLTE不可用的弹窗弹窗代码定位屏蔽弹出窗口 VoLTE简介 VoLTE&#xff08;Voice over LTE&#xff09;是一种基于4G LTE网络的语音通话技术。它允许用户在4G网络上进行高质量的语音通话和视频通话&#xff0c;而不需要回落到2G或3G网络。V…...

OceanBase 大数据量导入(obloader)

现需要将源数据库&#xff08;Oracle|MySQL等&#xff09;一些表的海量数据迁移到目标数据库 OceanBase 中&#xff0c;基于常规 jdbc 驱动编码的方式涉及开发工作&#xff0c;性能效率也要看编码的处理机制。 OceanBase 官方提供了的 OceanBase Migration Service (OMS) 数据…...

w058基于web的美发门店管理系统

&#x1f64a;作者简介&#xff1a;拥有多年开发工作经验&#xff0c;分享技术代码帮助学生学习&#xff0c;独立完成自己的项目或者毕业设计。 代码可以查看文章末尾⬇️联系方式获取&#xff0c;记得注明来意哦~&#x1f339;赠送计算机毕业设计600个选题excel文件&#xff0…...

实时数据开发 | checkpoints监控和调优

监控Checkpoints 监控 checkpoint 行为最简单的方法是通过 UI 的 checkpoint 部分。 监控这两个指标: 算子收到第一个 checkpoint barrier 的时间。当触发 checkpoint 的耗费时间一直很高时&#xff0c;这意味着 checkpoint barrier 需要很长时间才能从 source 到达 operator…...

el-tree的使用及控制全选、反选、获取选中

el-tree的使用及控制全选、反选、获取选中 组件使用获取选中的id全选实现反选实现全部代码 组件使用 引入组件&#xff0c;可以参考官网组件引入参考官网示例写好基础数据结构&#xff0c;不知道怎么转换树形机构的看文章&#xff1a;一维数组转树形 <template><el-…...

机器学习(二十五):决策树算法以及决策树和神经网络的对比

一、决策树集合 单一决策树会对训练数据的变化很敏感。例子&#xff1a;输入十个数据&#xff0c;判断是否是猫。只替换其中一个数据&#xff0c;信息增益最高的分裂特征就发生了改变&#xff0c;决策树就发生了变化。 使用决策树集合可以使算法更加健壮。例子&#xff1a;使用…...

新版布谷直播软件源码开发搭建功能更新明细

即将步入2025年也就是山东布谷科技专注直播系统开发,直播软件源码出售开发搭建等业务第9年,山东布谷科技不断更新直播软件功能&#xff0c;以适应当前新市场环境下的新要求。山东布谷科技始终秉承初心&#xff0c;做一款符合广大客户需求的直播系统软件。支持广大客户提交更多个…...

vue3 reactive响应式实现源码

Vue 3 的 reactive 是基于 JavaScript 的 Proxy 实现的&#xff0c;因此它通过代理机制来拦截对象的操作&#xff0c;从而实现响应式数据的追踪。下面是 Vue 3 的 reactive 源码简化版。 Vue 3 reactive 源码简化版 首先&#xff0c;我们需要了解 reactive 是如何工作的&…...

git的使用(简洁版)

什么是 Git&#xff1f; Git 是一个分布式版本控制系统 (DVCS)&#xff0c;用于跟踪文件的更改并协调多人之间的工作。它由 Linus Torvalds 在 2005 年创建&#xff0c;最初是为了管理 Linux 内核的开发。Git 的主要目标是提供高效、易用的版本控制工具&#xff0c;使得开发者…...

使用命令行创建 Maven 项目

本指南将引导您通过命令行创建一个简单的 Maven 项目。本教程假设您已经在您的机器上安装了 Maven 和 JDK 21。 第一步&#xff1a;验证 Maven 和 Java 安装 在开始之前&#xff0c;我们需要验证 Maven 和 Java 是否已正确安装在您的系统上。 验证 Maven 安装 打开命令行终…...

JVM_栈详解一

1、栈的存储单位 **栈中存储什么&#xff1f;**&#xff0c; 每个线程都有自己的栈&#xff0c;栈中的数据都是以栈帧&#xff08;Stack Frame&#xff09;的格式存在。在这个线程上正在执行的每个方法都各自对应一个栈帧&#xff08;Stack Frame&#xff09;。 栈帧是一个内存…...

Linux 金仓数据库安装和使用

文章目录 Linux 金仓数据库安装和简单使用 一、下载二、安装三、启动法1. 通用启动方式法2. 系统服务启动方式 四、测试五、DB管理工具1. 启动DB管理工具2. DB管理工具的常用功能 六、卸载 Linux 金仓数据库安装和简单使用 一、下载 打开官网 https://www.kingbase.com.cn/xzz…...

STM32笔记(串口IAP升级)

一、IAP简介 IAP&#xff08;In Application Programming&#xff09;即在应用编程&#xff0c; IAP 是用户自己的程序在运行过程中对 User Flash 的部分区域进行烧写&#xff0c;目的是为了在产品发布后可以方便地通过预留的通信口对产 品中的固件程序进行更新升级。 通常实…...

C++网络编程:select IO多路复用及TCP服务器开发

C网络编程&#xff1a;使用select实现IO多路复用 一、什么是 IO 多路复用&#xff1f;二、IO多路复用器 select三、相关接口3.1、fd_set 结构体3.2、宏和函数 四、select 实现 TCP 服务器五、总结 一、什么是 IO 多路复用&#xff1f; 在网络编程中&#xff0c;最容易想到的并…...

部署 L2JMobius 天堂2芙蕾雅版本

首先下载所需要的服务器端 “L2J_Mobius.zip” 和芙蕾雅客户端&#xff08;三个压缩文件&#xff09;&#xff0c; 我的网盘下载&#xff1a;https://pan.baidu.com/s/1XdlcCFPvXnzfwFoVK7Sn7Q?pwdavd4 所有文件都在“芙蕾雅”目录下&#xff0c;也可以加入企鹅交流裙 87470…...

C#开发合集

用C#轻松搞定m3u8视频下载与合并 嘿&#xff0c;程序员们&#xff01;今天咱们来聊聊如何用C#写个小程序&#xff0c;轻松下载和合并m3u8视频文件。没错&#xff0c;就是那种分段的流媒体视频。准备好了吗&#xff1f;让我们开始吧&#xff01; 准备工作 在动手之前&#xf…...

鸿蒙面试 --- 性能优化

性能优化可以从三个方面入手 感知流畅、渲染性能、运行性能 感知流畅 在应用开发中&#xff0c;动画可以为用户界面增添生动、流畅的交互效果&#xff0c;提升用户对应用的好感度。然而&#xff0c;滥用动画也会导致应用性能下降&#xff0c;消耗过多的系统资源&#xff0c;…...

React的基础知识:Context

1. Context 在 React 中&#xff0c;Context 提供了一种通过组件树传递数据的方式&#xff0c;无需手动在每个层级传递 props。这在处理一些全局应用状态时非常有用&#xff0c;比如用户认证、主题、语言偏好等。 如何使用 Context 创建 Context&#xff1a;首先&#xff0c;…...

微知-lspci访问到指定的PCIe设备的几种方式?(lspci -s bus;lspci -d devices)

通过bdf号查看 -s &#xff08;bus&#xff09; lspci -s 03:00.0通过vendor id或者device id等设备查看 -d &#xff08;device&#xff09; lspci -d 15b3: #这里是vendor号&#xff0c;所以在前面 lspci -d :1021 #这里是设备号&#xff0c;所以要:在前vendorid和deviceid…...

【Kubernetes 集群核心概念:Pod】pod生命周期介绍【五】

5.1 Pod生命周期 Pod的生命周期指的是从Pod创建到终止的整个过程。它分为以下两种常见情况&#xff1a; 长期运行Pod&#xff1a; 例如运行HTTP服务的Pod&#xff0c;它在正常情况下会一直运行&#xff0c;但可以手动删除或终止。短期运行Pod&#xff1a; 例如执行计算任务的…...

c++的虚继承说明、案例、代码

虚继承的基本概念 在 C 中&#xff0c;虚继承主要用于解决多继承时可能出现的菱形继承问题。菱形继承是指一个类有两个&#xff08;或更多&#xff09;子类&#xff0c;而这两个子类又同时继承自一个共同的基类&#xff0c;当这些子类又被另一个类继承时&#xff0c;就形成了菱…...

小米PC电脑手机互联互通,小米妙享,小米电脑管家,老款小米笔记本怎么使用,其他品牌笔记本怎么使用,一分钟教会你

说在前面 之前我们体验过妙享中心&#xff0c;里面就有互联互通的全部能力&#xff0c;现在有了小米电脑管家&#xff0c;老款的笔记本竟然用不了&#xff0c;也可以理解&#xff0c;毕竟老款笔记本做系统研发的时候没有预留适配的文件补丁&#xff0c;至于其他品牌的winPC小米…...

介绍SSD硬盘

SSD硬盘&#xff08;固态硬盘&#xff0c;Solid State Drive&#xff09;是一种利用闪存技术存储数据的存储设备&#xff0c;与传统的机械硬盘&#xff08;HDD&#xff09;不同&#xff0c;SSD没有任何活动部件&#xff0c;因此其性能和耐用性较为优越。以下是SSD硬盘的一些主要…...

CMAKE常用命令详解

NDK List基本用法 Get–获取列表中指定索引的元素 list(Get list_name index output_var)解释 list_name: 要操作集合的名称index: 要取得的元素下标output_var: 保存从集合中取得元素的结果 栗子 list(GET mylist 0 first_element) # 获取第一个元素APPEND–在列表末尾…...

怎样注册wordpress/百度一下首页登录

写了个VC调用PHP Sapi 的dll。功能&#xff1a;1.在你的vc、.net程序中嵌入PHP代码2.查看PHP的op-code用法说明&#xff1a;LinxPHPEmbedLib.dll导出4个函数&#xff1a;1.extern "C" __declspec(dllexport) void Test();说明&#xff1a;测试函数。在cmd下运行"…...

黄页网站推广app软件/涟源网站seo

http://www.cnblogs.com/tianzhiliang/archive/2011/01/06/1927691.html...

徐州网站建设找哪家/百度电话号码查询平台

题目描述 用筛法求之N内的素数。 输入 N 输出 0&#xff5e;N的素数 样例输入 100样例输出 2 3 5 7 11 13 17 19 23 29 31 37 41 43 47 53 59 61 67 71 73 79 83 89 97 提示 来源 代码示例一&#xff1a;使用穷举法&#xff0c;按照素数的定义&#xff0c;如果一个数除了1和…...

北京做网站公司有哪些/收录优美图片手机版

在某个微信群里看到个积分题目&#xff0c;说是用六年级的知识解&#xff1f;&#xff1f;&#xff1f;其实根据后面运行的结果来看&#xff0c;手算可能不太行。 import pandas as pd import matplotlib.pyplot as plt %matplotlib inline import numpy as npx1 np.arange(…...

广州制作网站公司电话/g3云推广

文章目录题目描述输入描述输出描述输入输出样例最终代码1. c/c2. java3. python过程理解题目描述 题目背景 在游戏《星际争霸 II》中&#xff0c;高阶圣堂武士作为星灵的重要 AOE 单位&#xff0c;在 游戏的中后期发挥着重要的作用&#xff0c;其技能"灵能风暴"可以…...

怎样用阿里云服务器做网站/宁波seo优化排名

https://blog.csdn.net/weixin_33923762/article/details/91556576...