No poetic female program apologise not good cook ~ reprint please indicate the source, the From Li Shiyu – [blog.csdn.net/cjm24848365…].

1. Garbage Collection

1.1 Who needs GC?

Heap: The point!

Method area/meta space :(just know that there is garbage collection here too)

Stack: No. Threads are private, they die with the thread, and you don’t need to worry too much about garbage collection.

1.2 GC trigger condition: Out of memory

Not enough new generations → Minor GC

The old days are not enough

Addendum: further division of the heap

▶ PSYoungGen

Is divided into:

Eden space,

• the From Survivor Spaces

• To Survivor Spaces

▶ Old Age (ParOldGen)

2. How does GC determine whether an object is alive?

2.1 Reference Counting Algorithm

Reference counting adds a reference counter to an object, incrementing it by 1 each time a reference is made to it. When a reference is invalid, the counter value is reduced by 1. When the counter is 0, the object is considered impossible to use again.

▶ advantages:

Fast, convenient and simple to implement.

▶ faults:

When objects refer to each other, it is difficult to determine whether objects should be reclaimed

2.2 Accessibility analysis (key points of interview) ▲▲▲

The basic idea of this algorithm is to search down from a series of objects called “GC Roots” as the starting point, and the search path is called the Reference Chain. When an object is not connected to GC Roots by any Reference Chain, it is proved that the object is unavailable.

Objects as GC Roots include the following:

♦ 1. Objects in the VM stack (local variable table in the stack frame). (Parameter in method, local variable in method body)

♦ 2. Static attribute-like objects in the method area. (static)

♦ 3. Constant objects in the method area. (final static)

♦ 4. JNI (generally called Native method) objects in Native method stack

Meet a chestnut

public class GCRoots {

Object o =new Object(); //o is not GCRoots, o can be recycled after the method is run.
static Object GCRoot1 =new Object(); //GC Roots-- Static attribute-like objects in the method area
final  static Object GCRoot2 =new Object();//GCRoots-- Constant objects in the method area

public static void main(String[] args) {
    / / can reach
    Object object1 = GCRoot1; // Note: "=" is not an assignment. In an object, it is a reference, passing the address of the object on the right
    Object object2 = object1;
    Object object3 = object1;
    Object object4 = object3;
}
public void method1(a){
    // Unreachable (the method can be recycled after running)
    Object object5 = o;/ / o not GCRoots
    Object object6 = object5;
    Object object7 = object5;
}
// The object referenced in the local variable table
public void stack(a){
    Object ostack =new Object();    // The object of the local variable table
    Object object9 = ostack;
    // The above object9 is reachable until the method exits the stack}}Copy the code

Let’s draw a picture to understand the code above:

2.3 References

Whether the reference counting algorithm determines the number of references to an object, or whether the reference chain of an object is reachable or whether the object is alive through the reachable analysis algorithm, it is all related to references. So let’s talk about references again.

▶ strong reference

Strong references are common in program code, like ** “Object obj = new Object()” **.

As long as strong references exist, the garbage collector will never reclaim the referenced object

▶ soft references

Soft references are used to describe objects that are useful but not necessary.

Objects associated with soft references will be reclaimed before OOM occurs.

Only men eat soft rice said, there is no man eat weak rice said, if the strong man row first, that eat soft rice man can row second, eat weak rice are not men, so row third, last is false quote.

Dynamic Change uses a chestnut to explain the use of soft references (PS: VM parameter is set to -xMS10m-XMx10m-xx :+PrintGC) :

public class TestSoftRef {
    public static class User{
        public int id = 0;
        public String name = "";
        public User(int id, String name) {
            super(a);this.id = id;
            this.name = name;
        }
        @Override
        public String toString(a) {
            return "User [id=" + id + ", name=" + name + "]"; }}public static void main(String[] args) {
        User u = new User(1."Zheng Shuang"); // New is a strong reference
        // Examples of soft references:
        SoftReference<User> userSoft = new SoftReference<User>(u);
        u = null;// Eliminate strong references and ensure that this instance has only soft references from userSoft
        //-- SoftReference
      
        userSoft = new SoftReference
       
        (new User()); You can't kill strong references
       
      
        System.out.println(userSoft.get());
        System.gc();// Do a GC garbage collection
        System.out.println("After gc");
        System.out.println(userSoft.get());
        // Populate the heap with data, resulting in OOM
        List<byte[]> list = new LinkedList<>();
        try {
            for(int i=0; i<100; i++) { System.out.println("* * * * * * * * * * * * *"+userSoft.get());
                list.add(new byte[1024*1024*1]); / / 1 m objects}}catch (Throwable e) {
            // Print soft reference objects when an OOM exception is thrown
            System.out.println("Exception*************"+userSoft.get()); }}}Copy the code

Take a look at the print result:

Soft reference usage scenarios:

For example, a program is used to process user-supplied images.

If all images are read into memory, the images can be opened quickly, but the memory space is huge, and some less used images waste memory space and need to be manually removed from memory.

If every time an image is opened, it is read into memory from the disk file and displayed again, although the memory footprint is small, some frequently used images need to access the disk every time they are opened, which is costly.

At this point you can build the cache with soft references.

▶ weak references

Objects associated with weak references that are useful (to a lesser degree than soft references) but not necessary will only survive until the next garbage collection, when GC occurs and will be collected regardless of memory availability.

Aid Talk is cheap,show me the code!

public class TestWeakRef {
    public static class User{
        public int id = 0;
        public String name = "";
        public User(int id, String name) {
            super(a);this.id = id;
            this.name = name;
        }
        @Override
        public String toString(a) {
            return "User [id=" + id + ", name=" + name + "]"; }}public static void main(String[] args) {
        User u = new User(1."Little cool");
        WeakReference<User> userWeak = new WeakReference<User>(u);
        u = null;// Kill strong references and ensure that this instance has only userWeak weak references
        System.out.println(userWeak.get());
        System.gc();// Do a GC garbage collection
        System.out.println("After gc"); System.out.println(userWeak.get()); }}Copy the code

Print result analysis:

Note: SoftReference SoftReference and WeakReference can be used to create less important data caches when memory resources are tight. When the system runs out of memory, the contents of the cache can be freed.

Practical application (WeakHashMap, ThreadLocal)

▶ reference

Ghost references, the weakest, receive a notification when garbage collected

Garbage collection algorithms

3.1 Copying Algorithms

Divide the available memory into two equally sized pieces by capacity and use only one piece at a time. When this area of memory is used up, the surviving objects are copied to the other area, and the used memory space is cleaned up again. In this way, the memory is reclaimed for the whole half area every time, and the complicated situation of memory fragmentation is not considered when allocating memory. As long as the memory is allocated in order, the implementation is simple and the operation is efficient. But this algorithm comes at the cost of reducing memory by half.


▶ characteristics

▪ Simple implementation and efficient operation

▪ Memory replication, no memory fragmentation

The utilization rate is only half


▶ Precautions:

▪ New generation uses the algorithm

▪ The ratio of the three regions in Cenozoic is 8:1:1

▪ Space guarantee


An illustration of the 8:1:1 ratio:

98% of the objects in the new generation are “live and die”, so there is no need to divide the memory space in a 1:1 ratio. Instead, the memory is divided into a large Eden space and two small Survivor Spaces, and Eden and Survivor are used each time. At the time of reclamation, the surviving objects in Eden and Survivor are copied to another Survivor space at a time, and Eden and the Survivor space that was just used are cleared. The default HotSpot VIRTUAL machine size ratio of Eden to Survivor is 8:1, which means that each new generation has 90%(80%+10%) of the available memory of the entire new generation, and only 10% of the memory is wasted.


Description of space guarantee:

Of course, 98% of the objects are recyclable, but this is only the data in the general scenario. There is no way to guarantee that no more than 10% of the objects will survive each collection. When Survivor space is insufficient, other memory (here refers to the old years) is required for allocation guarantee (Handle Promotion). The guarantee of memory allocation is like when we go to the bank for a loan. If we are creditworthy and pay back on time 98% of the time, then the bank may assume that we can pay back the loan on time next time. Only if we don’t need to have a guarantor can guarantee payment, can deduct money from his account, the bank is considered without risk, the allocation of memory guarantees, too, if there is not enough room for another piece of survival space, the last time the new generation collection down the survival of the object, the object will be directly by assigning guarantee mechanism into old age.

3.2 Mark-Sweep Algorithm

The algorithm is divided into “mark” and “clear” two stages: first, mark all the objects that need to be recycled, after the completion of marking all the marked objects.

The main problem of insufficient space is that a large number of discontinuous memory fragments will be generated after the mark is cleared. Too much space fragment may cause that when the program needs to allocate large objects in the future, it cannot find enough contiguous memory and has to trigger another garbage collection action in advance.


▶ characteristics

▪ 100% utilization rate

▪ No memory replication required

▪ Memory fragmentation

3.3 Mark-Compact Algorithm


▶ characteristics

▪ 100% utilization rate

▪ No memory fragmentation

▪ Memory replication is required

▪ Average efficiency

4. Heap memory allocation policy

▶ The object is allocated in Eden first

If Eden is out of memory, a Minor GC occurs

▶ Big object directly into the old era

Large objects: Java objects that require a large amount of contiguous memory space, such as long strings and large arrays,

Large objects are bad news for virtual machine memory allocation (∵1, Garbage collection is triggered in advance to obtain sufficient contiguous space to “place” large objects when there is not enough memory left).

The worse news than a large object is to encounter a group of “short-lived large objects” (∵ 2, can make a large amount of memory replication).

Virtual machine provides a – XX: PretenureSizeThreshold parameters,

If the value is greater than this value, it is directly allocated in the old age. The default value is 0, indicating that it is never allocated in the old age.

▶ The object of long-term survival will enter the old age

The default age is 15. -xx :MaxTenuringThreshold is adjustable

▶ Dynamic object age determination

In order to better adapt to the memory conditions of different programs, the virtual machine does not always require that the object age must reach MaxTenuringThreshold to advance to the old age. If the sum of all object sizes of the same age in the Survivor space is greater than half of the Survivor space, Objects older than or equal to this age can go directly to the old age without waiting until the age specified in MaxTenuringThreshold.

▶ Space allocation guarantee

A large number of objects survive in the new generation and survivor space is not enough. When a large number of objects survive after MinorGC (the most extreme case is that all objects survive in the new generation after memory reclamation), the old generation needs to allocate guarantee, and the objects that survivor cannot accommodate directly enter the old generation. Minor GC is performed as long as the contiguous space of the old generation is greater than the total size of the new generation object or the average size of the previous promotions, otherwise FullGC.

Therefore, the new generation generally does not run out of memory, because the old generation is guaranteed.