Add varfmt

This commit is contained in:
Fedor Korotkiy 2020-02-21 23:43:16 +03:00
parent a9f09dd3c9
commit b8bb9bee55
3 changed files with 137 additions and 0 deletions

17
varfmt/README.md Normal file
View file

@ -0,0 +1,17 @@
# varfmt
Реализуйте функцию `varfmt.Sprintf`. Функция принимает формат строку и переменное число аргументов.
Синтаксис формат-строки похож формат-строки питона:
- `{}` - задаёт ссылку на аргумент
- `{number}` - ссылается на агрумент с индексом `number`
- `{}` ссылается на аргумент с индексом равным позиции `{}` внутри паттерна
Например, `varfmt.Sprintf("{1} {0}", "Hello", "World)` должен вернуть строку `Hello World`.
Аргументы функции могут быть произвольными типами. Вам нужно форматировать их так же, как это
делает функция `fmt.Sprint`. Вызывать `fmt.Sprint` для форматирования отдельного аргумента
не запрещается.
Ваше решение будет сравниваться с baseline-решением на бенчмарке. Ваш код должен
быть не более чем в два раза хуже чем baseline.

7
varfmt/fmt.go Normal file
View file

@ -0,0 +1,7 @@
// +build !solution
package varfmt
func Sprintf(format string, args ...interface{}) string {
return ""
}

113
varfmt/fmt_test.go Normal file
View file

@ -0,0 +1,113 @@
package varfmt
import (
"strings"
"testing"
"github.com/stretchr/testify/require"
)
func TestFormat(t *testing.T) {
for _, tc := range []struct {
format string
args []interface{}
result string
}{
{
format: "{}",
args: []interface{}{0},
result: "0",
},
{
format: "{0} {0}",
args: []interface{}{1},
result: "1 1",
},
{
format: "{1} {5}",
args: []interface{}{0, 1, 2, 3, 4, 5, 6},
result: "1 5",
},
{
format: "{} {} {} {} {}",
args: []interface{}{0, 1, 2, 3, 4},
result: "0 1 2 3 4",
},
{
format: "{} {0} {0} {0} {}",
args: []interface{}{0, 1, 2, 3, 4},
result: "0 0 0 0 4",
},
{
format: "Hello, {2}",
args: []interface{}{0, 1, "World"},
result: "Hello, World",
},
} {
t.Run(tc.result, func(t *testing.T) {
require.Equal(t, tc.result, Sprintf(tc.format, tc.args...))
})
}
}
func BenchmarkFormat(b *testing.B) {
for _, tc := range []struct {
name string
format string
args []interface{}
}{
{
name: "small int",
format: "{}",
args: []interface{}{42},
},
{
name: "small string",
format: "{} {}",
args: []interface{}{"Hello", "World"},
},
{
name: "big",
format: strings.Repeat("{0}{1}", 1000),
args: []interface{}{42, 43},
},
} {
b.Run(tc.name, func(b *testing.B) {
b.ReportAllocs()
for i := 0; i < b.N; i++ {
_ = Sprintf(tc.format, tc.args...)
}
})
}
}
func BenchmarkSprintf(b *testing.B) {
for _, tc := range []struct {
name string
format string
args []interface{}
}{
{
name: "small",
format: "%d",
args: []interface{}{42},
},
{
name: "small string",
format: "%v %v",
args: []interface{}{"Hello", "World"},
}, {
name: "big",
format: strings.Repeat("%[0]v%[1]v", 1000),
args: []interface{}{42, 43},
},
} {
b.Run(tc.name, func(b *testing.B) {
b.ReportAllocs()
for i := 0; i < b.N; i++ {
_ = Sprintf(tc.format, tc.args...)
}
})
}
}