Typing: tryCatch
Introduction
The tryCatch()
function is a useful utility that allows developers to execute code that may potentially throw exceptions safely. It provides a structured and type-safe way to handle exceptions by wrapping the result in a Result
type (more on Result
type: Typing: Result).
Exception handling is an essential aspect of writing robust and reliable code. However, if not appropriately handled, traditional exception-handling mechanisms, such as try-catch
blocks, can sometimes lead to unhandled exceptions or unexpected crashes. The tryCatch()
function addresses this issue by encapsulating the execution of code within a controlled environment.
By using the tryCatch()
function, developers can execute code that may throw exceptions without the risk of unhandled exceptions propagating up the call stack. Instead of allowing exceptions to crash the program, the function catches any thrown exceptions and wraps them in an Error
instance of the Result
class. This ensures that exceptions are captured and can be handled in a controlled manner.
The tryCatch()
function is particularly useful when developers want to handle exceptions gracefully and provide alternative behavior or error-handling mechanisms. By returning a Result
type, the function allows the caller to quickly determine whether the operation was successful or resulted in an error. This promotes a more structured and predictable control flow, making the code more robust and easier to reason.
Additionally, the tryCatch()
function provides access to the caught exception and stack trace through the Error
instance. This allows developers to inspect and analyze the exception, enabling them to log relevant information, perform error recovery, or take appropriate actions based on the exception type.
Overall, the tryCatch()
function enhances the error-handling capabilities of Dart by providing a structured and type-safe approach to handle exceptions. It promotes code reliability, maintainability, and robustness by encapsulating exception-prone code and allowing for controlled error handling and alternative behavior.
Implementation
import 'result.dart';
Result<T, ({Object ex, StackTrace st})> tryCatch<T>(T Function() operation) {
try {
T result = operation();
return Ok(result);
} catch (e, s) {
return Error((ex: e, st: s));
}
}
void main() {
// this will wrap exception in Result type
var res = tryCatch<bool>(() {
dynamic foo = true;
return foo++;
});
print(res.isError()); // Output: true
print(res.isOk()); // Output: false
res.tapError((err) {
print(err.ex.runtimeType); // Output: NoSuchMethodError
print(err.st.runtimeType); // Output: _StackTrace
});
}
Explanation of the code:
- The code begins by importing the
Result
class from a file namedresult.dart
. This suggests that theResult
class is defined in a separate file and is being used in this code. - The
tryCatch
function is defined, which takes a parameteroperation
representing a function that returns a value of typeT
. This function attempts to execute theoperation
and returns aResult
instance. The result is wrapped in an Ok value if theoperation
executes successfully. If an exception occurs, the exception and the associated stack trace are wrapped in anError
value. - The
main
function is defined, which serves as the program’s entry point. - The
tryCatch
function is called to wrap an exception in aResult
type. In this case, theoperation
is an anonymous function that attempts to increment a variablefoo
of typedynamic
, which will result in aNoSuchMethodError
exception. - The
isError()
andisOk()
methods are called on theresError
instance to check if it represents an error or success. The output indicates that the result is an error. - The
tapError()
method is called on theresError
instance to perform a side effect on the error value. Inside the callback function, theex
property of the error is accessed using theerr
parameter, and its runtime type is printed. Similarly, the runtime type of the stack trace (st
) is also printed. The output shows that the exception is of typeNoSuchMethodError
and the stack trace is of type_StackTrace
. - The
mapError()
method is called on theresError
instance to transform the error value. In this case, the error value is mapped to theex
property of the error. ThetapError()
method is then called to perform a side effect on the transformed error value. The runtime type of the transformed error value is printed, which matches the original exception type (NoSuchMethodError
). - The
tryCatch
function is called again to wrap a successful operation in aResult
type. In this case, theoperation
is an anonymous function that converts the string “Hello” to lowercase. - The
map()
method is called multiple times on theresOk
instance to transform the success value. Eachmap()
call applies a different transformation to the value. Thetap()
method is then called to perform a side effect on the final transformed value. The transformed value is printed, which is “HELLOHELLO!”.
This code demonstrates the usage of the tryCatch
function to wrap exceptions in a Result
type. It showcases various methods available on the Result
instances, such as isError()
, isOk()
, tapError()
, mapError()
, map()
, and tap()
. These methods allow for safe and expressive handling of success and error values, enabling developers to handle exceptions and transform values in a structured and type-safe manner.