This is the 26th day of my participation in the August More Text Challenge
throw
In JavaScript, an exception is thrown whenever a runtime error occurs and when a program explicitly throws an exception using a throw statement. Catch exceptions with try/catch/finally statements. The Error class and its subclasses are used when the JavaScript interpreter itself throws an Error. The Error object has a name attribute that specifies the Error type and a Message attribute that holds the string passed to the constructor. Here is an example function that throws an Error object when called with invalid arguments:
function factorial(x) {
// If the input parameter is invalid, throw an exception!
if (x < 0) throw new Error("x must not be negative");
// Otherwise, a value is computed and returned as normal
let f;
for(f = 1; x > 1; f *= x, x--) /* empty */ ;
return f;
}
factorial(4) / / = > 24
Copy the code
When an exception is thrown, the JavaScript interpreter immediately stops normal program execution and jumps to the nearest exception handler. Exception handlers are written using the catch clause of a try/catch/finally statement. If the block of code that raises the exception does not have an associated catch clause, the interpreter checks for the next highest block of closed code to see if there is an exception handler associated with it. Until the handler is found. If an exception is thrown ina function that does not contain a try/catch/finally statement, the exception is propagated to the code calling the function. In this way, exceptions are propagated up the call stack through the lexical structure of JavaScript methods. If no exception handler is found, the exception is treated as an error and reported to the user.
try/catch/finally
Try /catch/finally statements are JavaScript’s exception handling mechanism. The try clause of this statement defines only the block of code to handle its exceptions. The try block is followed by the catch clause, which is the block of statements that is called whenever an exception occurs anywhere in the try block. The catch clause is followed by the finally block, which contains the cleanup code that guarantees execution regardless of what happens in the try block. Both catch and finally blocks are optional, but try blocks must come with at least one such block. Try, catch, and finally blocks all begin and end with braces. These braces are a required part of the syntax and cannot be omitted, even if the clause contains only one statement.
try {
Normally, this code runs from the top of the block to the bottom
// No problem. But sometimes exceptions can be thrown,
// You can use the throw statement directly or by calling it
// The method to throw an exception.
}
catch (e) {
// Statements in the try block are executed if and only if the block throws an exception.
// These statements can use the local variable e to refer to thrown error objects or other values.
// This block can handle exceptions in a way that ignores exceptions by doing nothing,
// Or you can rethrow the exception by throwing.
}
finally {
// This block contains statements that are always executed, regardless of what happens in the try block.
// Whether the try blocks are terminated or not, they are executed:
// 1) Usually, after reaching the bottom of the block
// 2) Due to a break, continue, or return statement
// 3) An exception is handled by the catch clause above
// 4) There is an uncaught exception that is still propagating
}
Copy the code
An identifier is usually used in parentheses after the catch keyword. This identifier is similar to a function parameter. When an exception is caught, the value associated with the exception (for example, an Error object) is assigned to this parameter. The identifier associated with the catch clause has block scope, which is defined only in the catch block.
Here is an example of a try/catch statement in action. Use the factorial() method defined in throw above:
try {
// Ask the user to enter a number
let n = Number(prompt("Please enter a positive integer".""));
// Assuming the input is valid, compute the factorial of the number
let f = factorial(n);
// Display the result
alert(n + ! "" =" + f);
}
catch (ex) { // If the user input is invalid, we execute here
alert(ex); // Tell the user what the error is
}
Copy the code
This example is a try/catch statement with no finally clause. Although finally is not used as often as catch, it can be useful. However, its behavior requires additional explanation. The finally clause is executed whenever any part of the try block is executed, regardless of how the code in the try block completes. It usually cleans up after the code in the try clause executes.
Under normal circumstances, the JavaScript interpreter reaches the end of the try block and then enters the finally block, which performs any necessary cleanup. If the interpreter leaves the try block because of a return, continue, or break statement, the finally block is executed before the interpreter jumps to the new target.
If an exception occurs ina try block and there is an associated catch block to handle the exception, the interpreter executes the catch block first, then the finally block. If there is no local catch block to handle the exception, the interpreter first executes the finally block and then propagates the exception up until it finds a catch block that can handle the exception.
If the finally block itself causes a jump with a return, continue, break, or throw statement, or by calling a method that throws an exception, the interpreter dismisses any pending jump and executes a new jump. For example, if the finally clause throws an exception, that exception replaces any exceptions that are being thrown. If the finally clause issues a return statement, the method returns normally, even if an exception has been thrown and not yet handled.
Try and finally can be used together without a catch clause. In this case, the finally block simply guarantees that the cleanup code executes, regardless of what happens in the try block. Recall that we cannot simulate the for loop completely with the while loop because the continue statement behaves differently for the two loops. If you add a try/finally statement, you can write a while loop that works like a for loop and handles the continue statement correctly:
// initialize for(initialize; test ; increment ) body;
initialize ;
while( test ) {
try { body ; }
finally{ increment ; }}Copy the code
Note, however, that the body containing the break statement behaves slightly differently in the while loop than it does in the for loop (resulting in additional increment before exiting), so it is impossible to simulate the for loop completely with the while, even with the finally clause.
An empty catch statement
Sometimes, you might find yourself using only the catch clause to detect and stop the propagation of an exception, regardless of the type or value of the exception. In ES2019 and later, it is possible to omit parentheses and identifiers and use the catch keyword only in the following example:
Parse () like json.parse () but returns undefined instead of throwing an exception
function parseJSON(s) {
try {
return JSON.parse(s);
} catch {
// There is something wrong, but we don't care what it is
return undefined; }}Copy the code