On Sat, Feb 24, 2018 at 11:04 PM Bakul Shah <ba...@bitblocks.com> wrote:

> r := os.Open(filename)
> if isError(r) { return r.(error) }
> f := r.(*os.File) // or better: var f *os.File; f = r
>

This still does not seem meaningfully different to me, though.
* You are writing isError(r), instead of err != nil
* You are writing r.(error) instead of err
* You are writing f := r.(*os.File), but is that so different from just
using f?

Yes, the compiler will prevent you from using the result without a
type-assertion. That's a real benefit. But given that you are not really
meaningfully change the amount of boilerplate and the benefit of detecting
a possible bug can also be achieved using errcheck, I agree with Roger that
this doesn't seem that meaningful an improvement to justify adding a new
language feature.



>
> Error checking being a common pattern, isError() can be added as a builtin
> or trivially in errors pkg. Likely the enclosing function also returns an
> error
> so the return in the second line above can be just return r but with the
> current product type approach you’d have return nil, err.
>
> You are only looking at code after returning. Code within a function
> benefits
> more (gets simplified).
>
> func f(s string) (int|error) { // instead of (int,error)
>   ...
>   return err // instead of return 0,err
>   ...
>   return 1 // instead of return 1,nil
>   // and return n, err case disappears
>
> Having to return an extra thing that gets discarded right away is annoying.
> Early on this was a common mistake for me. We are used to it now but the
> current idiom came about because of a lack of sum types. People can use
> object{} as a return type today but that more or less defeats type checking
> which may be why returning an extra thing was perhaps seen as acceptable.
> So yes, I do think sum types benefit here. May be not enough for people to
> want a change but net benefit nonetheless.
>
> The bigger benefit is their use where people currently end up using
> object{}
> or product type. With sum types you get better documentation and checking.
> If you think of sum types as just restricted object{} types, they are
> simple to
> explain and implement. Simpler than what you guys were discussing on issue
> 19412 thread. I will add a comment there.
>
> On Feb 24, 2018, at 12:03 PM, roger peppe <rogpe...@gmail.com> wrote:
>
> >> I once counted there were about 8000 uses of <type>,error as
> >> return types for functions in $GOROOT/src which could benefit
> >> from sum types.
> >
> > I'm not entirely convinced that sum types are a net win as a
> > substitute for (T, error) return values.
> >
> > Where currently we have:
> >
> >      f, err := os.Open(filename)
> >      if err != nil {
> >          return err
> >      }
> >      // use f
> >
> > we'd have to do something like:
> >
> >      fOrError := os.Open(filename)
> >      if err, ok := fOrError.(error); ok {
> >           // ... but what happens if we want to return a distinct
> >           // value that implements error?
> >           return err
> >      }
> >      f := fOrError.(*os.File)
> >      // use f
> >
> > or perhaps:
> >
> >      var f *os.File
> >      switch fOrError := os.Open(filename).(type) {
> >      case *os.File:
> >          f = fOrError
> >      case error:
> >          return fOrError
> >      }
> >      // use f
> >
> > or even, defensively (perhaps os.Open's type might change to admit
> > more possibilities:
> >
> >      var f *os.File
> >      switch fOrError := os.Open(filename).(type) {
> >      case *os.File:
> >          f = fOrError
> >      case error:
> >          return fOrError
> >      default:
> >          panic("unexpected type %T returned from os.Open", fOrError)
> >          // or return error
> >      }
> >      // use f
> >
> > I don't know about you, but I wouldn't consider any of those a
> > particular improvement on the current idiom. They're all more verbose
> > and not actually that much more type-safe.
> >
> > That doesn't mean that I don't think sum types are a decent idea, but
> > just that I'm not sure that they're as clear a win for this (a
> > commonly cited use case) as one might think.
> >
> >  cheers,
> >    rog.
> >
> >
> >> On 24 February 2018 at 01:23, Bakul Shah <ba...@bitblocks.com> wrote:
> >>> On Thu, 22 Feb 2018 12:55:01 +0000 Jesper Louis Andersen <
> jesper.louis.ander...@gmail.com> wrote:
> >>>
> >>> For sums, in Go, I have to return a pair,
> >>>
> >>> x, err := someOperation(..)
> >>>
> >>> but this is slightly inconsistent, insofar I can return "nil, nil",
> which
> >>> might not be a valid value, or I can return "3, Error(..)" but in that
> case
> >>> the 3 is not to be used. Only one "side" is valid at a given point in
> time.
> >>> If you have sum-types in the usual sense, you can define something
> along
> >>> the lines of (OCaml follows):
> >>>
> >>> type ('a, 'b) result = Ok of 'a | Error of 'b
> >>
> >> I once worked out some details of adding sum types to Go and I
> >> think it is quite doable and easy to implement. For example,
> >>
> >>    func f(i int) float64|error {
> >>        if i == 0 { return errors.New("not zero") }
> >>        return 1./float64(i)
> >>    }
> >>
> >> As in OCaml "|" is used for sum types and it binds less
> >> tightly than existing type "expressions".
> >>
> >>> And then you can discriminate on this value via pattern matching
> >>>
> >>> match res with
> >>> | Ok value -> ...
> >>> | Error err -> ...
> >>
> >> Not quite the same but something similar is doable with
> >> type switch.
> >>
> >>    res := f(j)
> >>    switch res.(type) {
> >>    case error: ...
> >>    case string: ...
> >>    }
> >>
> >> This use is identical with f returning interface{} (even the f
> >> body remains exactly the same).  This makes sense since
> >> interface{} is in a sense the sum of all other types.
> >>
> >> But by allowing sum type, we can do better checking within f
> >> (e.g. return "string" will fail to compile). And by using a
> >> sum type instead of a product type to return a value or error
> >> also makes the code clearer.
> >>
> >> I once counted there were about 8000 uses of <type>,error as
> >> return types for functions in $GOROOT/src which could benefit
> >> from sum types. Now there seem to be about 4253 instances
> >> (found using a crude regexp).
> >>
> >> I think I worked out the semantics of T1|T2 where T1 and T2
> >> are both interface types themselves. It all seem to fit
> >> together, at least on paper! I need to find my notes....
> >>
> >>> The other matching construction is a switch-statement, but such a
> statement
> >>> doesn't allow for matching deeply into an AST structure, which a
> >>> traditional pattern matcher does.
> >>
> >> Deeper matching also binds names to matched parts. e.g.
> >>
> >>    sumsqr [] = 0
> >>    sumsqr (x:xs) = x*x + sumsqr xs
> >>
> >> This sort of binding may be difficult to shoehorn into Go.
> >> There may be no real benefit of binding head, tail of a slide
> >> but consider an AST. If you are already cracking it open for
> >> matching it with a pattern, you may as well bind variables to
> >> interesting parts.
> >>
> >>    match stmt {
> >>    case If1stmt(test:, thenstmt:): ...
> >>    case If2stmt(test:, thenstmt:, elsestmt:): ...
> >>    ...
> >>    }
> >>
> >> Hard to come up with an intuitive syntax here.  Also probably
> >> impossible  to add func level patterns.
> >>
> >>> Coincidentally, given sum-types, you can write a regexp matching
> engine in
> >>> very few lines of code. See Bob Harper's "Programming in Standard ML"
> [2]
> >>> for example; it is the introductory example to get a feel for the
> language.
> >>> The solution uses sum types to define the AST for regular expressions,
> and
> >>> then uses pattern matching to build a matcher on regular expressions. I
> >>> can't remember how far Bob takes the exposition however.
> >>
> >> This should be doable given my sum-type proposal sketch above?!
> >>
> >>> [0] They are really the same in the right setting. In a boolean
> algebra,
> >>> for instance, + is OR and * is AND. If you look at them from the
> Category
> >>> Theory branch of mathematics they are related: they are duals of each
> other
> >>> which means that if you "invert" one you get the other and vice versa.
> >>>
> >>> [1] Obviously, Go, being a descendant of Algol has two syntactic
> classes:
> >>> statements and expressions, whereas OCaml is descendant from either
> Lisp or
> >>> the Lambda Calculus depending on view. Those languages only have
> >>> expressions as a syntactic class.
> >>
> >> sum-types (which are easy to implement) and pattern matching
> >> (may be not so easy) can be added to Go even if it has
> >> non-expression syntatic classes.
> >>
> >>> [2] http://www.cs.cmu.edu/~rwh/isml/book.pdf
> >>>
> >>> [3]
> >>>
> https://www.microsoft.com/en-us/research/wp-content/uploads/1987/01/slpj-book-1987-small.pdf
> >>
> >> SLPJ's book was quite an eye opener for me back in the '80s!
> >>
> >> --
> >> 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.
> >> For more options, visit https://groups.google.com/d/optout.
>
> --
> 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.
> For more options, visit https://groups.google.com/d/optout.
>

-- 
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.
For more options, visit https://groups.google.com/d/optout.

Reply via email to