| // 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)) | 
 | 	} | 
 | } |