1
0
mirror of https://github.com/golang/go synced 2024-11-26 03:17:57 -07:00

spec: document which recursive arrays and structs are valid/invalid

Fixes #5069.

Change-Id: I4bc0f52a9cd1e64a49846dffeb4be5cbecc29a96
Reviewed-on: https://go-review.googlesource.com/c/go/+/457342
TryBot-Bypass: Robert Griesemer <gri@google.com>
Reviewed-by: Rob Pike <r@golang.org>
Reviewed-by: Robert Griesemer <gri@google.com>
Reviewed-by: Robert Findley <rfindley@google.com>
This commit is contained in:
Robert Griesemer 2022-12-13 21:16:09 -08:00 committed by Robert Griesemer
parent 0b8add46ce
commit ea14d1b6e1

View File

@ -944,6 +944,29 @@ multi-dimensional types.
[2][2][2]float64 // same as [2]([2]([2]float64)) [2][2][2]float64 // same as [2]([2]([2]float64))
</pre> </pre>
<p>
An array type <code>T</code> may not have an element of type <code>T</code>,
or of a type containing <code>T</code> as a component, directly or indirectly,
if those containing types are only array or struct types.
</p>
<pre>
// valid array types
type (
T1 [10]T1 // element type of T1 is T1
T2 [10]struct{ f T2 } // T2 contains T2 as component of a struct
T3 [10]T4 // T3 contains T3 as component of a struct in T4
T4 struct{ f T3 } // T4 contains T4 as component of array T3 in a struct
)
// valid array types
type (
T5 [10]*T5 // T5 contains T5 as component of a pointer
T6 [10]func() T6 // T6 contains T6 as component of a function type
T7 [10]struct{ f []T7 } // T7 contains T7 as component of a slice in a struct
)
</pre>
<h3 id="Slice_types">Slice types</h3> <h3 id="Slice_types">Slice types</h3>
<p> <p>
@ -1136,6 +1159,29 @@ struct {
} }
</pre> </pre>
<p>
A struct type <code>T</code> may not contain a field of type <code>T</code>,
or of a type containing <code>T</code> as a component, directly or indirectly,
if those containing types are only array or struct types.
</p>
<pre>
// invalid struct types
type (
T1 struct{ T1 } // T1 contains a field of T1
T2 struct{ f [10]T2 } // T2 contains T2 as component of an array
T3 struct{ T4 } // T3 contains T3 as component of an array in struct T4
T4 struct{ f [10]T3 } // T4 contains T4 as component of struct T3 in an array
)
// valid struct types
type (
T5 struct{ f *T5 } // T5 contains T5 as component of a pointer
T6 struct{ f func() T6 } // T6 contains T6 as component of a function type
T7 struct{ f [10][]T7 } // T7 contains T7 as component of a slice in an array
)
</pre>
<h3 id="Pointer_types">Pointer types</h3> <h3 id="Pointer_types">Pointer types</h3>
<p> <p>
@ -1511,17 +1557,17 @@ type Floatish struct {
</pre> </pre>
<p> <p>
An interface type <code>T</code> may not embed any type element An interface type <code>T</code> may not embed a type element
that is, contains, or embeds <code>T</code>, recursively. that is, contains, or embeds <code>T</code>, directly or indirectly.
</p> </p>
<pre> <pre>
// illegal: Bad cannot embed itself // illegal: Bad may not embed itself
type Bad interface { type Bad interface {
Bad Bad
} }
// illegal: Bad1 cannot embed itself using Bad2 // illegal: Bad1 may not embed itself using Bad2
type Bad1 interface { type Bad1 interface {
Bad2 Bad2
} }
@ -1529,10 +1575,15 @@ type Bad2 interface {
Bad1 Bad1
} }
// illegal: Bad3 cannot embed a union containing Bad3 // illegal: Bad3 may not embed a union containing Bad3
type Bad3 interface { type Bad3 interface {
~int | ~string | Bad3 ~int | ~string | Bad3
} }
// illegal: Bad4 may not embed an array containing Bad4 as element type
type Bad4 interface {
[10]Bad4
}
</pre> </pre>
<h4 id="Implementing_an_interface">Implementing an interface</h4> <h4 id="Implementing_an_interface">Implementing an interface</h4>