Finding Memory Usage in Java

后端 未结 8 2334
野趣味
野趣味 2021-02-09 13:52

Following is the scenario i need to solve. I have struck with two solutions.

I need to maintain a cache of data fetched from database to be shown on a Swing GUI. Wheneve

相关标签:
8条回答
  • 2021-02-09 13:56

    Look into JConsole. It graphs the information you need so it is a matter of adapting this to your needs (given that you run on a Sun Java 6).

    This also allows you to detach the surveiling process from what you want to look at.

    0 讨论(0)
  • Very late after the original post, I know, but I thought I'd post an example of how I've done it. Hopefully it'll be of some use to someone (I stress, it's a proof of principal example, nothing else... not particularly elegant either :) )

    Just stick these two functions in a class, and it should work.

    EDIT: Oh, andimport java.util.ArrayList; import java.util.List;

    public static int MEM(){
        return (int)(Runtime.getRuntime().maxMemory()-Runtime.getRuntime().totalMemory() +Runtime.getRuntime().freeMemory())/1024/1024;
    }
    
    public static void main(String[] args) throws InterruptedException
    {
        List list = new ArrayList();
    
        //get available memory before filling list
        int initMem = MEM();
        int lowMemWarning = (int) (initMem * 0.2);
        int highMem = (int) (initMem *0.8);
    
    
        int iteration =0;
        while(true)
        {
            //use up some memory
            list.add(Math.random());
    
            //report
            if(++iteration%10000==0)
            {
                System.out.printf("Available Memory: %dMb \tListSize: %d\n", MEM(),list.size());
    
                //if low on memory, clear list and await garbage collection before continuing
                if(MEM()<lowMemWarning)
                {
                    System.out.printf("Warning! Low memory (%dMb remaining). Clearing list and cleaning up.\n",MEM());
    
                    //clear list
                    list = new ArrayList();  //obviously, here is a good place to put your warning logic
    
                    //ensure garbage collection occurs before continuing to re-add to list, to avoid immediately entering this block again
                    while(MEM()<highMem)
                    {
                        System.out.printf("Awaiting gc...(%dMb remaining)\n",MEM());
                        //give it a nudge
                        Runtime.getRuntime().gc(); 
                        Thread.sleep(250);
                    }
    
                    System.out.printf("gc successful! Continuing to fill list (%dMb remaining). List size: %d\n",MEM(),list.size());
                    Thread.sleep(3000); //just to view output
                }
            }
        }
    }
    

    EDIT: This approach still relies on sensible setting of memory in the jvm using -Xmx, however.

    EDIT2: It seems that the gc request line really does help things along, at least on my jvm. ymmv.

    0 讨论(0)
  • 2021-02-09 14:00

    It is entirely normal for a JVM to go up to 100% memory usage and them back to say 10% after a GC and do this every few second.

    You shouldn't need to try managing the memory in this way. You cannot say how much memory is being retained until a full GC has been run.

    I suggest you work out what you are really trying to achieve and look at the problem another way.

    0 讨论(0)
  • 2021-02-09 14:01

    The requirements you mention are a clear contradiction with how Garbage Collection works in a JVM.

    because of the behaviour of the JVM it will be very hard to warn you users in a correct way. Altogether stopping als database manipulation , cleaning stuff up and starting again really is not the way to go.

    Let the JVM do what it is supposed to do, handle all memory related for you. Modern generations of the JVM are very good at it and with some finetuning of the GC parameters you will get a a much cleaner memory handling then forcing things yourself

    Articles like http://www.kodewerk.com/advice_on_jvm_heap_tuning_dont_touch_that_dial.htm mention the pros and cons and offer a nice explanation of what the VM does for you

    0 讨论(0)
  • 2021-02-09 14:09

    Just a side note: Runtime.freeMemory() doesn't state the amount of memory that's left of allocating, it's just the amount of memory that's free within the currently allocated memory (which is initially smaller than the maximum memory the VM is configured to use), but grows over time.

    When starting a VM, the max memory (Runtime.maxMemory()) just defines the upper limit of memory that the VM may allocate (configurable using the -Xmx VM option). The total memory (Runtime.totalMemory()) is the initial size of the memory allocated for the VM process (configurable using the -Xms VM option), and will dynamically grow every time you allocate more than the currently free portion of it (Runtime.freeMemory()), until it reaches the max memory.

    The metric you're interested in is the memory available for further allocation:

    long usableFreeMemory= Runtime.getRuntime().maxMemory()
        -Runtime.getRuntime().totalMemory()
        +Runtime.getRuntime().freeMemory()
    

    or:

    double usedPercent=(double)(Runtime.getRuntime().totalMemory()
        -Runtime.getRuntime().freeMemory())/Runtime.getRuntime().maxMemory()
    
    0 讨论(0)
  • 2021-02-09 14:09

    VisualVM is a bit nicer than JConsole because it gives you a nice visual Garbage Collector view.

    0 讨论(0)
提交回复
热议问题