Batching GC with DelayGcContext

You are browsing legacy Javonet 1.5 (Java<>.NET bridge for Windows) documentation. Use the left side menu or click here to switch to latest Javonet 2.0 documentation. Javonet 2.0 allows you to use any module from JVM, CLR, Netcore, Python, Ruby, Perl, NodeJS on Windows, Linux and MacOs from any application created in Java, Clojure, Groovy, Kotlin, C#, F#, J#, VB.NET, Python, Perl, Ruby, JavaScript, TypeScript, C++ and GoLang

There could be a scenario where huge amount of short living instances of .NET objects is created on Java side. Huge amount is estimated above 500 000 objects per minute. It could be when .NET side triggers event subscribed on Java side with many reference-type arguments (NObjects) or when multiple objects are being created for executing business logic calculation in big loops. In each of these cases when memory utilization is increased Java garbage collector will quickly start collecting NObject instances. Javonet forwards such calls to .NET side. When there are more than 500 000 collections per minute the overall application performance might be affected by additional GC call for each of the NObject instances.
In such cases the new DelayGcContext class (introduced in Javonet 1.4) could be applied. The best way to determine if DelayGcContext is required is by performing performance tests with suspicious piece of code. DelayGcContext class allows to indicate the beginning and the end of the code for which the garbage collection calls should be batched. The GC buffer for these objects will be flushed to .NET side every time the buffer size exceeds 100 000 objects pending in the queue or when 30 seconds pass.

The beginning of delayed garbage collection section should be indicated by calling static "Begin" method on DelayGcContext class and terminated by calling "End" method as presented on the snippet below:

I code in:
// Todo: activate Javonet and add reference to .NET library

DelayGcContext.Begin();
for (int i = 0; i < 5000000; i++) {
    NObject objA = Javonet.New("SampleType", i);
    NObject objB = objA.get("B");

    //processItem(objB);
    //(... some other operations ...)
    NObject objC = objB.getRef("subProp").getRef("subProp").invoke("GetC");

    //in this block objA, objB, and objC instance if not referenced in other parts of code
    //will be subjected for garbage collection after each iteration of the loop
}
DelayGcContext.End();

More about DelayGcContext you can read in our docs section for DelayGcContext class