如何测量Erlang垃圾收集器的性能?

2ic8powd  于 2022-12-08  发布在  Erlang
关注(0)|答案(1)|浏览(138)

我最近开始用Erlang编程,我想了解一些关于垃圾收集(GC)的事情。据我所知,每个进程的私有堆有一个分代GC,全局共享堆有一个引用计数GC。
我想知道的是,是否有任何东西可以得到:
1.收集周期数是多少?
1.在全局级别或进程级别分配和取消分配了多少字节?
1.私有堆和共享堆的大小是什么?我们可以将其定义为GC参数吗?
1.收集垃圾需要多长时间?所需时间的百分比?
1.有没有办法在没有GC的情况下运行程序?
当我运行Erlang程序时,有没有一种方法可以获得这种信息,或者使用代码或者使用一些命令?

  • 谢谢-谢谢
xdnvmnnf

xdnvmnnf1#

  1. To get information for a single process, you can call erlang:process_info(Pid) . This will yield (as of Erlang 18.0) the following fields:
> erlang:process_info(self()).
[{current_function,{erl_eval,do_apply,6}},
 {initial_call,{erlang,apply,2}},
 {status,running},
 {message_queue_len,0},
 {messages,[]},
 {links,[<0.27.0>]},
 {dictionary,[]},
 {trap_exit,false},
 {error_handler,error_handler},
 {priority,normal},
 {group_leader,<0.26.0>},
 {total_heap_size,4184},
 {heap_size,2586},
 {stack_size,24},
 {reductions,3707},
 {garbage_collection,[{min_bin_vheap_size,46422},
                      {min_heap_size,233},
                      {fullsweep_after,65535},
                      {minor_gcs,7}]},
 {suspending,[]}]

The number of collection cycles for the process is available in the field minor_gcs under the section garbage_collection .

  1. Per Process
    The current heap size for the process is available in the field heap_size from the results above (in words, 4 bytes on a 32-bit VM and 8 bytes on a 64-bit VM). The total memory consumption of the process can be obtained by calling erlang:process_info(Pid, memory) which returns for example {memory,34312} for the above process. This includes call stack, heap and internal structures.
    Deallocations (and allocations) can be traced using erlang:trace/3. If the trace flag is garbage_collection you will received messages on the form {trace, Pid, gc_start, Info} and {trace, Pid, gc_end, Info} . The Info field of the gc_start message contains such things as heap_size and old_heap_size .

Per System

Top level statistics of the system can be obtained by erlang:memory/0:

> erlang:memory().
[{total,15023008},
 {processes,4215272},
 {processes_used,4215048},
 {system,10807736},
 {atom,202481},
 {atom_used,187597},
 {binary,325816},
 {code,4575293},
 {ets,234816}]

Garbage collection statistics can be obtained via erlang:statistics(garbage_collection) which yields:

> statistics(garbage_collection).
{85,23961,0}

Where (as of Erlang 18.0) the first field is the total number of garbage collections performed by the VM and the second field is the total number of words reclaimed.

  1. The heap sizes for a process are available under the fields total_heap_size (all heap fragments and stack) and heap_size (the size of the youngest heap generation) from the process info above.
    They can be controlled via spawn options, specifically min_heap_size which sets the initial heap size for a process.
    To set it for all process, erlang:system_flag(min_heap_size, MinHeapSize) can be called.
    You can also control global VM memory allocation via the +M... options to the Erlang VM. The flags are described here. However, this requires extensive knowledge about the internals of the Erlang VM and its allocators and using them should not be taken lightly.
  2. This can be obtained via the tracing described in answer 2. If you use the option timestamp when tracing, you will receive a timestamp with each trace message that can be used to calculate the total GC time.
  3. Short answer: no.
    Long answer: Maybe. You can control the initial heap size (via min_heap_size ) which will affect when garbage collection will occur the first time. You can also control when a full sweep will be performed with the fullsweep_after option.
    More information can be found in the Academic and Historical Questions and Processes section of the Efficiency Guide.
    The most practical way of introspecting Erlang memory usage at runtime is via the Recon library , as Steve Vinoski mentioned.

相关问题