Optimize JavaScript code
Author: Gregory Baker, GMail software engineer and Erik Arvidsson, Google Chrome software engineer
Experience required: knowledge about JavaScript
Client scripts allow your applications to be more dynamic and active, but the browser may cause efficiency problems in code parsing, and this performance difference varies across clients. here we will discuss and provide some tips and best practices for optimizing your JavaScript code.
Use string
The string connection operation will have a great impact on the garbage collection of Internet Explorer 6 and 7. although this problem is solved in Internet Explorer 8, the string is connected to IE8 and other non-IE browsers (such as Chrome) A little more efficient-if a large part of your users are using Internet Explorer 6 or 7, you need to pay attention to the way you build strings.
The following sample code is provided:
Copy codeThe Code is as follows:
Var veryLongMessage =
'This is a long string that due to our strict line length limit of '+
MaxCharsPerLine +
'Characters per line must be wrapped. '+
PercentWhoDislike +
'% Of engineers dislike this rule. The line length limit is for' +
'Style purposes, but we don't want it to have a performance impact. '+
So the question is how shoshould we do the wrapping? ';
Try to use join () instead of join ():
Copy codeThe Code is as follows:
Var veryLongMessage =
['This is a long string that due to our strict line length limit ',
MaxCharsPerLine,
'Characters per line must be wrapped .',
PercentWhoDislike,
'% Of engineers dislike this rule. The line length limit is ',
'Style purposes, but we don't want it to have a performance impact .',
So the question is how shoshould we do the wrapping? '
]. Join ();
Similarly, it is inefficient to construct strings in conditional statements and loops using the join method. The error method is as follows:
Copy codeThe Code is as follows:
Var fibonacciStr = 'first 20 Fibonacci number ';
For (var I = 0; I <20; I ++ ){
FibonacciStr + = I + '=' + fibonacci (I) +'
';
}
The correct method:
Copy codeThe Code is as follows:
Var strBuilder = ['first 20 Fibonacci numbers: '];
For (var I = 0; I <20; I ++ ){
StrBuilder. push (I, '=', FIG (I ));
}
Var fibonacciStr = strBuilder. join ('');
Construct a string generated by the Auxiliary Function
Create a long string by passing the string Builder (which can be an array or helper class) to the function to avoid the string that stores the temporary results.
For example, assume that buildMenuItemHtml _ needs to construct a string with a text string and a variable, and uses a character string builder internally, instead:
Copy codeThe Code is as follows:
Var strBuilder = [];
For (var I = 0; I <menuItems. length; I ++ ){
StrBuilder. push (this. buildMenuItemHtml _ (menuItems [I]);
}
Var menuHtml = strBuilder. join ();
Better use:
Copy codeThe Code is as follows:
Var strBuilder = [];
For (var I = 0; I <menuItems. length; I ++ ){
This. buildMenuItem _ (menuItems [I], strBuilder );
}
Var menuHtml = strBuilder. join ();
Define the Class Method
The following code is inefficient because every time you construct an instance of baz. Bar, a new function and closure (closure) will be created for foo ):
Copy codeThe Code is as follows:
Baz. Bar = function (){
// Constructor code
This. foo = function (){
// Method code
};
}
Recommended Methods:
Copy codeThe Code is as follows:
Baz. Bar = function (){
// Constructor code
};
Baz. Bar. prototype. foo = function (){
// Method code
};
In this way, no matter how many baz. Bar instances are constructed, only one function is created for foo and no closure is created.
Initialize instance variables
Initialize the value with a value type (non-referenced) (for example, the type is numeric, Boolean, null, undefined, or string value) the variable Declaration/initialization code is directly stored in the prototype. this avoids unnecessary initialization code execution every time you call the constructor. (this method cannot be applied to instance variables whose initial values are determined by the constructor parameters or whose status is not certain during the constructor .)
For example, compared to writing:
Copy codeThe Code is as follows:
Foo. Bar = function (){
This. prop1 _ = 4;
This. prop2 _ = true;
This. prop3 _ = [];
This. prop4 _ = 'blah ';
};
Write as follows:
Copy codeThe Code is as follows:
Foo. Bar = function (){
This. prop3 _ = [];
};
Foo. Bar. prototype. prop1 _ = 4;
Foo. Bar. prototype. prop2 _ = true;
Foo. Bar. prototype. prop4 _ = 'blah ';
Use closures with caution)
Closures are a powerful and useful feature in JavaScript; however, they also have disadvantages, including:
These are the most common sources of Memory leakage.
Creating a closure is obviously slower than creating an inline function without a closure, and slower than reusing a static function. For example:
Copy codeThe Code is as follows:
Function setupAlertTimeout (){
Var msg = 'message to be displayed ';
Window. setTimeout (function () {alert (msg) ;},100 );
}
It is slower than the following code:
Copy codeThe Code is as follows:
Function setupAlertTimeout (){
Window. setTimeout (function (){
Var msg = 'message to be displayed ';
Alert (msg );
},100 );
}
It is slower than the following code:
Copy codeThe Code is as follows:
Function alertMsg (){
Var msg = 'message to be displayed ';
Alert (msg );
}
Function setupAlertTimeout (){
Window. setTimeout (alertMsg, 100 );
}
They added the scope chain level. When the browser parses attributes, each level of the scope chain must be checked once. In the following example:
Copy codeThe Code is as follows:
Var a = 'a ';
Function createFunctionWithClosure (){
Var B = 'B ';
Return function (){
Var c = 'C ';
A;
C;
};
}
Var f = createFunctionWithClosure ();
F ();
When f is called, referencing a is slower than referencing B. They are slower than referencing c.
View IE + JScript Performance Recommendations Part 3: JavaScript Code inefficiencies for more information about using closures in IE.
Avoid using
with
Avoid usingwith
It has a very bad impact on performance, because it modifies the scope chain, making it expensive to search for variables in other scopes.
Avoid browser memory leakage
Memory leakage is a common problem for Web applications, which may cause serious performance problems. when the browser memory usage increases, your Web applications, along with other parts of the user system, will become slower. the most common cause of Memory leakage for Web applications is the circular reference between the JavaScript script engine and the C ++ object Implementation of the browser DOM (for example, between the JavaScript script engine and Internet Explorer's COM infrastructure, or between the JavaScript engine and Firefox's XPCOM infrastructure ).
The following are some empirical rules to avoid Memory leakage:
Use an Event System to attach event processing functions
The most common circular reference mode [DOM element --> Event Handler --> closure scope --> DOM] has been discussed in this Blog article on MSDN. to avoid this problem, you can use a strictly tested Event System to attach event processing functions, such as Google doctype, Dojo, or JQuery.
In addition, using inline event processing functions in IE will lead to another type of leakage. this is not the usual loop reference leakage, but the leakage of temporary anonymous Script objects in the memory. for more information, see section "DOM Insertion Sequence Leakage Model (DOM Insertion Order Leak Model)" in Understanding and Solving the Understanding and Solving Internet Explorer Leak Patterns, in addition, there is an example in the JavaScript Kit tutorial.
Avoid using extended (expando) attributes
Extended attributes are arbitrary JavaScript attributes appended to DOM elements, which are also common causes of circular references. you can use the extended attributes without causing memory leakage, but it is easy to introduce a leak accidentally. this mode of leakage is [DOM elements --> extended attributes --> intermediate objects --> DOM elements]. the best way is to avoid using them. if you want to use them, only simple value types are used. if you want a non-simple type, set it to null when you no longer need to extend the attribute ). for more information, see "Circular References" in the Understanding and Solving Internet Explorer Leak Patterns.