When to throw an exception?

后端 未结 30 2038
后悔当初
后悔当初 2020-11-21 23:48

I have exceptions created for every condition that my application does not expect. UserNameNotValidException, PasswordNotCorrectException etc.

相关标签:
30条回答
  • 2020-11-22 00:19

    I agree with japollock way up there--throw an acception when you are uncertain about the outcome of an operation. Calls to APIs, accessing filesystems, database calls, etc. Anytime you are moving past the "boundaries" of your programming languages.

    I'd like to add, feel free to throw a standard exception. Unless you are going to do something "different" (ignore, email, log, show that twitter whale picture thingy, etc), then don't bother with custom exceptions.

    0 讨论(0)
  • 2020-11-22 00:20

    Firstly, if the users of your API aren't interested in specific, fine-grained failures, then having specific exceptions for them isn't of any value.

    Since it's often not possible to know what may be useful to your users, a better approach is to have the specific exceptions, but ensure they inherit from a common class (e.g., std::exception or its derivatives in C++). That allows your client to catch specific exceptions if they choose, or the more general exception if they don't care.

    0 讨论(0)
  • 2020-11-22 00:23

    It is NOT an exception if the username is not valid or the password is not correct. Those are things you should expect in the normal flow of operation. Exceptions are things that are not part of the normal program operation and are rather rare.

    EDIT: I do not like using exceptions because you can not tell if a method throws an exception just by looking at the call. Thats why exceptions should only be used if you can't handle the situation in a decent manner (think "out of memory" or "computer is on fire").

    0 讨论(0)
  • 2020-11-22 00:25

    I would say there are no hard and fast rules on when to use exceptions. However there are good reasons for using or not using them:

    Reasons to use exceptions:

    • The code flow for the common case is clearer
    • Can return complex error information as an object (although this can also be achieved using error "out" parameter passed by reference)
    • Languages generally provide some facility for managing tidy cleanup in the event of the exception (try/finally in Java, using in C#, RAII in C++)
    • In the event no exception is thrown, execution can sometimes be faster than checking return codes
    • In Java, checked exceptions must be declared or caught (although this can be a reason against)

    Reasons not to use exceptions:

    • Sometimes it's overkill if the error handling is simple
    • If exceptions are not documented or declared, they may be uncaught by calling code, which may be worse than if the the calling code just ignored a return code (application exit vs silent failure - which is worse may depend on the scenario)
    • In C++, code that uses exceptions must be exception safe (even if you don't throw or catch them, but call a throwing function indirectly)
    • In C++, it is hard to tell when a function might throw, therefore you must be paranoid about exception safety if you use them
    • Throwing and catching exceptions is generally significantly more expensive compared to checking a return flag

    In general, I would be more inclined to use exceptions in Java than in C++ or C#, because I am of the opinion that an exception, declared or not, is fundamentally part of the formal interface of a function, since changing your exception guarantee may break calling code. The biggest advantage of using them in Java IMO, is that you know that your caller MUST handle the exception, and this improves the chance of correct behaviour.

    Because of this, in any language, I would always derive all exceptions in a layer of code or API from a common class, so that calling code can always guarantee to catch all exceptions. Also I would consider it bad to throw exception classes that are implementation-specific, when writing an API or library (i.e. wrap exceptions from lower layers so that the exception that your caller receives is understandable in the context of your interface).

    Note that Java makes the distinction between general and Runtime exceptions in that the latter need not be declared. I would only use Runtime exception classes when you know that the error is a result of a bug in the program.

    0 讨论(0)
  • 2020-11-22 00:27

    The simple answer is, whenever an operation is impossible (because of either application OR because it would violate business logic). If a method is invoked and it impossible to do what the method was written to do, throw an Exception. A good example is that constructors always throw ArgumentExceptions if an instance cannot be created using the supplied parameters. Another example is InvalidOperationException, which is thrown when an operation cannot be performed because of the state of another member or members of the class.

    In your case, if a method like Login(username, password) is invoked, if the username is not valid, it is indeed correct to throw a UserNameNotValidException, or PasswordNotCorrectException if password is incorrect. The user cannot be logged in using the supplied parameter(s) (i.e. it's impossible because it would violate authentication), so throw an Exception. Although I might have your two Exceptions inherit from ArgumentException.

    Having said that, if you wish NOT to throw an Exception because a login failure may be very common, one strategy is to instead create a method that returns types that represent different failures. Here's an example:

    { // class
        ...
    
        public LoginResult Login(string user, string password)
        {
            if (IsInvalidUser(user))
            {
                return new UserInvalidLoginResult(user);
            }
            else if (IsInvalidPassword(user, password))
            {
                return new PasswordInvalidLoginResult(user, password);
            }
            else
            {
                return new SuccessfulLoginResult();
            }
        }
    
        ...
    }
    
    public abstract class LoginResult
    {
        public readonly string Message;
    
        protected LoginResult(string message)
        {
            this.Message = message;
        }
    }
    
    public class SuccessfulLoginResult : LoginResult
    {
        public SucccessfulLogin(string user)
            : base(string.Format("Login for user '{0}' was successful.", user))
        { }
    }
    
    public class UserInvalidLoginResult : LoginResult
    {
        public UserInvalidLoginResult(string user)
            : base(string.Format("The username '{0}' is invalid.", user))
        { }
    }
    
    public class PasswordInvalidLoginResult : LoginResult
    {
        public PasswordInvalidLoginResult(string password, string user)
            : base(string.Format("The password '{0}' for username '{0}' is invalid.", password, user))
        { }
    }
    

    Most developers are taught to avoid Exceptions because of the overhead caused by throwing them. It's great to be resource-conscious, but usually not at the expense of your application design. That is probably the reason you were told not to throw your two Exceptions. Whether to use Exceptions or not usually boils down to how frequently the Exception will occur. If it's a fairly common or an fairly expectable result, this is when most developers will avoid Exceptions and instead create another method to indicate failure, because of the supposed consumption of resources.

    Here's an example of avoiding using Exceptions in a scenario like just described, using the Try() pattern:

    public class ValidatedLogin
    {
        public readonly string User;
        public readonly string Password;
    
        public ValidatedLogin(string user, string password)
        {
            if (IsInvalidUser(user))
            {
                throw new UserInvalidException(user);
            }
            else if (IsInvalidPassword(user, password))
            {
                throw new PasswordInvalidException(password);
            }
    
            this.User = user;
            this.Password = password;
        }
    
        public static bool TryCreate(string user, string password, out ValidatedLogin validatedLogin)
        {
            if (IsInvalidUser(user) || 
                IsInvalidPassword(user, password))
            {
                return false;
            }
    
            validatedLogin = new ValidatedLogin(user, password);
    
            return true;
        }
    }
    
    0 讨论(0)
  • 2020-11-22 00:29

    Others propose that exceptions should not be used because the bad login is to be expected in a normal flow if the user mistypes. I disagree and I don't get the reasoning. Compare it with opening a file.. if the file doesn't exist or is not available for some reason then an exception will be thrown by the framework. Using the logic above this was a mistake by Microsoft. They should have returned an error code. Same for parsing, webrequests, etc., etc..

    I don't consider a bad login part of a normal flow, it's exceptional. Normally the user types the correct password, and the file does exist. The exceptional cases are exceptional and it's perfectly fine to use exceptions for those. Complicating your code by propagating return values through n levels up the stack is a waste of energy and will result in messy code. Do the simplest thing that could possibly work. Don't prematurely optimize by using error codes, exceptional stuff by definition rarely happens, and exceptions don't cost anything unless you throw them.

    0 讨论(0)
提交回复
热议问题