diff --git a/packages/typeutils/typeutils.go b/packages/typeutils/typeutils.go
index eed90cf95478febc1fd3248a2d3ce5c90b7007bf..6bd97c8827ce824ca64a63a227680359712b769c 100644
--- a/packages/typeutils/typeutils.go
+++ b/packages/typeutils/typeutils.go
@@ -1,22 +1,10 @@
 package typeutils
 
 import (
-	"reflect"
 	"unsafe"
 )
 
-func BytesToString(b []byte) string {
-	bh := (*reflect.SliceHeader)(unsafe.Pointer(&b))
-
-	return *(*string)(unsafe.Pointer(&reflect.StringHeader{Data: bh.Data, Len: bh.Len}))
-}
-
-func StringToBytes(str string) []byte {
-	hdr := (*reflect.StringHeader)(unsafe.Pointer(&str))
-
-	return *(*[]byte)(unsafe.Pointer(&reflect.SliceHeader{Data: hdr.Data, Len: hdr.Len, Cap: hdr.Len}))
-}
-
+// Checks whether an interface is nil or has the value nil.
 func IsInterfaceNil(param interface{}) bool {
 	return param == nil || (*[2]uintptr)(unsafe.Pointer(&param))[1] == 0
 }
diff --git a/packages/typeutils/unsafe.go b/packages/typeutils/unsafe.go
new file mode 100644
index 0000000000000000000000000000000000000000..56e8a99697d960216ec5aecdad241f9e08b372d8
--- /dev/null
+++ b/packages/typeutils/unsafe.go
@@ -0,0 +1,30 @@
+package typeutils
+
+import (
+	"reflect"
+	"runtime"
+	"unsafe"
+)
+
+// Converts a slice of bytes into a string without performing a copy.
+// NOTE: This is an unsafe operation and may lead to problems if the bytes
+// passed as argument are changed while the string is used.  No checking whether
+// bytes are valid UTF-8 data is performed.
+func BytesToString(b []byte) string {
+	return *(*string)(unsafe.Pointer(&b))
+}
+
+// Converts a string into a slice of bytes without performing a copy.
+// NOTE: This is an unsafe operation and may lead to problems if the bytes are changed.
+func StringToBytes(s string) []byte {
+	sh := (*reflect.StringHeader)(unsafe.Pointer(&s))
+	b := *(*[]byte)(unsafe.Pointer(&reflect.SliceHeader{
+		Data: sh.Data,
+		Len:  sh.Len,
+		Cap:  sh.Len,
+	}))
+	// ensure the underlying string doesn't get GC'ed before the assignment happens
+	runtime.KeepAlive(&s)
+
+	return b
+}
diff --git a/packages/typeutils/unsafe_test.go b/packages/typeutils/unsafe_test.go
new file mode 100644
index 0000000000000000000000000000000000000000..d28a5c06f411c754399117d161cc0e7585cc5a19
--- /dev/null
+++ b/packages/typeutils/unsafe_test.go
@@ -0,0 +1,92 @@
+package typeutils
+
+import (
+	"bytes"
+	"strings"
+	"testing"
+)
+
+var testStrings = []string{
+	"",
+	" ",
+	"test",
+	"こんにちは、 世界",
+	strings.Repeat(" ", 10),
+	strings.Repeat(" ", 100),
+	strings.Repeat(" ", 10000),
+	strings.Repeat(" ", 1000000),
+}
+
+func TestBytesToString(t *testing.T) {
+	for _, expected := range testStrings {
+		arg := []byte(expected)
+		actual := BytesToString(arg)
+		if actual != expected {
+			t.Errorf("BytesToString(%q) = %q but expected %q", arg, actual, expected)
+		}
+	}
+}
+
+func TestStringToBytes(t *testing.T) {
+	for _, arg := range testStrings {
+		expected := []byte(arg)
+		actual := StringToBytes(arg)
+		if !bytes.Equal(actual, expected) {
+			t.Errorf("Bytes(%q) = %q but expected %q", arg, actual, expected)
+		}
+	}
+}
+
+func TestNil(t *testing.T) {
+	actual := BytesToString(nil)
+	expected := ""
+	if actual != expected {
+		t.Errorf("String(nil) = %q but expected %q", actual, expected)
+	}
+}
+
+func createTestBytes() [][]byte {
+	result := make([][]byte, len(testStrings))
+	for i, str := range testStrings {
+		result[i] = []byte(str)
+	}
+	return result
+}
+
+func BenchmarkNativeBytesToString(b *testing.B) {
+	testBytes := createTestBytes()
+	b.ResetTimer()
+
+	for i := 0; i < b.N; i++ {
+		for _, bs := range testBytes {
+			_ = string(bs)
+		}
+	}
+}
+
+func BenchmarkUnsafeBytesToString(b *testing.B) {
+	testBytes := createTestBytes()
+	b.ResetTimer()
+
+	for i := 0; i < b.N; i++ {
+		for _, bs := range testBytes {
+			_ = BytesToString(bs)
+		}
+	}
+}
+
+func BenchmarkNativeStringToBytes(b *testing.B) {
+	for i := 0; i < b.N; i++ {
+		for _, str := range testStrings {
+			_ = []byte(str)
+		}
+	}
+}
+
+func BenchmarkUnsafeStringToBytes(b *testing.B) {
+	for i := 0; i < b.N; i++ {
+		for _, str := range testStrings {
+			_ = StringToBytes(str)
+		}
+	}
+}