mirror of
https://github.com/duke-git/lancet.git
synced 2026-02-14 09:42:28 +08:00
feat: add new functions in datetime and strutils
This commit is contained in:
@@ -210,6 +210,9 @@ import "github.com/duke-git/lancet/datetime"
|
|||||||
- [TimestampMilli](https://github.com/duke-git/lancet/blob/v1/docs/datetime.md#TimestampMilli)
|
- [TimestampMilli](https://github.com/duke-git/lancet/blob/v1/docs/datetime.md#TimestampMilli)
|
||||||
- [TimestampMicro](https://github.com/duke-git/lancet/blob/v1/docs/datetime.md#TimestampMicro)
|
- [TimestampMicro](https://github.com/duke-git/lancet/blob/v1/docs/datetime.md#TimestampMicro)
|
||||||
- [TimestampNano](https://github.com/duke-git/lancet/blob/v1/docs/datetime.md#TimestampNano)
|
- [TimestampNano](https://github.com/duke-git/lancet/blob/v1/docs/datetime.md#TimestampNano)
|
||||||
|
- [TrackFuncTime](https://github.com/duke-git/lancet/blob/v1/docs/datetime.md#TrackFuncTime)
|
||||||
|
- [DaysBetween](https://github.com/duke-git/lancet/blob/v1/docs/datetime.md#DaysBetween)
|
||||||
|
- [GenerateDatetimesBetween](https://github.com/duke-git/lancet/blob/v1/docs/datetime.md#GenerateDatetimesBetween)
|
||||||
|
|
||||||
### 5. Fileutil package implements some basic functions for file operations.
|
### 5. Fileutil package implements some basic functions for file operations.
|
||||||
|
|
||||||
@@ -478,6 +481,8 @@ import "github.com/duke-git/lancet/strutil"
|
|||||||
- [Ellipsis](https://github.com/duke-git/lancet/blob/v1/docs/strutil.md#Ellipsis)
|
- [Ellipsis](https://github.com/duke-git/lancet/blob/v1/docs/strutil.md#Ellipsis)
|
||||||
- [Shuffle](https://github.com/duke-git/lancet/blob/v1/docs/strutil.md#Shuffle)
|
- [Shuffle](https://github.com/duke-git/lancet/blob/v1/docs/strutil.md#Shuffle)
|
||||||
- [Rotate](https://github.com/duke-git/lancet/blob/v1/docs/strutil.md#Rotate)
|
- [Rotate](https://github.com/duke-git/lancet/blob/v1/docs/strutil.md#Rotate)
|
||||||
|
- [TemplateReplace](https://github.com/duke-git/lancet/blob/v1/docs/strutil.md#TemplateReplace)
|
||||||
|
- [RegexMatchAllGroups](https://github.com/duke-git/lancet/blob/v1/docs/strutil.md#RegexMatchAllGroups)
|
||||||
|
|
||||||
|
|
||||||
### 14. System package contain some functions about os, runtime, shell command.
|
### 14. System package contain some functions about os, runtime, shell command.
|
||||||
|
|||||||
@@ -213,6 +213,9 @@ import "github.com/duke-git/lancet/datetime"
|
|||||||
- [TimestampMilli](https://github.com/duke-git/lancet/blob/v1/docs/datetime_zh-CN.md#TimestampMilli)
|
- [TimestampMilli](https://github.com/duke-git/lancet/blob/v1/docs/datetime_zh-CN.md#TimestampMilli)
|
||||||
- [TimestampMicro](https://github.com/duke-git/lancet/blob/v1/docs/datetime_zh-CN.md#TimestampMicro)
|
- [TimestampMicro](https://github.com/duke-git/lancet/blob/v1/docs/datetime_zh-CN.md#TimestampMicro)
|
||||||
- [TimestampNano](https://github.com/duke-git/lancet/blob/v1/docs/datetime_zh-CN.md#TimestampNano)
|
- [TimestampNano](https://github.com/duke-git/lancet/blob/v1/docs/datetime_zh-CN.md#TimestampNano)
|
||||||
|
- [TrackFuncTime](https://github.com/duke-git/lancet/blob/v1/docs/datetime_zh-CN.md#TrackFuncTime)
|
||||||
|
- [DaysBetween](https://github.com/duke-git/lancet/blob/v1/docs/datetime_zh-CN.md#DaysBetween)
|
||||||
|
- [GenerateDatetimesBetween](https://github.com/duke-git/lancet/blob/v1/docs/datetime_zh-CN.md#GenerateDatetimesBetween)
|
||||||
|
|
||||||
### 5. fileutil 包支持文件基本操作。
|
### 5. fileutil 包支持文件基本操作。
|
||||||
|
|
||||||
@@ -480,6 +483,8 @@ import "github.com/duke-git/lancet/strutil"
|
|||||||
- [Ellipsis](https://github.com/duke-git/lancet/blob/v1/docs/strutil_zh-CN.md#Ellipsis)
|
- [Ellipsis](https://github.com/duke-git/lancet/blob/v1/docs/strutil_zh-CN.md#Ellipsis)
|
||||||
- [Shuffle](https://github.com/duke-git/lancet/blob/v1/docs/strutil_zh-CN.md#Shuffle)
|
- [Shuffle](https://github.com/duke-git/lancet/blob/v1/docs/strutil_zh-CN.md#Shuffle)
|
||||||
- [Rotate](https://github.com/duke-git/lancet/blob/v1/docs/strutil_zh-CN.md#Rotate)
|
- [Rotate](https://github.com/duke-git/lancet/blob/v1/docs/strutil_zh-CN.md#Rotate)
|
||||||
|
- [TemplateReplace](https://github.com/duke-git/lancet/blob/v1/docs/strutil_zh-CN.md#TemplateReplace)
|
||||||
|
- [RegexMatchAllGroups](https://github.com/duke-git/lancet/blob/v1/docs/strutil_zh-CN.md#RegexMatchAllGroups)
|
||||||
|
|
||||||
### 14. system 包含 os, runtime, shell command 相关函数。
|
### 14. system 包含 os, runtime, shell command 相关函数。
|
||||||
|
|
||||||
|
|||||||
@@ -31,6 +31,7 @@ package datetime
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"runtime"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
@@ -321,3 +322,62 @@ func TimestampNano(timezone ...string) int64 {
|
|||||||
|
|
||||||
return t.UnixNano()
|
return t.UnixNano()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TrackFuncTime track the time of function execution.
|
||||||
|
// call it at top of the func like `defer TrackFuncTime(time.Now())()`
|
||||||
|
// Play: todo
|
||||||
|
func TrackFuncTime(pre time.Time) func() {
|
||||||
|
callerName := getCallerName()
|
||||||
|
return func() {
|
||||||
|
elapsed := time.Since(pre)
|
||||||
|
fmt.Printf("Function %s execution time:\t %v", callerName, elapsed)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func getCallerName() string {
|
||||||
|
pc, _, _, ok := runtime.Caller(2)
|
||||||
|
if !ok {
|
||||||
|
return "Unknown"
|
||||||
|
}
|
||||||
|
fn := runtime.FuncForPC(pc)
|
||||||
|
if fn == nil {
|
||||||
|
return "Unknown"
|
||||||
|
}
|
||||||
|
|
||||||
|
fullName := fn.Name()
|
||||||
|
if lastDot := strings.LastIndex(fullName, "."); lastDot != -1 {
|
||||||
|
return fullName[lastDot+1:]
|
||||||
|
}
|
||||||
|
|
||||||
|
return fullName
|
||||||
|
}
|
||||||
|
|
||||||
|
// DaysBetween returns the number of days between two times.
|
||||||
|
func DaysBetween(start, end time.Time) int {
|
||||||
|
duration := end.Sub(start)
|
||||||
|
days := int(duration.Hours() / 24)
|
||||||
|
|
||||||
|
return days
|
||||||
|
}
|
||||||
|
|
||||||
|
// GenerateDatetimesBetween returns a slice of strings between two times.
|
||||||
|
// layout: the format of the datetime string
|
||||||
|
// interval: the interval between two datetimes
|
||||||
|
func GenerateDatetimesBetween(start, end time.Time, layout string, interval string) ([]string, error) {
|
||||||
|
var result []string
|
||||||
|
|
||||||
|
if start.After(end) {
|
||||||
|
start, end = end, start
|
||||||
|
}
|
||||||
|
|
||||||
|
duration, err := time.ParseDuration(interval)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
for current := start; !current.After(end); current = current.Add(duration) {
|
||||||
|
result = append(result, current.Format(layout))
|
||||||
|
}
|
||||||
|
|
||||||
|
return result, nil
|
||||||
|
}
|
||||||
|
|||||||
@@ -333,3 +333,114 @@ func TestTimestamp(t *testing.T) {
|
|||||||
ts4 := TimestampNano()
|
ts4 := TimestampNano()
|
||||||
t.Log(ts4)
|
t.Log(ts4)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestTrackFuncTime(t *testing.T) {
|
||||||
|
defer TrackFuncTime(time.Now())()
|
||||||
|
|
||||||
|
var n int
|
||||||
|
for i := 0; i < 5000000; i++ {
|
||||||
|
n++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestDaysBetween(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
|
||||||
|
assert := internal.NewAssert(t, "TestDaysBetween")
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
start time.Time
|
||||||
|
end time.Time
|
||||||
|
expected int
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
start: time.Date(2024, time.September, 1, 0, 0, 0, 0, time.UTC),
|
||||||
|
end: time.Date(2024, time.September, 10, 0, 0, 0, 0, time.UTC),
|
||||||
|
expected: 9,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
start: time.Date(2024, time.September, 10, 0, 0, 0, 0, time.UTC),
|
||||||
|
end: time.Date(2024, time.September, 1, 0, 0, 0, 0, time.UTC),
|
||||||
|
expected: -9,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
start: time.Date(2024, time.September, 1, 0, 0, 0, 0, time.UTC),
|
||||||
|
end: time.Date(2024, time.September, 1, 0, 0, 0, 0, time.UTC),
|
||||||
|
expected: 0,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
start: time.Date(2024, time.January, 1, 0, 0, 0, 0, time.UTC),
|
||||||
|
end: time.Date(2024, time.December, 31, 0, 0, 0, 0, time.UTC),
|
||||||
|
expected: 365,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
start: time.Date(2024, time.March, 1, 0, 0, 0, 0, time.UTC),
|
||||||
|
end: time.Date(2024, time.March, 31, 0, 0, 0, 0, time.UTC),
|
||||||
|
expected: 30,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tt := range tests {
|
||||||
|
result := DaysBetween(tt.start, tt.end)
|
||||||
|
assert.Equal(tt.expected, result)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestGenerateDatetimesBetween(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
|
||||||
|
assert := internal.NewAssert(t, "TestGenerateDatetimesBetween")
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
start time.Time
|
||||||
|
end time.Time
|
||||||
|
layout string
|
||||||
|
interval string
|
||||||
|
expected []string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
start: time.Date(2024, time.September, 1, 0, 0, 0, 0, time.UTC),
|
||||||
|
end: time.Date(2024, time.September, 1, 2, 0, 0, 0, time.UTC),
|
||||||
|
layout: "2006-01-02 15:04:05",
|
||||||
|
interval: "30m",
|
||||||
|
expected: []string{
|
||||||
|
"2024-09-01 00:00:00",
|
||||||
|
"2024-09-01 00:30:00",
|
||||||
|
"2024-09-01 01:00:00",
|
||||||
|
"2024-09-01 01:30:00",
|
||||||
|
"2024-09-01 02:00:00",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
start: time.Date(2024, time.September, 1, 0, 0, 0, 0, time.UTC),
|
||||||
|
end: time.Date(2024, time.September, 1, 0, 0, 0, 0, time.UTC),
|
||||||
|
layout: "2006-01-02 15:04:05",
|
||||||
|
interval: "1h",
|
||||||
|
expected: []string{"2024-09-01 00:00:00"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
start: time.Date(2024, time.September, 1, 0, 0, 0, 0, time.UTC),
|
||||||
|
end: time.Date(2024, time.September, 1, 3, 0, 0, 0, time.UTC),
|
||||||
|
layout: "2006-01-02 15:04:05",
|
||||||
|
interval: "2h",
|
||||||
|
expected: []string{
|
||||||
|
"2024-09-01 00:00:00",
|
||||||
|
"2024-09-01 02:00:00",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tt := range tests {
|
||||||
|
result, err := GenerateDatetimesBetween(tt.start, tt.end, tt.layout, tt.interval)
|
||||||
|
|
||||||
|
assert.Equal(tt.expected, result)
|
||||||
|
assert.IsNil(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
t.Run("Invalid interval", func(t *testing.T) {
|
||||||
|
_, err := GenerateDatetimesBetween(time.Now(), time.Now(), "2006-01-02 15:04:05", "invalid")
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("Expected error, got nil")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|||||||
106
docs/datetime.md
106
docs/datetime.md
@@ -62,6 +62,9 @@ import (
|
|||||||
- [TimestampMilli](#TimestampMilli)
|
- [TimestampMilli](#TimestampMilli)
|
||||||
- [TimestampMicro](#TimestampMicro)
|
- [TimestampMicro](#TimestampMicro)
|
||||||
- [TimestampNano](#TimestampNano)
|
- [TimestampNano](#TimestampNano)
|
||||||
|
- [TrackFuncTime](#TrackFuncTime)
|
||||||
|
- [DaysBetween](#DaysBetween)
|
||||||
|
- [GenerateDatetimesBetween](#GenerateDatetimesBetween)
|
||||||
|
|
||||||
<div STYLE="page-break-after: always;"></div>
|
<div STYLE="page-break-after: always;"></div>
|
||||||
|
|
||||||
@@ -1360,3 +1363,106 @@ func main() {
|
|||||||
// 1690363051331788000
|
// 1690363051331788000
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
### <span id="TrackFuncTime">TrackFuncTime</span>
|
||||||
|
|
||||||
|
<p>Tracks function execution time.</p>
|
||||||
|
|
||||||
|
<b>Signature:</b>
|
||||||
|
|
||||||
|
```go
|
||||||
|
func TrackFuncTime(pre time.Time) func()
|
||||||
|
```
|
||||||
|
|
||||||
|
<b>Example:</b>
|
||||||
|
|
||||||
|
```go
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"github.com/duke-git/lancet/v2/datetime"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
defer datetime.TrackFuncTime(time.Now())()
|
||||||
|
|
||||||
|
var n int
|
||||||
|
for i := 0; i < 5000000; i++ {
|
||||||
|
n++
|
||||||
|
}
|
||||||
|
|
||||||
|
fmt.Println(1) // Function main execution time: 1.460287ms
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### <span id="DaysBetween">DaysBetween</span>
|
||||||
|
|
||||||
|
<p>Returns the number of days between two times.</p>
|
||||||
|
|
||||||
|
<b>Signature:</b>
|
||||||
|
|
||||||
|
```go
|
||||||
|
func DaysBetween(start, end time.Time) int
|
||||||
|
```
|
||||||
|
|
||||||
|
<b>Example:</b>
|
||||||
|
|
||||||
|
```go
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"github.com/duke-git/lancet/datetime"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
start := time.Date(2024, time.September, 1, 0, 0, 0, 0, time.UTC)
|
||||||
|
end := time.Date(2024, time.September, 10, 0, 0, 0, 0, time.UTC)
|
||||||
|
|
||||||
|
result := datetime.DaysBetween(start, end)
|
||||||
|
|
||||||
|
fmt.Println(result)
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// 9
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### <span id="GenerateDatetimesBetween">GenerateDatetimesBetween</span>
|
||||||
|
|
||||||
|
<p>Returns a slice of strings between two times. `layout`: the format of the datetime string.`interval`: the interval between two datetimes.</p>
|
||||||
|
|
||||||
|
<b>Signature:</b>
|
||||||
|
|
||||||
|
```go
|
||||||
|
func GenerateDatetimesBetween(start, end time.Time, layout string, interval string) ([]string, error)
|
||||||
|
```
|
||||||
|
|
||||||
|
<b>Example:</b>
|
||||||
|
|
||||||
|
```go
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"github.com/duke-git/lancet/datetime"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
start := time.Date(2024, time.September, 1, 0, 0, 0, 0, time.UTC)
|
||||||
|
end := time.Date(2024, time.September, 1, 2, 0, 0, 0, time.UTC)
|
||||||
|
|
||||||
|
layout := "2006-01-02 15:04:05"
|
||||||
|
interval := "1h"
|
||||||
|
|
||||||
|
result, err := datetime.GenerateDatetimesBetween(start, end, layout, interval)
|
||||||
|
|
||||||
|
fmt.Println(result)
|
||||||
|
fmt.Println(err)
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// [2024-09-01 00:00:00 2024-09-01 01:00:00 2024-09-01 02:00:00]
|
||||||
|
// <nil>
|
||||||
|
}
|
||||||
|
```
|
||||||
@@ -62,6 +62,10 @@ import (
|
|||||||
- [TimestampMilli](#TimestampMilli)
|
- [TimestampMilli](#TimestampMilli)
|
||||||
- [TimestampMicro](#TimestampMicro)
|
- [TimestampMicro](#TimestampMicro)
|
||||||
- [TimestampNano](#TimestampNano)
|
- [TimestampNano](#TimestampNano)
|
||||||
|
- [TrackFuncTime](#TrackFuncTime)
|
||||||
|
- [DaysBetween](#DaysBetween)
|
||||||
|
- [GenerateDatetimesBetween](#GenerateDatetimesBetween)
|
||||||
|
|
||||||
|
|
||||||
<div STYLE="page-break-after: always;"></div>
|
<div STYLE="page-break-after: always;"></div>
|
||||||
|
|
||||||
@@ -1279,4 +1283,108 @@ func main() {
|
|||||||
// Output:
|
// Output:
|
||||||
// 1690363051331788000
|
// 1690363051331788000
|
||||||
}
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### <span id="TrackFuncTime">TrackFuncTime</span>
|
||||||
|
|
||||||
|
<p>返回两个日期之间的天数差。</p>
|
||||||
|
|
||||||
|
<b>函数签名:</b>
|
||||||
|
|
||||||
|
```go
|
||||||
|
func TrackFuncTime(pre time.Time) func()
|
||||||
|
```
|
||||||
|
|
||||||
|
<b>示例:</b>
|
||||||
|
|
||||||
|
```go
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"github.com/duke-git/lancet/datetime"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
defer datetime.TrackFuncTime(time.Now())()
|
||||||
|
|
||||||
|
var n int
|
||||||
|
for i := 0; i < 5000000; i++ {
|
||||||
|
n++
|
||||||
|
}
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// Function main execution time: 1.608195ms
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### <span id="DaysBetween">DaysBetween</span>
|
||||||
|
|
||||||
|
<p>返回两个日期之间的天数差。</p>
|
||||||
|
|
||||||
|
<b>函数签名:</b>
|
||||||
|
|
||||||
|
```go
|
||||||
|
func DaysBetween(start, end time.Time) int
|
||||||
|
```
|
||||||
|
|
||||||
|
<b>示例:</b>
|
||||||
|
|
||||||
|
```go
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"github.com/duke-git/lancet/datetime"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
start := time.Date(2024, time.September, 1, 0, 0, 0, 0, time.UTC)
|
||||||
|
end := time.Date(2024, time.September, 10, 0, 0, 0, 0, time.UTC)
|
||||||
|
|
||||||
|
result := datetime.DaysBetween(start, end)
|
||||||
|
|
||||||
|
fmt.Println(result)
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// 9
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### <span id="GenerateDatetimesBetween">GenerateDatetimesBetween</span>
|
||||||
|
|
||||||
|
<p>生成从start到end的所有日期时间的字符串列表。layout参数表示时间格式,例如"2006-01-02 15:04:05",interval参数表示时间间隔,例如"1h"表示1小时,"30m"表示30分钟。</p>
|
||||||
|
|
||||||
|
<b>函数签名:</b>
|
||||||
|
|
||||||
|
```go
|
||||||
|
func GenerateDatetimesBetween(start, end time.Time, layout string, interval string) ([]string, error)
|
||||||
|
```
|
||||||
|
|
||||||
|
<b>示例:</b>
|
||||||
|
|
||||||
|
```go
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"github.com/duke-git/lancet/datetime"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
start := time.Date(2024, time.September, 1, 0, 0, 0, 0, time.UTC)
|
||||||
|
end := time.Date(2024, time.September, 1, 2, 0, 0, 0, time.UTC)
|
||||||
|
|
||||||
|
layout := "2006-01-02 15:04:05"
|
||||||
|
interval := "1h"
|
||||||
|
|
||||||
|
result, err := datetime.GenerateDatetimesBetween(start, end, layout, interval)
|
||||||
|
|
||||||
|
fmt.Println(result)
|
||||||
|
fmt.Println(err)
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// [2024-09-01 00:00:00 2024-09-01 01:00:00 2024-09-01 02:00:00]
|
||||||
|
// <nil>
|
||||||
|
}
|
||||||
```
|
```
|
||||||
@@ -64,6 +64,8 @@ import (
|
|||||||
- [Ellipsis](#Ellipsis)
|
- [Ellipsis](#Ellipsis)
|
||||||
- [Shuffle](#Shuffle)
|
- [Shuffle](#Shuffle)
|
||||||
- [Rotate](#Rotate)
|
- [Rotate](#Rotate)
|
||||||
|
- [TemplateReplace](#TemplateReplace)
|
||||||
|
- [RegexMatchAllGroups](#RegexMatchAllGroups)
|
||||||
|
|
||||||
<div STYLE="page-break-after: always;"></div>
|
<div STYLE="page-break-after: always;"></div>
|
||||||
|
|
||||||
@@ -1468,4 +1470,71 @@ func main() {
|
|||||||
// Hello
|
// Hello
|
||||||
// oHell
|
// oHell
|
||||||
// loHel
|
// loHel
|
||||||
}
|
}
|
||||||
|
|
||||||
|
### <span id="TemplateReplace">TemplateReplace</span>
|
||||||
|
|
||||||
|
<p>Replaces the placeholders in the template string with the corresponding values in the data map.The placeholders are enclosed in curly braces, e.g. {key}. for example, the template string is "Hello, {name}!", and the data map is {"name": "world"}, the result will be "Hello, world!".</p>
|
||||||
|
|
||||||
|
<b>Signature:</b>
|
||||||
|
|
||||||
|
```go
|
||||||
|
func TemplateReplace(template string, data map[string]string string
|
||||||
|
```
|
||||||
|
|
||||||
|
<b>example:</b>
|
||||||
|
|
||||||
|
```go
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"github.com/duke-git/lancet/strutil"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
template := `Hello, my name is {name}, I'm {age} years old.`
|
||||||
|
data := map[string]string{
|
||||||
|
"name": "Bob",
|
||||||
|
"age": "20",
|
||||||
|
}
|
||||||
|
|
||||||
|
result := strutil.TemplateReplace(template, data)
|
||||||
|
|
||||||
|
fmt.Println(result)
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// Hello, my name is Bob, I'm 20 years old.
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### <span id="RegexMatchAllGroups">RegexMatchAllGroups</span>
|
||||||
|
|
||||||
|
<p>Matches all subgroups in a string using a regular expression and returns the result.</p>
|
||||||
|
|
||||||
|
<b>Signature:</b>
|
||||||
|
|
||||||
|
```go
|
||||||
|
func RegexMatchAllGroups(pattern, str string) [][]string
|
||||||
|
```
|
||||||
|
|
||||||
|
<b>example:</b>
|
||||||
|
|
||||||
|
```go
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"github.com/duke-git/lancet/strutil"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
pattern := `(\w+\.+\w+)@(\w+)\.(\w+)`
|
||||||
|
str := "Emails: john.doe@example.com and jane.doe@example.com"
|
||||||
|
|
||||||
|
result := strutil.RegexMatchAllGroups(pattern, str)
|
||||||
|
|
||||||
|
fmt.Println(result[0])
|
||||||
|
fmt.Println(result[1])
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// [john.doe@example.com john.doe example com]
|
||||||
|
// [jane.doe@example.com jane.doe example com]
|
||||||
|
}
|
||||||
|
```
|
||||||
@@ -64,6 +64,8 @@ import (
|
|||||||
- [Ellipsis](#Ellipsis)
|
- [Ellipsis](#Ellipsis)
|
||||||
- [Shuffle](#Shuffle)
|
- [Shuffle](#Shuffle)
|
||||||
- [Rotate](#Rotate)
|
- [Rotate](#Rotate)
|
||||||
|
- [TemplateReplace](#TemplateReplace)
|
||||||
|
- [RegexMatchAllGroups](#RegexMatchAllGroups)
|
||||||
|
|
||||||
|
|
||||||
<div STYLE="page-break-after: always;"></div>
|
<div STYLE="page-break-after: always;"></div>
|
||||||
@@ -1503,4 +1505,71 @@ func main() {
|
|||||||
// oHell
|
// oHell
|
||||||
// loHel
|
// loHel
|
||||||
}
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### <span id="TemplateReplace">TemplateReplace</span>
|
||||||
|
|
||||||
|
<p>将模板字符串中的占位符替换为数据映射中的相应值。占位符括在花括号中,例如 {key}。例如,模板字符串为“Hello, {name}!”,数据映射为{"name": "world"},结果将为“Hello, world!”。</p>
|
||||||
|
|
||||||
|
<b>函数签名:</b>
|
||||||
|
|
||||||
|
```go
|
||||||
|
func TemplateReplace(template string, data map[string]string) string
|
||||||
|
```
|
||||||
|
|
||||||
|
<b>示例:</b>
|
||||||
|
|
||||||
|
```go
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"github.com/duke-git/lancet/strutil"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
template := `Hello, my name is {name}, I'm {age} years old.`
|
||||||
|
data := map[string]string{
|
||||||
|
"name": "Bob",
|
||||||
|
"age": "20",
|
||||||
|
}
|
||||||
|
|
||||||
|
result := strutil.TemplateReplace(template, data)
|
||||||
|
|
||||||
|
fmt.Println(result)
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// Hello, my name is Bob, I'm 20 years old.
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### <span id="RegexMatchAllGroups">RegexMatchAllGroups</span>
|
||||||
|
|
||||||
|
<p>使用正则表达式匹配字符串中的所有子组并返回结果。</p>
|
||||||
|
|
||||||
|
<b>函数签名:</b>
|
||||||
|
|
||||||
|
```go
|
||||||
|
func RegexMatchAllGroups(pattern, str string) [][]string
|
||||||
|
```
|
||||||
|
|
||||||
|
<b>示例:</b>
|
||||||
|
|
||||||
|
```go
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"github.com/duke-git/lancet/strutil"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
pattern := `(\w+\.+\w+)@(\w+)\.(\w+)`
|
||||||
|
str := "Emails: john.doe@example.com and jane.doe@example.com"
|
||||||
|
|
||||||
|
result := strutil.RegexMatchAllGroups(pattern, str)
|
||||||
|
|
||||||
|
fmt.Println(result[0])
|
||||||
|
fmt.Println(result[1])
|
||||||
|
|
||||||
|
// Output:
|
||||||
|
// [john.doe@example.com john.doe example com]
|
||||||
|
// [jane.doe@example.com jane.doe example com]
|
||||||
|
}
|
||||||
```
|
```
|
||||||
@@ -642,3 +642,32 @@ func Rotate(str string, shift int) string {
|
|||||||
|
|
||||||
return sb.String()
|
return sb.String()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TemplateReplace replaces the placeholders in the template string with the corresponding values in the data map.
|
||||||
|
// The placeholders are enclosed in curly braces, e.g. {key}.
|
||||||
|
// for example, the template string is "Hello, {name}!", and the data map is {"name": "world"},
|
||||||
|
// the result will be "Hello, world!".
|
||||||
|
func TemplateReplace(template string, data map[string]string) string {
|
||||||
|
re := regexp.MustCompile(`\{(\w+)\}`)
|
||||||
|
|
||||||
|
result := re.ReplaceAllStringFunc(template, func(s string) string {
|
||||||
|
key := strings.Trim(s, "{}")
|
||||||
|
if val, ok := data[key]; ok {
|
||||||
|
return val
|
||||||
|
}
|
||||||
|
|
||||||
|
return s
|
||||||
|
})
|
||||||
|
|
||||||
|
result = strings.ReplaceAll(result, "{{", "{")
|
||||||
|
result = strings.ReplaceAll(result, "}}", "}")
|
||||||
|
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
// RegexMatchAllGroups Matches all subgroups in a string using a regular expression and returns the result.
|
||||||
|
func RegexMatchAllGroups(pattern, str string) [][]string {
|
||||||
|
re := regexp.MustCompile(pattern)
|
||||||
|
matches := re.FindAllStringSubmatch(str, -1)
|
||||||
|
return matches
|
||||||
|
}
|
||||||
|
|||||||
@@ -581,3 +581,90 @@ func TestRotate(t *testing.T) {
|
|||||||
assert.Equal(tt.expected, Rotate(tt.input, tt.shift))
|
assert.Equal(tt.expected, Rotate(tt.input, tt.shift))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestTemplateReplace(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
|
||||||
|
assert := internal.NewAssert(t, "TestTemplateReplace")
|
||||||
|
|
||||||
|
t.Run("basic", func(t *testing.T) {
|
||||||
|
template := `Hello, my name is {name}, I'm {age} years old.`
|
||||||
|
data := map[string]string{
|
||||||
|
"name": "Bob",
|
||||||
|
"age": "20",
|
||||||
|
}
|
||||||
|
|
||||||
|
expected := `Hello, my name is Bob, I'm 20 years old.`
|
||||||
|
result := TemplateReplace(template, data)
|
||||||
|
|
||||||
|
assert.Equal(expected, result)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("not found", func(t *testing.T) {
|
||||||
|
template := `Hello, my name is {name}, I'm {age} years old.`
|
||||||
|
data := map[string]string{
|
||||||
|
"name": "Bob",
|
||||||
|
}
|
||||||
|
|
||||||
|
expected := `Hello, my name is Bob, I'm {age} years old.`
|
||||||
|
result := TemplateReplace(template, data)
|
||||||
|
|
||||||
|
assert.Equal(expected, result)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("brackets", func(t *testing.T) {
|
||||||
|
template := `Hello, my name is {name}, I'm {{age}} years old.`
|
||||||
|
data := map[string]string{
|
||||||
|
"name": "Bob",
|
||||||
|
"age": "20",
|
||||||
|
}
|
||||||
|
|
||||||
|
expected := `Hello, my name is Bob, I'm {20} years old.`
|
||||||
|
result := TemplateReplace(template, data)
|
||||||
|
|
||||||
|
assert.Equal(expected, result)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestRegexMatchAllGroups(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
|
||||||
|
assert := internal.NewAssert(t, "TestRegexMatchAllGroups")
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
pattern string
|
||||||
|
str string
|
||||||
|
expected [][]string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
pattern: `(\w+\.+\w+)@(\w+)\.(\w+)`,
|
||||||
|
str: "Emails: john.doe@example.com and jane.doe@example.com",
|
||||||
|
expected: [][]string{{"john.doe@example.com", "john.doe", "example", "com"}, {"jane.doe@example.com", "jane.doe", "example", "com"}},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
pattern: `(\d+)`,
|
||||||
|
str: "No numbers here!",
|
||||||
|
expected: nil,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
pattern: `(\d{3})-(\d{2})-(\d{4})`,
|
||||||
|
str: "My number is 123-45-6789",
|
||||||
|
expected: [][]string{{"123-45-6789", "123", "45", "6789"}},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
pattern: `(\w+)\s(\d+)`,
|
||||||
|
str: "Item A 123, Item B 456",
|
||||||
|
expected: [][]string{{"A 123", "A", "123"}, {"B 456", "B", "456"}},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
pattern: `(\d{2})-(\d{2})-(\d{4})`,
|
||||||
|
str: "Dates: 01-01-2020, 12-31-1999",
|
||||||
|
expected: [][]string{{"01-01-2020", "01", "01", "2020"}, {"12-31-1999", "12", "31", "1999"}},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tt := range tests {
|
||||||
|
result := RegexMatchAllGroups(tt.pattern, tt.str)
|
||||||
|
assert.Equal(tt.expected, result)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user