1
0
mirror of https://github.com/charlienet/go-mixed.git synced 2025-07-18 00:22:41 +08:00

随机数生成

This commit is contained in:
2022-03-31 11:41:41 +08:00
parent eacf6016aa
commit 6efb6dfac8
2 changed files with 77 additions and 26 deletions

View File

@ -6,6 +6,7 @@ import (
"sync" "sync"
"time" "time"
"math/big"
mrnd "math/rand" mrnd "math/rand"
"github.com/charlienet/go-mixed/bytesconv" "github.com/charlienet/go-mixed/bytesconv"
@ -75,7 +76,7 @@ var (
) )
// 生成指定长度的随机字符串 // 生成指定长度的随机字符串
func (scope *charScope) RandString(length int) string { func (scope *charScope) Generate(length int) string {
n := length n := length
if scope.lenFunc != nil { if scope.lenFunc != nil {
n = scope.lenFunc(n) n = scope.lenFunc(n)
@ -99,15 +100,32 @@ func (scope *charScope) RandString(length int) string {
return bytesconv.BytesToString(ret) return bytesconv.BytesToString(ret)
} }
// 获取指定范围内的随机数 type scopeConstraint interface {
func RandInt(min, max int) int { ~int | ~int32 | ~int64
n := randNumber2(max - min)
return n + min
} }
// 生成指定范围的随机数 // 生成区间 n >= 0, n < max
func RandInt32(min, max int32) int32 { func Intn[T scopeConstraint](max T) T {
return int32(RandInt(int(min), int(max))) n := mrnd.Int63n(int64(max))
return T(n) % max
}
// 生成区间 n >= min, n < max
func IntRange[T scopeConstraint](min, max T) T {
n := Intn(max - min)
return T(n + min)
}
func CryptoRange[T scopeConstraint](min, max T) T {
n := CryptoIntn(max - min)
return min + n
}
func CryptoIntn[T ~int | ~int32 | ~int64](max T) T {
b := big.NewInt(int64(max))
n, _ := rand.Int(rand.Reader, b)
return T(n.Int64())
} }
func RandBytes(len int) ([]byte, error) { func RandBytes(len int) ([]byte, error) {

View File

@ -11,7 +11,7 @@ import (
) )
func TestRandString(t *testing.T) { func TestRandString(t *testing.T) {
t.Log(rand.AllChars.RandString(20)) t.Log(rand.AllChars.Generate(20))
// b, err := rand.RandBytes(32) // b, err := rand.RandBytes(32)
// t.Log(err) // t.Log(err)
@ -19,7 +19,7 @@ func TestRandString(t *testing.T) {
} }
func TestRandHex(t *testing.T) { func TestRandHex(t *testing.T) {
h := rand.Hex.RandString(8) h := rand.Hex.Generate(8)
t.Log(h) t.Log(h)
} }
@ -27,43 +27,61 @@ func TestRandMax(t *testing.T) {
mrnd.Seed(time.Now().UnixNano()) mrnd.Seed(time.Now().UnixNano())
} }
func TestCryptRand(t *testing.T) {
t.Log(rand.CryptoIntn(56545))
}
func TestGenericsInterger(t *testing.T) {
var max int32 = 55
for i := 0; i < 1000; i++ {
if rand.Intn(max) >= max {
t.Fatal("生成的值大于最大值:", max)
}
}
}
func TestRange(t *testing.T) {
min := 20
max := 200000
for i := 0; i < 100000000; i++ {
n := rand.IntRange(min, max)
if n < min {
t.Fatal("生成的值小于最小值:", min, n)
}
if n >= max {
t.Fatal("生成的值大于最大值:", min, n)
}
}
}
func BenchmarkParallel(b *testing.B) { func BenchmarkParallel(b *testing.B) {
rand.Hex.RandString(16) rand.Hex.Generate(16)
b.RunParallel(func(pb *testing.PB) { b.RunParallel(func(pb *testing.PB) {
for pb.Next() { for pb.Next() {
rand.Hex.RandString(16) rand.Hex.Generate(16)
} }
}) })
} }
func BenchmarkNoop(b *testing.B) { func BenchmarkNoop(b *testing.B) {
for i := 0; i < b.N; i++ { for i := 0; i < b.N; i++ {
rand.AllChars.RandString(16) rand.AllChars.Generate(16)
} }
} }
func BenchmarkRandString(b *testing.B) { func BenchmarkRandString(b *testing.B) {
for i := 0; i < 10; i++ { for i := 0; i < 10; i++ {
rand.Hex.RandString(10) rand.Hex.Generate(10)
} }
b.ResetTimer() b.ResetTimer()
for i := 0; i < b.N; i++ { for i := 0; i < b.N; i++ {
rand.Hex.RandString(20) rand.Hex.Generate(20)
} }
// b.Run("randString", func(b *testing.B) {
// for i := 0; i < b.N; i++ {
// rand.Hex.RandString(256)
// }
// })
// for i := 0; i < b.N; i++ {
// rand.RandBytes(16)
// }
} }
func BenchmarkString(b *testing.B) { func BenchmarkString(b *testing.B) {
@ -116,3 +134,18 @@ func BenchmarkConcatString(b *testing.B) {
} }
}) })
} }
func BenchmarkRand(b *testing.B) {
b.Run("math/rand", func(b *testing.B) {
for i := 0; i < b.N; i++ {
rand.Intn(100)
}
})
b.Run("crypto/rand", func(b *testing.B) {
for i := 0; i < b.N; i++ {
rand.CryptoIntn(100)
}
})
}