Skip to content

Commit b35a231

Browse files
committed
reflect: add TypeAssert[T]
1 parent ff27d27 commit b35a231

File tree

4 files changed

+114
-0
lines changed

4 files changed

+114
-0
lines changed

api/next/62121.txt

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
pkg reflect, func TypeAssert[$0 interface{}](Value) ($0, bool) #62121
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
The new [TypeAssert] function permits converting a [Value] directly to a Go type.
2+
This is like using a type assertion on the result of [Value.Interface].

src/reflect/all_test.go

+80
Original file line numberDiff line numberDiff line change
@@ -8681,3 +8681,83 @@ func TestMapOfKeyPanic(t *testing.T) {
86818681
var slice []int
86828682
m.MapIndex(ValueOf(slice))
86838683
}
8684+
8685+
func testTypeAssert[T comparable](t *testing.T, val T) {
8686+
t.Helper()
8687+
v, ok := TypeAssert[T](ValueOf(val))
8688+
if v != val || !ok {
8689+
t.Errorf("TypeAssert[%T](%v) = (%v, %v); want = (%v, true)", *new(T), val, v, ok, val)
8690+
}
8691+
}
8692+
8693+
func testTypeAssertDifferentType[T, T2 comparable](t *testing.T, val T2) {
8694+
t.Helper()
8695+
if v, ok := TypeAssert[T](ValueOf(val)); ok {
8696+
t.Errorf("TypeAssert[%T](%v) = (%v, %v); want = (%v, false)", *new(T), val, v, ok, *new(T))
8697+
}
8698+
}
8699+
8700+
func newPtr[T any](t T) *T {
8701+
return &t
8702+
}
8703+
8704+
func TestTypeAssert(t *testing.T) {
8705+
testTypeAssert(t, int(1111))
8706+
testTypeAssert(t, int(111111111))
8707+
testTypeAssert(t, int(-111111111))
8708+
testTypeAssert(t, int32(111111111))
8709+
testTypeAssert(t, int32(-111111111))
8710+
testTypeAssert(t, uint32(111111111))
8711+
testTypeAssert(t, [2]int{111111111, 22222222})
8712+
testTypeAssert(t, [2]int{-111111111, -22222222})
8713+
testTypeAssert(t, newPtr(1111))
8714+
testTypeAssert(t, newPtr(111111111))
8715+
testTypeAssert(t, newPtr(-111111111))
8716+
testTypeAssert(t, newPtr([2]int{-111111111, -22222222}))
8717+
8718+
testTypeAssert(t, newPtr(time.Now()))
8719+
8720+
testTypeAssertDifferentType[uint](t, int(111111111))
8721+
testTypeAssertDifferentType[uint](t, int(-111111111))
8722+
}
8723+
8724+
type testTypeWithMethod struct {
8725+
val string
8726+
}
8727+
8728+
func (v testTypeWithMethod) String() string { return v.val }
8729+
8730+
func TestTypeAssertMethod(t *testing.T) {
8731+
method := ValueOf(&testTypeWithMethod{val: "test value"}).MethodByName("String")
8732+
f, ok := TypeAssert[func() string](method)
8733+
if !ok {
8734+
t.Fatalf(`TypeAssert[func() string](method) = (,false); want = (,true)`)
8735+
}
8736+
8737+
out := f()
8738+
if out != "test value" {
8739+
t.Fatalf(`TypeAssert[func() string](method)() = %q; want "test value"`, out)
8740+
}
8741+
}
8742+
8743+
func TestTypeAssertZeroValPanic(t *testing.T) {
8744+
defer func() { recover() }()
8745+
TypeAssert[int](Value{})
8746+
t.Fatalf("TypeAssert did not panic")
8747+
}
8748+
8749+
func TestTypeAssertReadOnlyPanic(t *testing.T) {
8750+
defer func() { recover() }()
8751+
TypeAssert[int](ValueOf(&testTypeWithMethod{}).FieldByName("val"))
8752+
t.Fatalf("TypeAssert did not panic")
8753+
}
8754+
8755+
func TestTypeAssertAllocs(t *testing.T) {
8756+
val := ValueOf(new(time.Time)).Elem()
8757+
allocs := testing.AllocsPerRun(100, func() {
8758+
TypeAssert[time.Time](val)
8759+
})
8760+
if allocs != 0 {
8761+
t.Errorf("unexpected amount of allocations = %v; want = 0", allocs)
8762+
}
8763+
}

src/reflect/value.go

+31
Original file line numberDiff line numberDiff line change
@@ -1510,6 +1510,37 @@ func valueInterface(v Value, safe bool) any {
15101510
return packEface(v)
15111511
}
15121512

1513+
// TypeAssert is semantically equivalent to:
1514+
//
1515+
// v2, ok := v.Interface().(T)
1516+
func TypeAssert[T any](v Value) (T, bool) {
1517+
if v.flag == 0 {
1518+
panic(&ValueError{"reflect.TypeAssert", Invalid})
1519+
}
1520+
1521+
if v.flag&flagRO != 0 {
1522+
// Do not allow access to unexported values via Interface,
1523+
// because they might be pointers that should not be
1524+
// writable or methods or function that should not be callable.
1525+
panic("reflect.TypeAssert: cannot return value obtained from unexported field or method")
1526+
}
1527+
1528+
if v.flag&flagMethod != 0 {
1529+
v = makeMethodValue("TypeAssert", v)
1530+
}
1531+
1532+
if abi.TypeFor[T]() != v.typ_ {
1533+
var zero T
1534+
return zero, false
1535+
}
1536+
1537+
if v.typ_.IsDirectIface() {
1538+
return *(*T)(unsafe.Pointer(&v.ptr)), true
1539+
}
1540+
1541+
return *(*T)(v.ptr), true
1542+
}
1543+
15131544
// InterfaceData returns a pair of unspecified uintptr values.
15141545
// It panics if v's Kind is not Interface.
15151546
//

0 commit comments

Comments
 (0)