This is the sixth day of my participation in the August More text Challenge. For details, see: August More Text Challenge
Memory management
V8 Memory Limit
Limit the size of the
1.4GB for 64-bit and 0.7GB for 32-bit
Limited reason
V8 limits the size of memory, ostensibly because it was originally designed as a JavaScript engine for browsers, and is unlikely to encounter large amounts of memory, but further because of the limitations of V8’s garbage collection mechanism. Because V8 needs to ensure that the JavaScript application logic is different from what the garbage collector sees, V8 blocks the JavaScript application logic during garbage collection until it is finished and then executes the JavaScript application logic again. This behavior is called stop-the-world. If the heap memory of V8 is 1.5GB, V8 takes more than 50ms to do a small garbage collection, and more than a second to do a non-incremental garbage collection. The browser will not respond to the user for 1 second, causing suspended animation. If there is animation, the presentation of animation will also be significantly affected.
V8 garbage collection policy
- Adopt the idea of generational recycling
- Memory is divided into the new generation, the old generation
- Different algorithms are used to improve the efficiency of garbage collection for new and old generations
New generation objects are short-lived objects, while old generation objects are long-lived or resident objects.
V8 New generation and old generation memory size
The memory size of the new generation of the V8 engine is 32MB (64-bit) and 16MB (32-bit). The memory size of the old generation is 1400MB (64-bit) and 700MB (32-bit).
New generation object recycling implementation
- The recycling process adopts the copy algorithm + mark arrangement
- The new generation memory area is divided equally into two Spaces
- Use space is From, free space is To
- Copy the live object To after tag collation
- From and To swap space To complete the release
promotion
Move Cenozoic objects to older generations
Promotion condition
- A new generation of GC still alive needs promotion
- When an object is copied From the From space To the To space, if the To space is more than 25% used, the object is copied directly To the old generation
Old generation object recycling implementation
- It mainly adopts the algorithm of mark clearing, mark sorting and incremental mark
- Garbage space is first reclaimed using tag cleanup
- Label arrangement is used to optimize the space
- Incremental marking is used for efficiency optimization
Details of the contrast
In the Cenozoic region, the replication algorithm is adopted, so there is free space inside it every moment (in order To complete the object replication From To). However, the Cenozoic region has a small space (32M) and is divided into two, so the waste of space is relatively trivial.
Since the old generation has a large space (1.4g), it is a waste of space if the old generation is divided into two. Moreover, the old generation has a large space and more objects are stored. If the replication algorithm is adopted, the consumption of time will also be greater. In other words, whether to use the replication algorithm for garbage collection is a relationship between time T and the memory size. When the memory size is small, the replication algorithm consumes a relatively short time, while when the memory is large, the replication algorithm consumes more time.
The V8 optimization
Incremental tag
Since a complete pause would leave the browser unresponsive for a period of time, V8 uses an incremental markup approach, where the complete tag is broken up into many parts and then stopped after each part to allow the JS application logic to execute for a while, thus alternating garbage collection with the application logic. With the improvement of incremental marking, the maximum pause time of garbage collection can be reduced to about one-sixth of the original
Inert to clean up
Since all objects have been marked, either dead or alive, by the time the marking is complete, how much space on the heap is determined. We can delay the execution of the cleanup process without having to rush to free the space occupied by the dead objects. The garbage collector can clean up the memory space of dead objects one by one as needed
other
V8 later introduced incremental compaction, as well as parallel flagging and parallel cleanup, to improve garbage collection performance by using multi-core cpus in parallel
Monitor the memory
External manifestations of memory problems
- Pages that load lazily or pause frequently: There may be frequent when GC operations occur, and there may be some code that fills up memory in an instant.
- Pages exhibit persistent poor performance: The program has applied a large amount of memory space to achieve optimal running speed, but the size of the space exceeds that provided by the device.
- Page usage gets slower and slower over time: there may be a memory leak.
Standards for defining memory issues
- Memory leak: Memory usage continues to rise
- Memory ballooning: Performance issues exist on most devices
- Frequent garbage collection: Analysis through memory change sequence diagrams
Monitoring Memory
Task manager
In this example, Shift + Esc is used to invoke Google’s built-in task manager
- The Memory column represents native Memory. DOM nodes are stored in native memory. If this value is increasing, the DOM node is being created.
- The JavaScript Memory column represents the JS heap. This column contains two values. The value you are interested in is the live number (the number in parentheses). Real-time numbers represent the amount of memory being used by reachable objects on your page. If this number is increasing, either new objects are being created or existing objects are growing.
Simulate memory leak
JavaScript memory continues to rise in the task manager
document.body.innerHTML = `<button id="add">add</button>`;
document.getElementById('add').addEventListener('click', function (e) {
simulateMemoryLeak();
});
let result = [];
function simulateMemoryLeak() {
setInterval(function () {
result.push(new Array(1000000).join('x'));
document.body.innerHTML = result;
}, 100);
}
Copy the code
Timeline Record memory
Here, Google browser is used as an example. Use F12 to start the mode, select Performance, click Record to perform page operations, click Stop to end recording, open the memory check box, drag the screenshot to a specified period of time to view the page display when memory problems occur, and locate the problem. At the same time, you can view the corresponding red dot to the execution script to locate the problem code.
Using the browser memory module, look for the detached DOM
Here, Take Google browser as an example. After performing related operations on the page, use F12 to open the mode, select Memory, and click Take Snapshot. Look for the Detached HTMLElement in the snapshot, and go back to the code to find the corresponding Detached DOM. After the relevant operation code, release the Detached DOM to prevent memory leaks.
DOM nodes are garbage collected only if the PAGE’s DOM tree or JavaScript code no longer references them. If a node has been removed from the DOM tree, but some JavaScript still references it, we say the node is “detached.” Detached DOM nodes are a common cause of memory leaks.
Simulate detached DOM nodes
document.body.innerHTML = `<button id="add">add</button>`;
document.getElementById('add').addEventListener('click', function (e) {
create();
});
let detachedTree;
function create() {
let ul = document.createElement('ul');
for (let i = 0; i < 10; i++) {
let li = document.createElement('li');
ul.appendChild(li);
}
detachedTree = ul;
}
Copy the code
How to determine the frequency of garbage collection
- When GC is working, the program is paused. Frequent/long GC will cause the program to fake death and the user will be aware of the lag.
- Check whether the memory trend frequently rises or falls within a short period of time in the Timeline. Whether the browser task manager increases or decreases frequently.
Code optimization
JsPerf (JavaScript performance testing)
Based on the Benchmark. Js
Use global variables with caution
- Global variables are defined in the context of global execution and are at the top of all scope chains
- The global execution context exists in the context execution stack until the program exits
- If a variable with the same name is present in a local scope, it will mask or contaminate the global scope
- The execution speed of global variables is lower than that of local variables, so some global variables that need to be accessed frequently can be cached in the local scope
As you can see from the figure above, the performance of Test2 is better than that of Test1, which means that the execution and access speed of global variables are lower than that of local variables
Avoid global lookup
As you can see from the figure above, test2 performs better than Test1, so caching global variables for later use can improve performance
Add additional methods with prototype objects to improve performance
As you can see from the figure above, the performance of Test2 is better than that of Test1, so that property access on the prototype object is faster by adding methods to the prototype object than by adding member methods directly to the object.
Avoid the closure trap
Closure features
- The outside has a reference to the inside
- Access data from the “outer” part scope to the “inner” part scope
function foo() {
let name = 'heath';
function fn() {
console.log(name);
}
return fn;
}
let a = foo();
a();
Copy the code
Improperly used closures are prone to memory leaks
Function f5() {let el = document.getelementByid (' BTN '); let el = document.getelementByid (' BTN '); el.onclick = function (e) { console.log(e.id); }; } f5(); Function f6() {let el = document.getelementByid (' BTN '); let el = document.getelementByid (' BTN '); el.onclick = function (e) { console.log(e.id); }; el = null; / / we manually releasing el memory here, so after the BTN node to be deleted, may be garbage collected} f6 ();Copy the code
Avoid using property access methods
Object orientation in JavaScript
- JS does not need access methods for properties, all properties are externally visible
- Using attribute access methods only adds another layer of redefinition, with no control over access
As you can see from the figure above, the performance of Test2 is much better than that of Test1, which means that accessing properties directly is much faster than accessing properties through methods.
Traverse speed
As shown in the figure above, the loop traversal speed forEach > optimization for > for of > for > for in
Dom Node Operations
The upload – images. Jianshu. IO/upload_imag… As you can see from the figure above, node clones generate nodes faster than they create them.
Replace the New operation with a literal
As you can see from the figure above, the data generated by literal declarations is faster than the data generated by attribute assignment alone.
Finally:
Public number: xiao He growth, The Buddha department more text, are their own once stepped on the pit or is learned
Interested partners welcome to pay attention to me, I am: He young life. Everybody progress duck together