When should I use a ThrowHelper method instead of throwing directly?

前端 未结 6 1738
傲寒
傲寒 2021-02-05 10:16

When is it appropriate to use a ThrowHelper method instead of throwing directly?

void MyMethod() {
    ...
    //throw new ArgumentNullException(\"param         


        
相关标签:
6条回答
  • 2021-02-05 10:28

    From what I've understood, the smaller bytecode is pretty much the only advantage (See this question), so I don't expect you'll see a lot of arguments for it in the answers here.

    0 讨论(0)
  • 2021-02-05 10:28

    I always try to throw my own exceptions for the mere purpose of customization. I can throw a message that is going to tell me in short order what the problem is (and possibly where it originated from). As far as performance issues go, I try to limit the number of exceptions thrown on the release versions of my software as much as possible, so I have never actually thought about it too much. I am interested to see what other say though.

    That is my 2 cents. Please take it as such.

    0 讨论(0)
  • 2021-02-05 10:32

    My default approach is to throw directly from the exceptional code branch.

    The DRY principle and the rule of 3 guides when I would wrap that in a method: if I find myself writing the same 'throw' code 3 or more times, I consider wrapping it in a helper method.

    However, instead of a method that throws, it's much better to write a Factory Method that creates the desired Exception and then throw it from the original place:

    public void DoStuff(string stuff)
    {
        // Do something
    
        throw this.CreateException("Boo hiss!");
    }
    
    private MyException CreateException(string message)
    {
        return new MyException(message);
    }
    

    This preserves the stack trace.

    0 讨论(0)
  • 2021-02-05 10:36

    This seems like a needless abstraction to me. You have a method that does one thing, named as verbosely as the thing that it does.

    The argument about less bytecode is practically meaningless since the size of your code rarely matters (you wouldn't be saving more than a kilobyte per instance of your program, unless you throw that exception from a ridiculous number of places in your source code). Meanwhile your stated drawbacks are all real concerns, especially where clarity of exception handling is concerned.

    Basically abstracting away minor things like this usually comes back to bite you. (Some entertaining reading on the subject: http://www.joelonsoftware.com/articles/LeakyAbstractions.html)

    0 讨论(0)
  • 2021-02-05 10:45

    I would say that the only reasonable time to do this is in something like the BCL where the amount of usage of the class is wide spread enough that the savings may be worth it. It would have to make sense, however. In your case, I don't think you're actually saving any space. Instances of ThrowHelper in the BCL reduce size by substituting enums for strings and method calls (to get string messages). Your case simply passes the same arguments and thus doesn't achieve any savings. It costs just as much to call your method as it does to throw the exception in place.

    0 讨论(0)
  • 2021-02-05 10:50

    One advantage not yet mentioned to using either a throw-helper or exception-factory method is the possibility of passing a delegate for such a purpose. When using a throw-helper delegate, one gains the possibility of not throwing (which would in some cases be a good thing; in other cases, a bad thing). For example:

    string TryGetEntry(string key, Funct<problemCause, string> errorHandler)
    {
      if (disposed)
        return errorHandler(problemCause.ObjectDisposed);
      else if (...entry_doesnt_exist...)
        return errorHandler(problemCause.ObjectNotFound);
      else
        return ...entry...;
    }
    string GetEntry(string key)
    {
      return TryGetEntry(key, ThrowExceptionOnError);
    }
    bool TryGetEntry(string key, ref result)
    {
      bool ok;
      string result;
      result = TryGetEntry(key, (problemCause theProblem) => {ok=false; return (string)null;});
      return ok;
    }
    

    Using this approach, one may easily use one routine with a variety of error-handling strategies. One could eliminate the need for a closure by having the TryGetEntry method accept a generic type parameter along with a 'ref' parameter of that type and a delegate that accepts such a 'ref' parameter; the example is simpler, though, just using a closure.

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