An approach to JavaScript from the perspective of a battle-hardened .NET shop.
- General Formatting 1. Spacing 2. Semicolons 3. Commas 4. Braces 5. Strings 6. length 7. Comments
- Style 1. Declarations 2. Naming Conventions 3. Literals 4. Functions 5. Comparisons 6. Type Checking 6. Namespacing 7. Global 8. Errors
- Implementation 1. Prototypes 2. Async 3. Eval, With, Object.freeze, Object.seal, Object.preventExtensions ... 4. Performance 5. Unit Testing
-
Use soft tabs of four spaces.
// Bad function sample() { var x = 2; } // Bad function sample() { ∙∙var x = 2; } // Bad function sample() { → var x = 2; } // Good function sample() { ∙∙∙∙var x = 2; }
-
Place one space before leading braces and after commas.
// Bad function sample(){ // ... implementation ... } // Good function sample()∙{ // ... implementation ... } // Bad update('contact',{ name: 'Reptar' }); // Good update('contact',∙{ name: 'Reptar' });
-
Use indentation when chaining.
// Bad $element.on().off().remove(); // Bad $element.on() .off() .remove(); // Good $element.on() .off() .remove();
There are those out there who do not believe in the semicolon, but alas, we do! Semicolons aid in readability and minification. Two things that are very important, so just remember to use them.
Note that semicolons aren't necessary at the end of function declarations, just at the end of function expressions. Semicolons serve to separate statements from each other and function declarations are not statements.
-
Leading? Nope. This style has helps when troubleshooting visually, but we should all be linting our code anyways, right? If you find this is an issue, make sure you've got lint running and you should be good to go.
// Bad var x = 2 , y = 3;
-
Trailing? Yep. Much cleaner looking and not as hard to type.
// Good var x = 2, y = 3;
-
Functions: Always use braces. This helps with readability in general. Awkward multiple line statements are bound to occur that may not make sense immediately to another developer who is skimming over your code. Braces help ensure blocks are well defined with a good visual weight.
// Bad if (true) return false; // Bad if (true) return false; // Good if (true) { return false; } // Acceptable (only when immediately exiting a function, ex. argument validation) if (true) { return false; }
-
Use single quotes
''
for strings. This helps alleviate the pain of having to escape quotes in normal copy as well as in XML.// Bad var name = 'Reptar'; // Good var name = 'Reptar';
-
Use commonsense when deciding to wrap strings across multiple lines. Build wrapped strings using array concatenation (mainly for performance benefits in Internet Explorer).
// Bad var message = 'This message' + 'goes across several' + 'lines.'; // Good var message = [ 'This message', 'goes across several', 'lines.' ].join('');
- Line Length: General rule of thumb is somewhere between 80 and 120 lines of code. No more than 120 characters though. While we don't want to have an explicit standard, just try and be conscious of long lines.
Keeping line length under control has these two main benefits:
* You don't have to scroll horizontally to see the end of the line.
* Side-by-side views are more useful when your code doesn't cover the whole width of your 20" wide-screen monitor.
- Function Length: Even though you may be able to collapse the function contents and logic in your editor, long functions are inherently harder to debug. Longer stack traces are better than longer functions. If your function is starting to look really long then chunk it up into pieces. General rule of thumb: you should be considering refactoring and chunking if the length of your function is about a full screen-height's worth of code.
-
Placement: General rule of thumb: if you think that you won't remember why you did something after a month of vacation, then make a note. If it's something you had to learn how to do and isn't in the beginner's tutorials, then make a note.
-
Formatting: The exception to the below rules would be for documenting libraries. Fancier comments are sometimes necessary depending on your documentation generation strategy -- for example VSDoc.
-
Single line comments should use the double forward slash with a space after the slashes.
//Bad comment // Good comment
-
Multiple line comments should use the block comment style.
/* Comments that span multiple lines. Are better off in a simple, unadorned block. */
-
-
Avoid interrupting cow goes moo. Interrupting comments do not have a place in our world. They can go elsewhere easily and be just as effective.
// Bad function sample(param1, /* OPTIONAL */ param2, param3) { // ... implementation ... } // Good /// <summary> /// Sample function. /// </summary> /// <param name="param1">Param #1</param> /// <param name="param2">Param #2</param> /// <param name="param3" optional="true">Param #3</param> function sample(param1, param2, param3) { // ... implementation ... }
-
Regions are OK to use when supported by your environment.
//#region Sample Code function sample() { } //#endregion
-
Variables: Should be declared at the start of a block and should be declared once. Each block should only have one
var
statement. A side-benefit of this is that if you have a large number of variables clustered at the top, you may need to rethink your function length and chunk it into more manageable pieces.// Bad var x = 2; var y = 2; // Good var x = 2, y = 2; // Bad var x = 2; x = x * 5; var y = 2; // Good var x = 2, y = 2; x = x * 5;
-
Hoisting: Be aware that variable declarations get hoisted to the top of their scope. Function declarations (not function expressions) are hoisted to the top of their scope as well. Function expressions have their host variable name hoisted to the top of their scope, but not their function body or function name.
-
Function Declaration:
function sample() { // ... implementation ... }
-
Function Expression:
var x = function x() { // ... implementation ... }; (function () { // ... implementation ... });
-
-
General Guidelines:
-
Name Choices: Avoid using abbreviations (use
getWindow
instead ofgetWin
). Don't use hungarian notation (strName
) to note the type of the variable. Only use the underscore character to prefix names of private members. -
Reserved Words: Be aware of and do not use reserved words as keys.
-
-
Variables: Use camel casing. Example:
backgroundColor
. -
Properties: Use camel casing. Example:
element.backgroudColor
. -
Constants: Use pascal casing. Example:
var DOMElementId = 'firstName';
. -
File Names: Use camel casing. Example:
contactRibbon.js
. -
Functions as Classes/Constructors: Use pascal casing. Example:
function Dog() { }
. -
Other Functions: Use pascal casing, unless used as a class or constructor. Example:
function createDog() { }
. -
Private Members: Private functions, variables, et cetera should be prefixed with an underscore. Example:
function _executeInternal() { }
.
-
Objects should be defined using literal notation -- it's just cleaner.
// Bad var coords = new Object(); coords.x = 0; coords.y = 0; // Good var coords = { x = 0, y = 0 };
-
Arrays should be defined using literal notation as well.
// Bad var x = new Array(); x.push(2); x.push(3); // Good var x = [2, 3];
-
Namespaces should not be defined as literals. They are messy. (See above.)
-
Anonymous: Anonymous functions (such as may exist in the form of inline callbacks), simply put, should almost never be anonymous. Functions always serve a purpose and thus always deserve a name. Naming all functions will help greatly with stack traces if your code ever encounters an error. If a client sends you a stack trace consisting entirely of anonymous -> anonymous -> anonymous, that won't help you troubleshoot the problem, will it?
// Bad var errorHandler = function() { }; // Good var onError = function onError() { };
-
Closures: Closures are very, very handy but be wary of circular references and the memory leaks they can bring about in IE.
-
Callbacks: If you have a lot of asynchronous calls to make and have a bunch of inline callbacks, you may start to notice a sort of pyramid shape to your code. Try to avoid this and split your callbacks out.
// Bad retrieveDinosaur(function retrievedDinosaur(dino) { if (dino.name === 'Reptar') { retrieveChildren(function retrievedChildren(children) { // ... and the pyramid gets taller and taller from here ... }); } }); // Good function _retrievedDinosaur(dino) { if (dino.name !== 'Reptar') { return; } retrieveChildren(_retrievedChildren); } // ... retrieveDinosaur(_retrievedDinsoaur);
-
Equality: Always use strict comparisons (
===
and!==
) as opposed to type-converting (abstract) comparisons (==
and!=
). The only exception to this rule would be for checking if something is null or undefined.// Bad if (name == 'Reptar') { } // Good if (name === 'Reptar') { }
-
Conditions: Non-trivial conditions should be assigned to a descriptive variable.
// Bad if (name === 'Reptar' && type === 'dinosaur') { // ... implementation ... } // Good var isMatch = (name === 'Reptar' && type === 'dinosaur'); if (isMatch) { // ... implementation ... }
-
String:
typeof variable === 'string'
. -
Number:
typeof variable === 'number'
. -
Boolean:
typeof variable === 'boolean'
. -
Object:
typeof variable === 'object'
. -
Array:
Array.isArray(arrayLikeObject)
. -
Null:
variable === null
. -
Null or Undefined:
variable == null
. -
Undefined:
typeof variable === 'undefined'
. -
Instance of an Object:
classInstance.Type === 'className'
.- Note: We are purposefully avoiding the usage of
instanceof
in this case because it is unsafe across dialogs/iframes. If the class definition is reloaded in another context (or frame/window) and objects from one context are sent to another context, theinstanceof
check may fail. This does require some extra plumbing, but it is minimal.
- Note: We are purposefully avoiding the usage of
-
Type Conversion: When converting a string to an integer using
parseInt()
, always pass in the radix. Therefore instead ofparseInt('2');
useparseInt('2', 10);
. -
Type Coercion: Coercion is OK to use when it makes sense, and if it looks like it might be confusing, change your implementation or wrap it in parenthesis.
// Bad: could be misconstrued as using the '++' operator var x = 0; x = 2 + +document.getElementById('countInput').value; // Good var x = 0; x = 2 + (+document.getElementById('countInput').value);
Namespacing should be used to group functions together and to help avoid pollution of the global namespace. There are several different ways to namespace your code, but it is preferable to create our namespaces and scopes using an Immediately Invoked Function Expressions (IIFE). This is the cleaner way of organizing your code. If you use object literals to define a namespace/block/class-like-object, you have to name all your functions twice. With the IIFE pattern you have better control of private vs public members. Also, as part of good styling keep the invocation of your IIFE inside the parenthesis that contain it.
// Bad
var Container = {
_isOpen: false,
open: function open() {
this._isOpen = true;
}
};
// Good
var Container = (function createNamespace() {
var _isOpen = false;
function open() {
isOpen = true;
}
// Expose public members below
return {
open: open
};
}());
-
Pollution: You too can prevent global namespace pollution. Only add objects/variables to the global namespace if necessary, otherwise keep it in the normal scope. In general wrap all of your code in an IIFE block that is in strict mode (
'use strict';
is set) and pass in a reference to the global object to more vividly define your exports.// Bad var x = 2; function addToX(y) { return x + y; } addToX(1); // Good (function (global) { 'use strict'; global.x = 2; function addToX(y) { return x + y; } addToX(1); }(this));
-
Performance: JavaScript searches up the scope chain for variables and thus globals take the longest to find. Therefore if you must define a global and you use that global variable more than once in a block, assign a local reference to the global.
// Bad var x = 2; (function () { 'use strict'; x = x + 2; // ... x = x + y; }()); // Good var x = 2; (function () { 'use strict'; var count = x; count = count + 2; // ... count = count + y; }());
-
Detail: Keep in mind that you may never expect an error to happen, but when it does, you'll be really frustrated if you didn't put in the extra minute to write a relevant and detailed message.
-
Consistency: If you handle an error with an alert in one place, it should be handled with alerts everywhere. If you use a modal dialog for one error, consider using that everywhere as well.
-
Messaging: When handling errors, note these two great guidelines for messaging:
- Can we tell the user why the error occurred?
- Can we tell the user what to do to fix it?
The answer to these is not always yes, but if either is possible, then they should be included in the messaging.
Do not extend native object prototypes, it will make re-factoring dependent code down the line a lot more difficult. A prime example of this is with Date.js. If you include it, find out it has a problem, then need to swap it out for something like Moment.js, you are going to have the least fun experience of your life trying to hunt down all the places it is used.
If you have the option between asynchronous and synchronous, always try to start down the asynchronous route. If you end up going synchronous, you better have a very good explanation of why. Locking up the User Interface is no good for anyone.
There is about a one in a million chance that you will need to use any of this stuff on a day-to-day basis. Stay clear unless you know for sure what you're doing with them, and even then, probably still stay clear of them.
-
Loops: There are several ways to optimize loops but most of the time nowadays we can shoot for readability unless we really need to ensure our loops are going to be performant. In general we should start with the below and then if we need to be faster, evaluate other options such as decrementing while loops.
// Example var i, len; for (i = 0, len = arrayLikeObject.length; i < len; i++) { }
-
Copying arrays: Use
Array.slice
.// Bad var i, len, itemsCopy = []; for (i = 0, len = items.length; i < len; i++) { itemsCopy[i] = items[i]; } // Good itemsCopy = items.slice();
-
Follow all the other rules already laid out in this document. Things like using type-converting comparison operators can be slower.
Do it and don't let it fall by the wayside, give a good fight for extra time to make sure your code is tested. We prefer to use QUnit and PhantomJS.