Custom Exceptions in Python


Python allows developers to define their own exceptions by creating custom exception classes. This is useful when the built-in exceptions do not cover specific error scenarios in your application. Custom exceptions provide a more descriptive and meaningful way to handle errors.

Defining a Custom Exception

To create a custom exception, you define a new class that inherits from Python's built-in Exception class. Here is the basic structure:

    class CustomException(Exception):
        pass
        

By inheriting from Exception, the custom class becomes a valid exception that can be raised and handled using try and except.

Example 1: Basic Custom Exception

    class CustomError(Exception):
        pass

    try:
        raise CustomError("This is a custom error")
    except CustomError as e:
        print(f"Caught an exception: {e}")
        

In this example, the CustomError exception is raised with a custom error message and handled in the except block.

Adding Custom Attributes

You can add custom attributes to your exception class to store additional information about the error.

Example 2: Custom Exception with Attributes

    class InvalidInputError(Exception):
        def __init__(self, value, message="Invalid input provided"):
            self.value = value
            self.message = message
            super().__init__(self.message)

    try:
        raise InvalidInputError("abc", "Input must be a number")
    except InvalidInputError as e:
        print(f"Error: {e.message}. Value: {e.value}")
        

Here, the custom exception stores the invalid value and a detailed message, making the error handling more informative.

Customizing the String Representation

You can override the __str__ method in your custom exception class to define a custom string representation for the error.

Example 3: Overriding the String Representation

    class DivisionError(Exception):
        def __init__(self, numerator, denominator):
            self.numerator = numerator
            self.denominator = denominator

        def __str__(self):
            return f"Cannot divide {self.numerator} by {self.denominator}"

    try:
        raise DivisionError(10, 0)
    except DivisionError as e:
        print(e)
        

In this example, the __str__ method customizes the error message, providing a clear explanation of the error.

Using Custom Exceptions in Functions

Custom exceptions are especially useful in functions to validate input or enforce business logic.

Example 4: Custom Exception in a Function

    class AgeError(Exception):
        pass

    def validate_age(age):
        if age < 0:
            raise AgeError("Age cannot be negative")
        print(f"Valid age: {age}")

    try:
        validate_age(-5)
    except AgeError as e:
        print(f"Validation error: {e}")
        

Here, the AgeError exception is raised if the provided age is negative, ensuring that the input is validated before proceeding.

Subclassing Custom Exceptions

You can create a hierarchy of exceptions by subclassing your custom exception class. This is useful for grouping related errors.

Example 5: Exception Hierarchy

    class ApplicationError(Exception):
        pass

    class DatabaseError(ApplicationError):
        pass

    class FileNotFoundError(ApplicationError):
        pass

    try:
        raise DatabaseError("Database connection failed")
    except ApplicationError as e:
        print(f"Application error: {e}")
        

In this example, DatabaseError and FileNotFoundError are subclasses of ApplicationError. The exception hierarchy allows you to catch specific errors or general application errors.

Key Points

  • Custom exceptions are created by inheriting from the Exception class.
  • You can add attributes and methods to provide additional error details.
  • Custom exceptions make your code more descriptive and easier to debug.
  • Exception hierarchies help in organizing and handling related errors efficiently.

Conclusion

Custom exceptions are a powerful feature in Python that allow you to define and handle application-specific errors in a clear and meaningful way. By using custom exception classes, you can make your code more robust, maintainable, and user-friendly.





Advertisement