// run // Copyright 2022 The Go Authors. All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. // absdiff example in which an Abs method is attached to a generic type, which is a // structure with a single field that may be a list of possible basic types. package main import ( "fmt" "math" ) type Numeric interface { ~int | ~int8 | ~int16 | ~int32 | ~int64 | ~uint | ~uint8 | ~uint16 | ~uint32 | ~uint64 | ~uintptr | ~float32 | ~float64 | ~complex64 | ~complex128 } // numericAbs matches a struct containing a numeric type that has an Abs method. type numericAbs[T Numeric] interface { ~struct{ Value_ T } Abs() T Value() T } // absDifference computes the absolute value of the difference of // a and b, where the absolute value is determined by the Abs method. func absDifference[T Numeric, U numericAbs[T]](a, b U) T { d := a.Value() - b.Value() dt := U{Value_: d} return dt.Abs() } // orderedNumeric matches numeric types that support the < operator. type orderedNumeric interface { ~int | ~int8 | ~int16 | ~int32 | ~int64 | ~uint | ~uint8 | ~uint16 | ~uint32 | ~uint64 | ~uintptr | ~float32 | ~float64 } // Complex matches the two complex types, which do not have a < operator. type Complex interface { ~complex64 | ~complex128 } // orderedAbs is a helper type that defines an Abs method for // a struct containing an ordered numeric type. type orderedAbs[T orderedNumeric] struct { Value_ T } func (a orderedAbs[T]) Abs() T { if a.Value_ < 0 { return -a.Value_ } return a.Value_ } // Field accesses through type parameters are disabled // until we have a more thorough understanding of the // implications on the spec. See issue #51576. // Use accessor method instead. func (a orderedAbs[T]) Value() T { return a.Value_ } // complexAbs is a helper type that defines an Abs method for // a struct containing a complex type. type complexAbs[T Complex] struct { Value_ T } func realimag(x any) (re, im float64) { switch z := x.(type) { case complex64: re = float64(real(z)) im = float64(imag(z)) case complex128: re = real(z) im = imag(z) default: panic("unknown complex type") } return } func (a complexAbs[T]) Abs() T { // TODO use direct conversion instead of realimag once #50937 is fixed r, i := realimag(a.Value_) // r := float64(real(a.Value)) // i := float64(imag(a.Value)) d := math.Sqrt(r*r + i*i) return T(complex(d, 0)) } func (a complexAbs[T]) Value() T { return a.Value_ } // OrderedAbsDifference returns the absolute value of the difference // between a and b, where a and b are of an ordered type. func OrderedAbsDifference[T orderedNumeric](a, b T) T { return absDifference(orderedAbs[T]{a}, orderedAbs[T]{b}) } // ComplexAbsDifference returns the absolute value of the difference // between a and b, where a and b are of a complex type. func ComplexAbsDifference[T Complex](a, b T) T { return absDifference(complexAbs[T]{a}, complexAbs[T]{b}) } func main() { if got, want := OrderedAbsDifference(1.0, -2.0), 3.0; got != want { panic(fmt.Sprintf("got = %v, want = %v", got, want)) } if got, want := OrderedAbsDifference(-1.0, 2.0), 3.0; got != want { panic(fmt.Sprintf("got = %v, want = %v", got, want)) } if got, want := OrderedAbsDifference(-20, 15), 35; got != want { panic(fmt.Sprintf("got = %v, want = %v", got, want)) } if got, want := ComplexAbsDifference(5.0+2.0i, 2.0-2.0i), 5+0i; got != want { panic(fmt.Sprintf("got = %v, want = %v", got, want)) } if got, want := ComplexAbsDifference(2.0-2.0i, 5.0+2.0i), 5+0i; got != want { panic(fmt.Sprintf("got = %v, want = %v", got, want)) } }