JVM性能调优——GC日志分析

文章目录

  • 1、概述
  • 2、生成GC日志
  • 3、Parallel垃圾收集器日志解析
    • 3.1、Minor GC
    • 3.2、FULL GC
  • 4、G1垃圾收集器日志解析
    • 4.1、Minor GC
    • 4.2、并发收集
    • 4.3、混合收集
    • 4.4、Full GC
  • 5、CMS垃圾收集器日志解析
    • 5.1、Minor GC
    • 5.2、Major GC
    • 5.3、浮动垃圾
  • 6、日志解析工具
    • 6.1、GCeasy
    • 6.2、GCViewer
    • 6.3、其他工具
  • 7、根据日志信息解析堆空间数据分配
  • 8、小结

GC日志是JVM产生的一种描述性的文本日志。就像开发Java程序需要输出日志一样,JVM通过GC日志来描述垃圾收集的情况。通过GC日志,我们能直观地看到内存清理的工作过程,了解垃圾收集的行为,比如何时在新生代执行垃圾收集,何时在老年代执行垃圾收集。本贴将详细讲解如何分析GC日志。

1、概述

GC日志主要用于快速定位系统潜在的内存故障和性能瓶颈,通过阅读GC日志,我们可以了解JVM的内存分配与回收策略。GC日志根据垃圾收集器分类可以分为Parallel垃圾收集器日志、G1垃圾收集器日志和CMS垃圾收集器日志。垃圾收集分为部分收集和整堆收集,所以也可以把GC日志分为Minor GC日志、Major GC日志和Full GC日志。下面开始解析不同垃圾收集器的GC日志。

2、生成GC日志

解析日志之前,我们需要先生成日志,打印内存分配与垃圾收集日志信息的相关参数如下:

1)-XX:+PrintGC
该参数表示输出GC日志,和参数-verbose:gc效果一样。

2)-XX:+PrintGCDetails
该参数表示输出GC的详细日志。

3)-XX:+PrintGCTimeStamps
该参数表示输出GC的时间戳(以基准时间的形式)。

4)-XX:+PrintGCDateStamps
该参数表示输出GC的时间戳(以日期的形式,如2013-05-04T21:53:59.234+0800)。

5)-XX:+PrintHeapAtGC
该参数表示在进行GC的前后打印出堆的信息。

6)-Xloggc:…/logs/gc.log
该参数表示日志文件的输出路径。

使用代码清单如下演示不同的GC日志参数打印出来的日志效果:

import java.util.ArrayList;/*** @title GCLogTest* @description GC日志演示* @author: yangyongbing* @date: 2024/4/11 8:30*/
public class GCLogTest {public static void main(String[] args) {ArrayList<byte[]> list=new ArrayList<>();for (int i = 0; i < 500; i++) {byte[] arr=new byte[1024*100];// 100kblist.add(arr);try {Thread.sleep(50);} catch (InterruptedException e) {e.printStackTrace();}}}
}

配置JVM参数如下:

     -Xms60m -Xmx60m -XX:SurvivorRatio=8

(1)增加输出GC日志参数如下:

     -verbose:gc

这个参数只会显示总的GC堆的变化,结果如下:
在这里插入图片描述

(2)在控制台输出GC日志详情命令如下:

     -verbose:gc -XX:+PrintGCDetails

输出日志信息如下:
在这里插入图片描述
可以发现较之前的日志信息更加详细了,可以明确看到每个区域的内存变化,这使得对日志的分析更加精确了。

(3)增加GC日志打印时间命令如下:

    -verbose:gc -XX:+PrintGCDetails -XX:+PrintGCTimeStamps
-XX:+PrintGCDateStamps

输出日志信息如下:

> Task :GCLogTest.main()
2024-04-11T08:41:16.064+0800: 9.493: [GC (Allocation Failure) [PSYoungGen: 16313K->2032K(18432K)] 16313K->15706K(59392K), 0.0054594 secs] [Times: user=0.00 sys=0.00, real=0.01 secs] 
2024-04-11T08:41:26.262+0800: 19.690: [GC (Allocation Failure) [PSYoungGen: 18355K->2016K(18432K)] 32029K->31984K(59392K), 0.0038027 secs] [Times: user=0.00 sys=0.06, real=0.00 secs] 
2024-04-11T08:41:26.266+0800: 19.694: [Full GC (Ergonomics) [PSYoungGen: 2016K->0K(18432K)] [ParOldGen: 29968K->31926K(40960K)] 31984K->31926K(59392K), [Metaspace: 2671K->2671K(1056768K)], 0.0049717 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
2024-04-11T08:41:36.466+0800: 29.894: [Full GC (Ergonomics) [PSYoungGen: 16324K->7601K(18432K)] [ParOldGen: 31926K->40629K(40960K)] 48251K->48230K(59392K), [Metaspace: 2671K->2671K(1056768K)], 0.0042181 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
HeapPSYoungGen      total 18432K, used 10350K [0x00000000fec00000, 0x0000000100000000, 0x0000000100000000)eden space 16384K, 63% used [0x00000000fec00000,0x00000000ff61bae8,0x00000000ffc00000)from space 2048K, 0% used [0x00000000ffe00000,0x00000000ffe00000,0x0000000100000000)to   space 2048K, 0% used [0x00000000ffc00000,0x00000000ffc00000,0x00000000ffe00000)ParOldGen       total 40960K, used 40629K [0x00000000fc400000, 0x00000000fec00000, 0x00000000fec00000)object space 40960K, 99% used [0x00000000fc400000,0x00000000febad418,0x00000000fec00000)Metaspace       used 2678K, capacity 4486K, committed 4864K, reserved 1056768Kclass space    used 284K, capacity 386K, committed 512K, reserved 1048576K

可以看到日志信息中带上了日期,方便在生产环境中根据日期去定位GC日志2024-04-11T08:41:16.064+0800表示的日志打印时间,该信息是参数“-XX:+PrintGCDateStamps”起的作用;后面的9.493表示虚拟机启动以来到目前打印日志经历的时间,该信息由参数“-XX:+PrintGCTimeStamps”起作用。

(4)在生产环境中,一般都会把日志存放到某个文件中,如果想要达到这一效果可以使用下面的参数:

     -Xloggc:path/gc.log

这里依然使用上面的代码,执行代码清单之前,增加配置参数如下:

     -Xloggc:log/gc.log

其中log表示当前目录下的log文件夹,所以首先需要创建log目录,之后执行代码即可生成日志文件。

3、Parallel垃圾收集器日志解析

3.1、Minor GC

下面是一段Parallel垃圾收集器在新生代产生的Minor GC日志,接下来逐步展开解析:

    2020-11-20T17:19:43.265-0800:0.822:[GC(ALLOCATION FAILURE)
[PSYOUNGGEN:76800K->8433K(89600K)] 76800K->8449K(294400K),0.0088371 SECS]
[TIMES:USER=0.02 SYS=0.01,REAL=0.01 SECS]

日志解析如下表所示:
在这里插入图片描述

3.2、FULL GC

下面解析一段Parallel垃圾收集器产生的FULL GC日志:
在这里插入图片描述
日志解析如下表所示:
在这里插入图片描述
通过日志分析可以总结出Parallel垃圾收集器输出日志的规律,如下图所示:
在这里插入图片描述

4、G1垃圾收集器日志解析

G1垃圾收集器的垃圾收集过程在前面的章节已经讲过了,它是区域化分代式垃圾收集器。G1垃圾收集器的垃圾收集包含四个环节,分别是Minor GC、并发收集、混合收集(Mixed GC)和Full GC,下面针对每个环节的GC日志进行解析。

4.1、Minor GC

下面解析G1垃圾收集器产生的Minor GC日志:
在这里插入图片描述
日志解析如下表所示:
在这里插入图片描述
在这里插入图片描述

4.2、并发收集

经过Minor GC之后就会来到G1垃圾收集的下一个阶段:并发收集,以下面一段G1垃圾收集器并发收集为案例进行解析。
在这里插入图片描述
1、并发垃圾收集阶段的开始
GC pause(G1 Evacuation Pause)(young)(initial-mark)标志着并发垃圾收集阶段的初始标记开始,该阶段会伴随一次Minor GC。

2、根分区扫描
GC concurrent-root-region-scan-start:根分区扫描开始,根分区扫描主要扫描新的Survivor分区,找到这些分区内的对象指向当前分区的引用,如果发现有引用,则做个记录。

GC concurrent-root-region-scan-start:根分区扫描开始,根分区扫描主要扫描新的Survivor分区,找到这些分区内的对象指向当前分区的引用,如果发现有引用,则做个记录。

3、并发标记阶段
GC Concurrent-mark-start:并发标记阶段开始。并发标记阶段的线程是跟应用线程一起运行的,不会STW,所以称为并发,此过程可能被Minor GC中断。在并发标记阶段,若发现区域对象中的所有对象都是垃圾,那这个区域会被立即收集。

GC concurrent-mark-reset-for-overflow:表示全局标记栈已满,发生了栈溢出。并发标记检测到该溢出并重置数据结构,之后重新启动标记。

GC Concurrent-mark-end:并发标记阶段结束,耗时2.3018752 s。

4、重新标记阶段
Finalize Marking:Finalizer列表里的Finalizer对象处理,耗时0.0002438 s;

GC ref-proc:引用(soft、weak、final、phantom、JNI等)处理,耗时0.0018184 s;

Unloading:类卸载,耗时0.0042254 s。

除了前面这几个事情,这个阶段最关键的结果是绘制出当前并发周期中整个堆的最后面貌,剩余的SATB缓冲区会在这里被处理,所有存活的对象都会被标记。

5、清理阶段
[GC cleanup 1912M->1753M(2000M),0.0019143 secs]:清理阶段会发生STW。它遍历所有区域的标记信息,计算每个区域的活跃数据信息,重置标记数据结构,根据垃圾收集效率对区域进行排序。总堆大小是2000M,计算活跃数据之后,发现总活跃数据大小从1912M降到了1753M,耗时0.0019143secs。

6、并发清理阶段
2021-06-08T20:18:26.024-0800:12.431:[GC concurrent-cleanup-start]:表示并发清理阶段开始,它释放在上一个STW阶段期间被发现为空的regions(不包含任何的活跃数据的区域)。

GC concurrent-cleanup-end:并发清理阶段结束,耗时0.0012954s。

4.3、混合收集

在并发收集阶段结束后,会看到混合收集阶段的日志。该日志的大部分内容跟之前讨论的新生代收集相同,只有第1部分不一样,即GC pause(G1Evacuation Pause)(mixed),0.0129474s,这一行表示垃圾混合收集。在混合垃圾收集处理的CSet不仅包括新生代的分区,还包括并发标记阶段标记出来的那些老年代分区。

4.4、Full GC

如果堆内存空间不足以分配新的对象,或者是Metasapce空间使用率达到了设定的阈值,那么就会触发Full GC,在使用G1的时候应该尽量避免这种情况发生,因为G1的Full GC是单线程,会发生STW,代价非常高。Full GC的日志如下所示:
在这里插入图片描述
Full GC(Allocation Failure),表示Full GC的原因,这里是Allocation Failure,表示空间不足,1852M->1615M(2000M)表示内存区域收集,和之前讲解的含义一样,不再赘述,可以看到GC的原因是由堆内存不足导致的。4.1360525 secs表示Full GC的耗时。Full GC频率不能太快,每隔几天发生一次Full GC暂且可以接受,但是每隔1小时发生一次Full GC则不可接受。

5、CMS垃圾收集器日志解析

5.1、Minor GC

选择了CMS垃圾收集器之后,新生代默认选择了ParNew垃圾收集器,以下面一段ParNew垃圾收集器GC日志案例进行解析。
在这里插入图片描述
日志解析如下表所示:
在这里插入图片描述

5.2、Major GC

CMS垃圾收集器主要收集老年代的垃圾,所以产生的日志称为Major GC。CMS垃圾收集器的垃圾收集过程分为7个阶段,分别是初始标记、并发标记、并发预清除、可终止的并发预清理、最终标记、并发清除和并发重置,其中初始标记和最终标记阶段是需要暂停用户线程的,其他阶段垃圾收集线程与用户线程并发执行。下面解析CMS垃圾收集器的GC日志。
在这里插入图片描述

1、初始标记(Initial Mark)
在这里插入图片描述
初始标记是CMS中两次STW事件中的一次。它有两个目标,一是标记老年代中所有的GC Roots;二是标记被年轻代中活着的对象引用的对象。各段日志表示的含义如下,前面的日期和上面讲述的是一样的,此处不再赘述:

  • (1)1 CMS-initial-mark:收集阶段,开始收集所有的GC Roots和直接引用到的对象。
  • (2)1366463K(1366464K):当前老年代的使用情况,括号中表示老年代可用容量。
  • (3)1664869K(1979904K):当前整个堆的使用情况,括号中表示整个堆的容量,所以新生代容量=整个堆(1979904K)-老年代(1366464K)=613440K。

2、并发标记(Concurrent Mark)
在这里插入图片描述
这个阶段会遍历整个老年代并且标记所有存活的对象,从“初始化标记”阶段找到的GC Roots开始。并发标记的特点是和应用程序线程同时运行,并不是老年代的所有存活对象都会被标记,因为标记的同时应用程序会改变一些对象的引用。

(1)CMS-concurrent-mark:进入并发收集阶段,这个阶段会遍历老年代并且标记活着的对象。

(2)0.014/0.014 secs:该阶段持续的时间。

3、并发预清除(Concurrent Preclean)
在这里插入图片描述
这个阶段也是一个并发的过程,即垃圾收集线程和应用线程并行运行,不会中断应用线程。在并发标记的过程中,一些对象的引用也在发生变化,此时JVM会标记堆的这个区域为Dirty Card(包含被标记但是改变了的对象,被认为“dirty”),这就是Card Marking。

在pre-clean阶段,那些能够从Dirty Card对象到达的对象也会被标记,这个标记做完之后,Dirty Card标记就会被清除了。

一些必要的清扫工作也会做,还会做一些Final Remark阶段需要的准备工作。

CMS-concurrent-preclean在这个阶段负责前一个阶段标记了又发生改变的对象标记。

4、可终止的并发预清理(Concurrent Abortable Preclean)
在这里插入图片描述
该阶段依然不会停止应用程序线程。该阶段尝试着去承担STW的Final Remark阶段足够多的工作。这个阶段持续的时间依赖很多因素,由于这个阶段是重复的做相同的事情直到发生aboart的条件(比如重复的次数、多少量的工作、持续的时间等)之一才会停止。

这个阶段很大程度地影响着即将来临的Final Remark的停顿,有相当一部分重要的configuration options和失败的模式。

5、最终标记(Final Remark)
在这里插入图片描述
这个阶段是CMS中第二个并且是最后一个STW的阶段。该阶段的任务是完成标记整个老年代的所有的存活对象。由于之前的预处理是并发的,它可能跟不上应用程序改变的速度,这个时候,是很有必要通过STW来完成最终标记阶段。

通常CMS运行Final Remark阶段是在年轻代足够干净的时候,目的是消除紧接着的连续的几个STW阶段。CMS Final Remark收集阶段,会标记老年代全部的存活对象,包括那些在并发标记阶段更改的或者新创建的引用对象。

YG occupancy:298405 K(613440 K)年轻代当前占用的情况和容量;

Rescan(parallel):这个阶段在应用停止的阶段完成存活对象的标记工作;

weak refs processing:第一个子阶段,随着这个阶段的进行处理弱引用;

class unloading:第二个子阶段,类的卸载;

scrub symbol table:最后一个子阶段,清理字符引用等;

[1 CMS-remark:1366463K(1366464K)]:在这个阶段之后老年代占有的内存大小和老年代的容量;

1664869K(1979904K):在这个阶段之后整个堆的内存大小和整个堆的容量。

6、并发清除(Concurrent Sweep)
通过以上5个阶段的标记,老年代所有存活的对象已经被标记并且清除那些没有标记的对象并且收集空间。该阶段和应用线程同时进行,不需要STW。并发清除阶段的日志如下所示:
在这里插入图片描述

7、并发重置(Concurrent Reset)
CMS-concurrent-reset阶段重新设置CMS算法内部的数据结构,为下一个收集阶段做准备。并发重置阶段的日志如下所示:
在这里插入图片描述

5.3、浮动垃圾

标记阶段是从GCRoots开始标记可达对象,那么在并发标记阶段可能产生两种变动。

(1)本来可达的对象,变得不可达
由于应用线程和垃圾收集线程是同时运行或者交叉运行的,那么在并发标记阶段如果产生新的垃圾对象,CMS将无法对这些垃圾对象进行标记。最终会导致这些新产生的垃圾对象没有被及时收集,从而只能在下一次执行垃圾收集时释放这些之前未被收集的内存空间。这些没有被及时收集的对象称为浮动垃圾。

(2)本来不可达的对象,变得可达:
如果并发标记阶段应用线程创建了一个对象,而它在初始标记和并发标记中是不能被标记的,也就是遗漏了该对象。如果没有最终标记阶段来将这个对象标记为可达,那么它会在清理阶段被收集,这是很严重的错误。所以这也是为什么需要最终标记阶段的原因。

这两种变动相比,浮动垃圾是可容忍的问题,而不是错误。那么为什么最终标记阶段不处理第一种变动呢?由可达变为不可达这样的变化需要重新从GC Roots开始遍历,相当于再完成一次初始标记和并发标记的工作,这样不仅前两个阶段变成多余,造成了开销浪费,还会大大增加重新标记阶段的开销,所带来的暂停时间是追求低延迟的CMS所不能容忍的。

6、日志解析工具

介绍了如何看懂GC日志,但是GC日志看起来比较麻烦,本节将会介绍GC日志可视化分析工具GCeasy和GCviewer等。通过可视化分析工具,可以很方便地看到JVM的内存使用情况、垃圾收集次数、垃圾收集的原因、垃圾收集占用时间、吞吐量等指标,这些指标在JVM调优的时候非常有用。

6.1、GCeasy

GCeasy是一款非常方便的在线分析GC日志的网站。官网首页如下图所示,单击“选择文件”即可上传日志,最后单击“Analyze”按钮便可开始分析日志。

在这里插入图片描述

通过如下代码清单生成日志文件:

/*** @title GCLogTest* @description GC日志演示* @author: yangyongbing* @date: 2024/4/11 8:30*/
public class GCLogTest {public static void main(String[] args) {ArrayList<byte[]> list=new ArrayList<>();for (int i = 0; i < 10000; i++) {byte[] arr=new byte[1024*100];// 100kblist.add(arr);try {Thread.sleep(50);} catch (InterruptedException e) {e.printStackTrace();}}}
}

JVM参数配置如下,其中log表示在工作目录下的log文件夹,所以首先需要创建log目录,之后执行代码即可生成日志文件gc.log。

    -Xms600m -Xmx600m -XX:SurvivorRatio=8 -Xloggc:log/gc.log
-XX:+PrintGCDetails -XX:+PrintGCTimeStamps -XX:+PrintGCDateStamps

内存的分析报告如下图所示:
在这里插入图片描述

其中新生代内存大小为180M,最多使用了179.91M。老年代内存大小为400M,最多使用了400M。

吞吐量和停顿时间分析结果如下图所示:
在这里插入图片描述

其中吞吐量为99.915%,平均停顿时间为27.5ms,最长停顿时间为60ms。

垃圾收集报告如下图所示:
在这里插入图片描述

可以看到GC的次数、收集的内存空间、总时间、平均时间、最短时间和最长时间等相关信息。

6.2、GCViewer

GCViewer是一个免费的、开源的分析小工具,用于可视化查看由SUN/Oracle、IBM、HP和BEA虚拟机产生的垃圾收集器的日志。

GCViewer用于可视化JVM参数-verbose:gc和.NET生成的数据-Xloggc:。它还计算与垃圾收集相关的性能指标,比如吞吐量、累积的暂停、最长的暂停等。当通过更改世代大小或设置初始堆大小来调整特定应用程序的垃圾收集时,此功能非常有用。

下载完成之后执行mvn clean install -Dmaven.test.skip=true命令进行编译,编译完成后在target目录下会看到jar包,打开即可。也可以直接下载运行版本。

通过java命令即可运行该工具,命令如下:

     java -jar gcviewer-1.3x.jar

打开界面如下图所示:
在这里插入图片描述
打开之后,选择“File”→“Open File”选项,选择GC日志,可以看到下图所示页面,图标是可以放大缩小的,主要内容就是图中标记的部分,里面的内容跟上面的GCeasy比较类似。
在这里插入图片描述

6.3、其他工具

GChisto也是一款专业分析垃圾收集器日志的工具,可以通过垃圾收集器日志来分析Minor GC、Full GC的次数、频率、持续时间等。最后通过列表、报表、图表等不同形式来反映垃圾收集器的情况。

另外还有HPjmeter,该工具很强大,但只能打开由-verbose:gc和-Xloggc:gc.log参数生成的GC日志。添加其他参数生成的gc.log无法打开。

7、根据日志信息解析堆空间数据分配

请看如下代码清单:
在这里插入图片描述
运行代码的时候加入以下JVM参数配置,该参数可以使得老年代和新生代的内存分别是10M,垃圾收集器使用Serial GC。首先在JDK7中测试。

     -verbose:gc -Xms20M -Xmx20M -Xmn10M -XX:+PrintGCDetails-XX:SurvivorRatio=8 -XX:+UseSerialGC

当堆内存中存储allocation4对象时发现Eden区中的内存不足,S0区和S1区的空间也不足以存下新对象,如下图所示:
在这里插入图片描述
这时进行GC把Eden区中的数据转移到老年代,再把新对象的数据存放到Eden区,结果如下图所示。Eden区放入对象allocation4,老年代放入另外三个数组对象,内存大小总和为6M,占比60%。
在这里插入图片描述
其日志输出结果如下图所示。可以看到Eden区占比为65%,老年代占比为60%,正好对应了前面的说法。
在这里插入图片描述
需要注意的是在JDK 1.8中,可能出现两种结果,一种是老年代占比为60%,和JDK 1.7中内存分配是一样的,还有一种情况是老年代占比为40%,这是由于JDK 1.8小版本号的不同导致的,这里的40%指的是allocation1和allocation2的内存之和,allocation3并没有转移到老年代,这只是小版本号之间的差异,大家只要能够根据GC日志分析清楚哪些对象在哪个区域即可。

如果使用的是ParallelGC,也可能出现直接把allocation4放入老年代的情况,占比为50%,其日志输出结果如下图所示:
在这里插入图片描述

8、小结

讲解了GC日志分析,主要针对三种垃圾收集器产生日志进行分析,分别是Parallel垃圾收集器日志解析、G1垃圾收集器日志解析和CMS垃圾收集器日志解析,讲述了每一段日志的含义以及垃圾收集器在不同阶段产生的日志信息。

在工作中,GC日志文件往往会比较大,我们手动翻阅查看很容易忽略掉关键信息,接下来介绍了常用的日志分析工具,通过日志分析工具可以获得很多关键信息,比如堆内存分析、GC吞吐量和GC时间等信息。根据这些信息调整JVM参数进而观察应用的表现,最终达到比较理想的程度。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/615766.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【300套】基于Springboot+Vue的Java实战开发项目(附源码+演示视频+LW)

大家好&#xff01;我是程序员一帆&#xff0c;感谢您阅读本文&#xff0c;欢迎一键三连哦。 &#x1f9e1;今天给大家分享300的Java毕业设计&#xff0c;基于Springbootvue框架&#xff0c;这些项目都经过精心挑选&#xff0c;涵盖了不同的实战主题和用例&#xff0c;可做毕业…

基于多种语言,使用Selenium实现自动化的常用单元测试框架!

简介 Selenium是自动化网络应用程序的首选工具。Selenium支持基于Java、C#、PHP、Ruby、Perl、JavaScript和Python等多种编程语言的各种单元测试框架。这些框架用于在 Windows、MacOS 和 Linux 等不同平台的网络应用程序上执行测试脚本。任何成功的自动化流程都有赖于强大的测…

恶意样本溯源法

恶意样本溯源追踪主要去了解攻击者或者团队的意图&#xff1b;&#xff08;有因才有果&#xff0c;这里主要是通过分析结果&#xff0c;去了解攻击者的意图&#xff0c;比如&#xff1a;政治&#xff0c;私怨&#xff0c;业务&#xff0c;经济等&#xff09;。 了解攻击链 恶…

SF58-ASEMI适配器二极管SF58

编辑&#xff1a;ll SF58-ASEMI适配器二极管SF58 型号&#xff1a;SF58 品牌&#xff1a;ASEMI 封装&#xff1a;DO-27 最大平均正向电流&#xff08;IF&#xff09;&#xff1a;5A 最大循环峰值反向电压&#xff08;VRRM&#xff09;&#xff1a;600V 最大正向电压&…

逆向案例二十三——某租逆向,总是有映射源文件怎么办以及分析webpack代码

网址&#xff1a;aHR0cHM6Ly93d3cubWFvbWFvenUuY29tLyMvYnVpbGQ 抓取数据包发现载荷以及数据都进行了加密&#xff1a; 定位方法一&#xff1a;直接搜decrypt(,进入js文件&#xff0c;可以发现就是直接AES的解密方法&#xff0c;打上断点&#xff0c; 下方的d是解密函数 现在有…

视频评论ID提取工具|视频关键词评论批量采集软件

视频评论ID提取工具&#xff1a;批量抓取视频评论 视频评论ID提取工具是一款功能强大的软件&#xff0c;可以帮助您批量抓取视频视频下的评论信息。通过输入关键词和评论监控词&#xff0c;即可进行评论的抓取&#xff0c;并提供评论昵称、评论日期、评论内容、命中关键词以及所…

ExpressLRS硬件实测性能分析

ExpressLRS硬件实测性能分析 1. 源由2. 远航测试3. 实验室测试3.1 芯片RSSI与实测功率差异3.2 SNR信噪比稳定3.3 140db衰减器衰减&#xff0c;40个频点信号稳定 4. 外场测试4.1 无屏蔽样品4.2 有屏蔽样品4.3 有屏蔽vs无屏蔽样品 5. 估算6. 总结7. 补充说明 -- 50mW视频 1. 源由…

【论文笔记】Planning-oriented Autonomous Driving

原文链接&#xff1a;https://arxiv.org/abs/2212.10156 1. 引言 目前的自动驾驶工业界通常为不同任务部署不同的模型&#xff0c;但优化的孤立性会导致模块之间的信息损失、误差积累和特征不对齐。 一种更好的设计是将各种任务整合为多任务学习&#xff0c;即为共享的特征提…

学习python你必须弄懂的 Python、Pycharm、Anaconda 三者之间的关系

省流版 Python作为深度学习和人工智能学习的热门语言&#xff0c;学习一门语言&#xff0c;除了学会其简单的语法之外还需要对其进行运行和实现&#xff0c;才能实现和发挥其功能和作用。下面来介绍运行Python代码常用到的工具总结。 文中部分内容来自GPT&#xff0c;他已经是…

【日常记录】【CSS】利用动画延迟实现复杂动画

文章目录 1、介绍2、原理3、代码4、参考链接 1、介绍 对于这个效果而言&#xff0c;最先想到的就是 监听滑块的input事件来做一些操作 ,但是会发现&#xff0c;对于某一个节点的时候&#xff0c;这个样式操作起来比较麻烦 只看这个代码的话&#xff0c;发现他用的是动画&#x…

SAP HCM GET pernr无法获取到数据二

今天遇到一个比较奇怪的问题&#xff0c;PA30能查到员工主数据&#xff0c;任何信息类型也没有错误&#xff0c;但是核算工资的时候发现无法找到此人。 但是核算工资无法核算 断点到逻辑数据get pernr&#xff0c;也不会进入断点 查看0000数据有间隔 具体错误的代码位置如下&am…

软考 - 系统架构设计师 - 质量属性例题

问题 1&#xff1a; 问题 2&#xff1a; 系统架构风险&#xff1a;m &#xff08;描述尚未达到共识就说明具有风险&#xff09; 敏感点&#xff1a;e &#xff08;敏感点是指为了实现某种特定的质量属性&#xff0c;一个或多个构件所具有的特征&#xff0c;对查询请求处理时间的…