See below:

> On Feb 21, 2021, at 3:17 PM, Ian Lance Taylor <i...@golang.org> wrote:
> 
> On Sun, Feb 21, 2021 at 9:23 AM robert engels <reng...@ix.netcom.com> wrote:
>> 
>> Can someone please explain the benefit of ‘error return’ over ‘checked 
>> exceptions’ ? I have made the point a few times and it goes to crickets - I 
>> have to believe that is because there is none, or it is difficult to 
>> communicate.
> 
> As I wrote earlier, I believe that experience shows that it is
> extremely difficult to write a large scale program with multiple
> maintainers that uses checked exceptions.  In practice, programs
> always fall back to unchecked exceptions.  So I would say that one
> benefit of error return over checked exceptions is simply that error
> returns scale to large programs and many maintainers.
> 
I don’t think answering, “people fall back to unchecked exceptions” is an 
answer to “how do error returns and checked exceptions differ”.

> Another benefit is described by Rob's blog post, which I'm sure you've
> seen: Errors are values (https://blog.golang.org/errors-are-values).
> 
> More generally, errors are not exceptional conditions.  They are part
> and parcel of how programs work.  They are part of the normal program
> flow, and they should be part of that flow.  They shouldn't be
> regarded as exceptional and unusual flow.
> 
Yes, with all due respect to Rob, we'll probably disagree on that blog post. 
Take the Scan() example. This is a justification of how to write code that 
isn’t painful when you don't have exceptions - instead it puts the burden on 
the writer of Scan() to set a boolean internally to ‘don’t do anything’, write 
an additional method Err(), check the boolean on all other methods, etc., then 
the caller needs to remember to call Err() after all processing - and the code 
might loop through a billion elements doing nothing - potentially creating huge 
inefficiencies. Lastly, a reader of the code where the developer had failed to 
call Err() would not know unless they were intimately aware of the particular 
API - this is not good for long term maintenance. 

I can provide examples for each case in the blog as to why it seems more 
'defensive justification’ rather than evidence of being better.

> I'm not going to claim that any of these benefits are slam dunks.  If
> there were obvious solutions to how to handle errors, nobody would
> still be discussing different approaches.  But I am claiming that Go's
> approach to error handling has real benefits.
> 
> 
>> The ‘handle where they occur claim’ is weak, as you can handle exceptions at 
>> the call site with similar LOC and constructs - but why when in most cases 
>> (of non-trivial software) the error value is just returned up. No offense to 
>> Wojciech - the rpc transport layer doesn’t do the retires, it passes up an 
>> error, and a higher level performs retry logic, if all retries fail, it will 
>> pass up another error that the rpc failed, etc. This pattern is used 
>> throughout well designed software. If you hard using this pattern you are 
>> creating very hard to maintain software.
> 
> I can't agree that in most cases the error is just returned up.  I
> would agree that within a single package an error is often (but not
> always) returned up without change.  But in large Go programs that
> I've seen, when error values cross package boundaries, they are almost
> always annotated with additional information.
> 
> 

There is no difference between

if err != nil {
        return someWrappingMethod(err)
}

and

catch (SomeException e) {
        throw SomeOtherException(e);
}

but, if you don’t want or can’t provide additional information you don’t need 
to do anything, BUT you still have to declare that your method throws the 
exception - this is an important benefit.

and if you have several methods or loops that throw the exception you can move 
the handler to outside - in Go this is plain tiresome and ugly - the blog post 
tries to handle this with the Scan() and ErrWriter() - again - not handling the 
errors as they occur (so no, Go code doesn’t ‘read’ linearly - many of the LOC 
may become noops interspersed with other non-noop code which is a nightmare).
        
Exceptions as advanced error values as more benefits:

   1. the developer needs to declare what exceptions the method throws if any
       - this is unbelievably valuable for efficient development and 
maintenance - rather than trying to read through documentation (hopefully up to 
date) or code to try and figure out all of the possible errors a method will 
return and what they mean
   2. the ‘value’ of an exception contains a lot more information by default
        - often no need to wrap, as the stack trace provides sufficient 
information, and a ‘developer label’ can easily become out of date


>> Checked exceptions are ‘error values’ with more information (e.g stack 
>> traces, causes, etc.), and can be enforced by the compiler (declared) and 
>> inspected by modern IDE’s for code complete. ‘error returns’ have none of 
>> these features. The only valid claim is that they are more expensive than 
>> error values, BUT you should not be using exceptions - or error values - for 
>> flow control. This is the fundamental problem to me with Go’s error handling 
>> - the flow control and errors are mixed leading to very difficult code to 
>> read & validate that all potential paths are handled now and in the future.
> 
> My opinion, which contradicts yours, is that when the flow control is
> written down on the page, as it is in Go, the flow is easy to read,
> and it is much easier to check that all potential paths are handled.
> With exceptions the control flow is not written down, so it is easier
> to miss cases where an exception can arise.  With exceptions the
> normal control flow is easier to see, but the error control flow is
> harder.  And since the error control flow happens less frequently,
> it's more important to make it clearly visible to the original
> programmer and, more importantly, to the future maintainer.
> 
> Again, no slam dunks here.  Reasonable people can disagree.  But I
> think the Go position is entirely defensible for large scale programs
> with many programmers.
> 

I agree. I still believe that ‘checked exceptions’ are just ‘better error 
values’. As to ‘unchecked exceptions’ Go falls back on unchecked exceptions all 
over the place - otherwise the it would be incredibly painful for the developer 
to check every allocation or memory access.

Regardless, I don’t think Go can afford to rely on third-party packages for 
something as critical as error handling. These packages, even if limited, need 
to be made part of the core stdlib in order to facilitate easier long-term 
maintenance. These packages can be viewed as  ‘exceptions light’ or ‘better 
errors’.

So even if Go needs to pick it now, something is better than nothing (and 
increased fragmentation).

R


> Ian

-- 
You received this message because you are subscribed to the Google Groups 
"golang-nuts" group.
To unsubscribe from this group and stop receiving emails from it, send an email 
to golang-nuts+unsubscr...@googlegroups.com.
To view this discussion on the web visit 
https://groups.google.com/d/msgid/golang-nuts/5AADD753-20C0-4C2B-90AE-4CEBFD5A1FB6%40ix.netcom.com.

Reply via email to