Skip to content

Commit 23c3614

Browse files
committed
improve performance of formatting timestamps by 35%
Use an optimized routine for formatting timestamps in the default layout for a 35% speedup. goos: darwin goarch: arm64 pkg: github.com/charlievieth/go-sqlite3/internal/timefmt cpu: Apple M4 Pro │ old.txt │ new.txt │ │ sec/op │ sec/op vs base │ Format-14 98.27n ± 1% 63.80n ± 1% -35.08% (p=0.000 n=10) │ old.txt │ new.txt │ │ B/op │ B/op vs base │ Format-14 48.00 ± 0% 48.00 ± 0% ~ (p=1.000 n=10) ¹ ¹ all samples are equal │ old.txt │ new.txt │ │ allocs/op │ allocs/op vs base │ Format-14 1.000 ± 0% 1.000 ± 0% ~ (p=1.000 n=10) ¹ ¹ all samples are equal
1 parent 9536c08 commit 23c3614

File tree

3 files changed

+194
-6
lines changed

3 files changed

+194
-6
lines changed

internal/timefmt/timefmt.go

+122
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,122 @@
1+
package timefmt
2+
3+
import "time"
4+
5+
const digits = "0123456789"
6+
7+
const smallsString = "00010203040506070809" +
8+
"10111213141516171819" +
9+
"20212223242526272829" +
10+
"30313233343536373839" +
11+
"40414243444546474849" +
12+
"50515253545556575859" +
13+
"60616263646566676869" +
14+
"70717273747576777879" +
15+
"80818283848586878889" +
16+
"90919293949596979899"
17+
18+
func appendInt2(dst []byte, i int) []byte {
19+
u := uint(i)
20+
if u < 10 {
21+
return append(dst, '0', digits[u])
22+
}
23+
return append(dst, smallsString[u*2:u*2+2]...)
24+
}
25+
26+
// appendInt appends the decimal form of x to b and returns the result.
27+
// If the decimal form (excluding sign) is shorter than width, the result is padded with leading 0's.
28+
// Duplicates functionality in strconv, but avoids dependency.
29+
func appendInt(b []byte, x int, width int) []byte {
30+
u := uint(x)
31+
if x < 0 {
32+
b = append(b, '-')
33+
u = uint(-x)
34+
}
35+
36+
// 2-digit and 4-digit fields are the most common in time formats.
37+
utod := func(u uint) byte { return '0' + byte(u) }
38+
switch {
39+
case width == 2 && u < 1e2:
40+
return append(b, utod(u/1e1), utod(u%1e1))
41+
case width == 4 && u < 1e4:
42+
return append(b, utod(u/1e3), utod(u/1e2%1e1), utod(u/1e1%1e1), utod(u%1e1))
43+
}
44+
45+
// Compute the number of decimal digits.
46+
var n int
47+
if u == 0 {
48+
n = 1
49+
}
50+
for u2 := u; u2 > 0; u2 /= 10 {
51+
n++
52+
}
53+
54+
// Add 0-padding.
55+
for pad := width - n; pad > 0; pad-- {
56+
b = append(b, '0')
57+
}
58+
59+
// Ensure capacity.
60+
if len(b)+n <= cap(b) {
61+
b = b[:len(b)+n]
62+
} else {
63+
b = append(b, make([]byte, n)...)
64+
}
65+
66+
// Assemble decimal in reverse order.
67+
i := len(b) - 1
68+
for u >= 10 && i > 0 {
69+
q := u / 10
70+
b[i] = utod(u - q*10)
71+
u = q
72+
i--
73+
}
74+
b[i] = utod(u)
75+
return b
76+
}
77+
78+
// formatTime formats time t with layout "2006-01-02 15:04:05.999999999-07:00"
79+
func Format(t time.Time) []byte {
80+
b := make([]byte, 0, len("2006-01-02 15:04:05.999999999-07:00"))
81+
82+
// 2006-01-02
83+
year, month, day := t.Date()
84+
b = appendInt(b, year, 4)
85+
b = append(b, '-')
86+
b = appendInt2(b, int(month))
87+
b = append(b, '-')
88+
b = appendInt2(b, day)
89+
90+
// 15:04:05
91+
b = append(b, ' ')
92+
b = appendInt2(b, t.Hour())
93+
b = append(b, ':')
94+
b = appendInt2(b, t.Minute())
95+
b = append(b, ':')
96+
b = appendInt2(b, t.Second())
97+
98+
// .999999999
99+
b = append(b, '.')
100+
b = appendInt(b, t.Nanosecond(), 9)
101+
for len(b) > 0 && b[len(b)-1] == '0' {
102+
b = b[:len(b)-1]
103+
}
104+
if len(b) > 0 && b[len(b)-1] == '.' {
105+
b = b[:len(b)-1]
106+
}
107+
108+
// -07:00
109+
_, offset := t.Zone()
110+
zone := offset / 60
111+
if zone < 0 {
112+
b = append(b, '-')
113+
zone = -zone
114+
} else {
115+
b = append(b, '+')
116+
}
117+
b = appendInt2(b, zone/60)
118+
b = append(b, ':')
119+
b = appendInt2(b, zone%60)
120+
121+
return b
122+
}

internal/timefmt/timefmt_test.go

+66
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,66 @@
1+
package timefmt_test
2+
3+
import (
4+
"math/rand"
5+
"strconv"
6+
"testing"
7+
"time"
8+
9+
"github.com/charlievieth/go-sqlite3"
10+
"github.com/charlievieth/go-sqlite3/internal/timefmt"
11+
)
12+
13+
func TestFormatTime(t *testing.T) {
14+
// Create time locations with random offsets
15+
rr := rand.New(rand.NewSource(time.Now().UnixNano()))
16+
locs := make([]*time.Location, 1000)
17+
for i := range locs {
18+
offset := rr.Intn(60 * 60 * 14) // 14 hours
19+
if rr.Int()&1 != 0 {
20+
offset = -offset
21+
}
22+
locs[i] = time.FixedZone(strconv.Itoa(offset), offset)
23+
}
24+
// Append some standard locations
25+
locs = append(locs, time.Local, time.UTC)
26+
27+
times := []time.Time{
28+
{},
29+
time.Now(),
30+
time.Date(0, 0, 0, 0, 0, 0, 0, time.UTC),
31+
time.Date(1, 1, 1, 1, 1, 1, 1, time.UTC),
32+
time.Date(20_000, 1, 1, 1, 1, 1, 1, time.UTC),
33+
time.Date(-1, 0, 0, 0, 0, 0, 0, time.UTC),
34+
}
35+
36+
for _, loc := range locs {
37+
for _, tt := range times {
38+
tt = tt.In(loc)
39+
got := timefmt.Format(tt)
40+
want := tt.Format(sqlite3.SQLiteTimestampFormats[0])
41+
if string(got) != want {
42+
t.Errorf("Format(%q) = %q; want: %q", tt.Format(time.RFC3339Nano), got, want)
43+
}
44+
}
45+
}
46+
}
47+
48+
func TestFormatTimeAllocs(t *testing.T) {
49+
allocs := testing.AllocsPerRun(100, func() {
50+
_ = timefmt.Format(time.Now())
51+
})
52+
if allocs != 1 {
53+
t.Fatalf("expected 1 allocation per-run got: %.1f", allocs)
54+
}
55+
}
56+
57+
func BenchmarkFormat(b *testing.B) {
58+
loc, err := time.LoadLocation("America/New_York")
59+
if err != nil {
60+
b.Fatal(err)
61+
}
62+
ts := time.Date(2024, 1, 2, 15, 4, 5, 123456789, loc)
63+
for i := 0; i < b.N; i++ {
64+
_ = timefmt.Format(ts)
65+
}
66+
}

sqlite3.go

+6-6
Original file line numberDiff line numberDiff line change
@@ -432,6 +432,8 @@ import (
432432
"syscall"
433433
"time"
434434
"unsafe"
435+
436+
"github.com/charlievieth/go-sqlite3/internal/timefmt"
435437
)
436438

437439
// SQLiteTimestampFormats is timestamp formats understood by both this module
@@ -2290,9 +2292,8 @@ func (s *SQLiteStmt) bind(args []driver.NamedValue) error {
22902292
rv = C._sqlite3_bind_blob(s.s, n, unsafe.Pointer(&v[0]), C.int(ln))
22912293
}
22922294
case time.Time:
2293-
ts := v.Format(SQLiteTimestampFormats[0])
2294-
p := stringData(ts)
2295-
rv = C._sqlite3_bind_text(s.s, n, (*C.char)(unsafe.Pointer(p)), C.int(len(ts)))
2295+
b := timefmt.Format(v)
2296+
rv = C._sqlite3_bind_text(s.s, n, (*C.char)(unsafe.Pointer(&b[0])), C.int(len(b)))
22962297
}
22972298
if rv != C.SQLITE_OK {
22982299
return s.c.lastError(int(rv))
@@ -2359,9 +2360,8 @@ func (s *SQLiteStmt) bindIndices(args []driver.NamedValue) error {
23592360
rv = C._sqlite3_bind_blob(s.s, n, unsafe.Pointer(&v[0]), C.int(ln))
23602361
}
23612362
case time.Time:
2362-
ts := v.Format(SQLiteTimestampFormats[0])
2363-
p := stringData(ts)
2364-
rv = C._sqlite3_bind_text(s.s, n, (*C.char)(unsafe.Pointer(p)), C.int(len(ts)))
2363+
b := timefmt.Format(v)
2364+
rv = C._sqlite3_bind_text(s.s, n, (*C.char)(unsafe.Pointer(&b[0])), C.int(len(b)))
23652365
}
23662366
if rv != C.SQLITE_OK {
23672367
return s.c.lastError(int(rv))

0 commit comments

Comments
 (0)