diff --git a/docs/function.md b/docs/function.md new file mode 100644 index 0000000..226c948 --- /dev/null +++ b/docs/function.md @@ -0,0 +1,365 @@ +# Function +Package function can control the flow of function execution and support part of functional programming. + +
+ +## Source: + +[https://github.com/duke-git/lancet/blob/main/function/function.go](https://github.com/duke-git/lancet/blob/main/function/function.go) +[https://github.com/duke-git/lancet/blob/main/function/watcher.go](https://github.com/duke-git/lancet/blob/main/function/watcher.go) + +
+ +## Usage: +```go +import ( + "github.com/duke-git/lancet/function" +) +``` + +
+ +## Index +- [After](#After) +- [Before](#Before) +- [Curry](#Curry) +- [Compose](#Compose) +- [Debounced](#Debounced) +- [Delay](#Delay) +- [Delay](#Delay) +- [Watcher](#Watcher) + +
+ +## Documentation + + + +### After +

Creates a function that invokes given func once it's called n or more times.

+ +Signature: + +```go +func After(n int, fn interface{}) func(args ...interface{}) []reflect.Value +``` +Example: + +```go +package main + +import ( + "fmt" + "github.com/duke-git/lancet/function" +) + +func main() { + arr := []string{"a", "b"} + f := function.After(len(arr), func(i int) int { + fmt.Println("last print") + return i + }) + + type cb func(args ...interface{}) []reflect.Value + print := func(i int, s string, fn cb) { + fmt.Printf("arr[%d] is %s \n", i, s) + fn(i) + } + + fmt.Println("arr is", arr) + for i := 0; i < len(arr); i++ { + print(i, arr[i], f) + } + + //output: + // arr is [a b] + // arr[0] is a + // arr[1] is b + // last print +} +``` + + + +### Before + +

creates a function that invokes func once it's called less than n times.

+ +Signature: + +```go +func Before(n int, fn interface{}) func(args ...interface{}) []reflect.Value +``` +Example: + +```go +package main + +import ( + "fmt" + "github.com/duke-git/lancet/function" + "github.com/duke-git/lancet/internal" +) + +func main() { + assert := internal.NewAssert(t, "TestBefore") + + arr := []string{"a", "b", "c", "d", "e"} + f := function.Before(3, func(i int) int { + return i + }) + + var res []int64 + type cb func(args ...interface{}) []reflect.Value + appendStr := func(i int, s string, fn cb) { + v := fn(i) + res = append(res, v[0].Int()) + } + + for i := 0; i < len(arr); i++ { + appendStr(i, arr[i], f) + } + + expected := []int64{0, 1, 2, 2, 2} + assert.Equal(expected, res) +} +``` + + + +### Curry + +

Make a curry function.

+ +Signature: + +```go +type Fn func(...interface{}) interface{} +func (f Fn) Curry(i interface{}) func(...interface{}) interface{} +``` +Example: + +```go +package main + +import ( + "fmt" + "github.com/duke-git/lancet/function" +) + +func main() { + add := func(a, b int) int { + return a + b + } + var addCurry function.Fn = func(values ...interface{}) interface{} { + return add(values[0].(int), values[1].(int)) + } + add1 := addCurry.Curry(1) + result := add1(2) + fmt.Println(result) //3 +} +``` + + + +### Compose + +

Compose the function list from right to left, then return the composed function.

+ +Signature: + +```go +func Compose(fnList ...func(...interface{}) interface{}) func(...interface{}) interface{} +``` +Example: + +```go +package main + +import ( + "fmt" + "github.com/duke-git/lancet/function" +) + +func main() { + add1 := func(v ...interface{}) interface{} { + return v[0].(int) + 1 + } + add2 := func(v ...interface{}) interface{} { + return v[0].(int) + 2 + } + + add3 := function.Compose(add1, add2) + result := add3(1) + + fmt.Println(result) //4 +} +``` + + + +### Debounced + +

Creates a debounced function that delays invoking fn until after wait duration have elapsed since the last time the debounced function was invoked.

+ +Signature: + +```go +func Debounced(fn func(), duration time.Duration) func() +``` +Example: + +```go +package main + +import ( + "fmt" + "github.com/duke-git/lancet/function" +) + +func main() { + count := 0 + add := func() { + count++ + } + + debouncedAdd := function.Debounced(add, 50*time.Microsecond) + function.debouncedAdd() + function.debouncedAdd() + function.debouncedAdd() + function.debouncedAdd() + + time.Sleep(100 * time.Millisecond) + fmt.Println(count) //1 + + function.debouncedAdd() + time.Sleep(100 * time.Millisecond) + fmt.Println(count) //2 +} +``` + + + +### Delay + +

Invoke function after delayed time.

+ +Signature: + +```go +func Delay(delay time.Duration, fn interface{}, args ...interface{}) +``` +Example: + +```go +package main + +import ( + "fmt" + "github.com/duke-git/lancet/function" +) + +func main() { + var print = func(s string) { + fmt.Println(count) //test delay + } + function.Delay(2*time.Second, print, "test delay") +} +``` + + + +### Schedule + +

Invoke function every duration time, until close the returned bool chan.

+ +Signature: + +```go +func Schedule(d time.Duration, fn interface{}, args ...interface{}) chan bool +``` +Example: + +```go +package main + +import ( + "fmt" + "github.com/duke-git/lancet/function" +) + +func main() { + var res []string + appendStr := func(s string) { + res = append(res, s) + } + + stop := function.Schedule(1*time.Second, appendStr, "*") + time.Sleep(5 * time.Second) + close(stop) + + fmt.Println(res) //[* * * * *] +} +``` + + + +### Watcher + +

Watcher is used for record code excution time. can start/stop/reset the watch timer. get the elapsed time of function execution.

+ +Signature: + +```go +type Watcher struct { + startTime int64 + stopTime int64 + excuting bool +} +func (w *Watcher) Start() //start the watcher +func (w *Watcher) Stop() //stop the watcher +func (w *Watcher) Reset() //reset the watcher +func (w *Watcher) GetElapsedTime() time.Duration //get the elapsed time of function execution +``` +Example: + +```go +package main + +import ( + "fmt" + "github.com/duke-git/lancet/function" +) + +func main() { + w := &function.Watcher{} + w.Start() + + longRunningTask() + + fmt.Println(w.excuting) //true + + w.Stop() + + eapsedTime := w.GetElapsedTime().Milliseconds() + fmt.Println(eapsedTime) + + w.Reset() + + fmt.Println(w.excuting) //false + + fmt.Println(w.startTime) //0 + fmt.Println(w.stopTime) //0 +} + +func longRunningTask() { + var slice []int64 + for i := 0; i < 10000000; i++ { + slice = append(slice, int64(i)) + } +} + +``` + + + diff --git a/docs/function_zh-CN.md b/docs/function_zh-CN.md new file mode 100644 index 0000000..cf96bc2 --- /dev/null +++ b/docs/function_zh-CN.md @@ -0,0 +1,365 @@ +# Function +function函数包控制函数执行流程,包含部分函数式编程。 + +
+ +## 源码: + +[https://github.com/duke-git/lancet/blob/main/function/function.go](https://github.com/duke-git/lancet/blob/main/function/function.go) +[https://github.com/duke-git/lancet/blob/main/function/watcher.go](https://github.com/duke-git/lancet/blob/main/function/watcher.go) + +
+ +## 用法: +```go +import ( + "github.com/duke-git/lancet/function" +) +``` + +
+ +## 目录 +- [After](#After) +- [Before](#Before) +- [Curry](#Curry) +- [Compose](#Compose) +- [Debounced](#Debounced) +- [Delay](#Delay) +- [Delay](#Delay) +- [Watcher](#Watcher) + +
+ +## 文档 + + + +### After +

创建一个函数,当他被调用n或更多次之后将马上触发fn

+ +函数签名: + +```go +func After(n int, fn interface{}) func(args ...interface{}) []reflect.Value +``` +例子: + +```go +package main + +import ( + "fmt" + "github.com/duke-git/lancet/function" +) + +func main() { + arr := []string{"a", "b"} + f := function.After(len(arr), func(i int) int { + fmt.Println("last print") + return i + }) + + type cb func(args ...interface{}) []reflect.Value + print := func(i int, s string, fn cb) { + fmt.Printf("arr[%d] is %s \n", i, s) + fn(i) + } + + fmt.Println("arr is", arr) + for i := 0; i < len(arr); i++ { + print(i, arr[i], f) + } + + //output: + // arr is [a b] + // arr[0] is a + // arr[1] is b + // last print +} +``` + + + +### Before + +

创建一个函数,调用次数不超过n次,之后再调用这个函数,将返回一次最后调用fn的结果

+ +函数签名: + +```go +func Before(n int, fn interface{}) func(args ...interface{}) []reflect.Value +``` +例子: + +```go +package main + +import ( + "fmt" + "github.com/duke-git/lancet/function" + "github.com/duke-git/lancet/internal" +) + +func main() { + assert := internal.NewAssert(t, "TestBefore") + + arr := []string{"a", "b", "c", "d", "e"} + f := function.Before(3, func(i int) int { + return i + }) + + var res []int64 + type cb func(args ...interface{}) []reflect.Value + appendStr := func(i int, s string, fn cb) { + v := fn(i) + res = append(res, v[0].Int()) + } + + for i := 0; i < len(arr); i++ { + appendStr(i, arr[i], f) + } + + expected := []int64{0, 1, 2, 2, 2} + assert.Equal(expected, res) +} +``` + + + +### Curry + +

创建一个柯里化的函数

+ +函数签名: + +```go +type Fn func(...interface{}) interface{} +func (f Fn) Curry(i interface{}) func(...interface{}) interface{} +``` +例子: + +```go +package main + +import ( + "fmt" + "github.com/duke-git/lancet/function" +) + +func main() { + add := func(a, b int) int { + return a + b + } + var addCurry function.Fn = func(values ...interface{}) interface{} { + return add(values[0].(int), values[1].(int)) + } + add1 := addCurry.Curry(1) + result := add1(2) + fmt.Println(result) //3 +} +``` + + + +### Compose + +

从右至左组合函数列表fnList, 返回组合后的函数

+ +函数签名: + +```go +func Compose(fnList ...func(...interface{}) interface{}) func(...interface{}) interface{} +``` +例子: + +```go +package main + +import ( + "fmt" + "github.com/duke-git/lancet/function" +) + +func main() { + add1 := func(v ...interface{}) interface{} { + return v[0].(int) + 1 + } + add2 := func(v ...interface{}) interface{} { + return v[0].(int) + 2 + } + + add3 := function.Compose(add1, add2) + result := add3(1) + + fmt.Println(result) //4 +} +``` + + + +### Debounced + +

创建一个 debounced 函数,该函数延迟调用 fn 直到自上次调用 debounced 函数后等待持续时间过去。

+ +函数签名: + +```go +func Debounced(fn func(), duration time.Duration) func() +``` +例子: + +```go +package main + +import ( + "fmt" + "github.com/duke-git/lancet/function" +) + +func main() { + count := 0 + add := func() { + count++ + } + + debouncedAdd := function.Debounced(add, 50*time.Microsecond) + function.debouncedAdd() + function.debouncedAdd() + function.debouncedAdd() + function.debouncedAdd() + + time.Sleep(100 * time.Millisecond) + fmt.Println(count) //1 + + function.debouncedAdd() + time.Sleep(100 * time.Millisecond) + fmt.Println(count) //2 +} +``` + + + +### Delay + +

延迟delay时间后调用函数

+ +函数签名: + +```go +func Delay(delay time.Duration, fn interface{}, args ...interface{}) +``` +例子: + +```go +package main + +import ( + "fmt" + "github.com/duke-git/lancet/function" +) + +func main() { + var print = func(s string) { + fmt.Println(count) //test delay + } + function.Delay(2*time.Second, print, "test delay") +} +``` + + + +### Schedule + +

每次持续时间调用函数,直到关闭返回的 bool chan

+ +函数签名: + +```go +func Schedule(d time.Duration, fn interface{}, args ...interface{}) chan bool +``` +例子: + +```go +package main + +import ( + "fmt" + "github.com/duke-git/lancet/function" +) + +func main() { + var res []string + appendStr := func(s string) { + res = append(res, s) + } + + stop := function.Schedule(1*time.Second, appendStr, "*") + time.Sleep(5 * time.Second) + close(stop) + + fmt.Println(res) //[* * * * *] +} +``` + + + +### Watcher + +

Watcher 用于记录代码执行时间。可以启动/停止/重置手表定时器。获取函数执行的时间。

+ +函数签名: + +```go +type Watcher struct { + startTime int64 + stopTime int64 + excuting bool +} +func (w *Watcher) Start() //start the watcher +func (w *Watcher) Stop() //stop the watcher +func (w *Watcher) Reset() //reset the watcher +func (w *Watcher) GetElapsedTime() time.Duration //get the elapsed time of function execution +``` +例子: + +```go +package main + +import ( + "fmt" + "github.com/duke-git/lancet/function" +) + +func main() { + w := &function.Watcher{} + w.Start() + + longRunningTask() + + fmt.Println(w.excuting) //true + + w.Stop() + + eapsedTime := w.GetElapsedTime().Milliseconds() + fmt.Println(eapsedTime) + + w.Reset() + + fmt.Println(w.excuting) //false + + fmt.Println(w.startTime) //0 + fmt.Println(w.stopTime) //0 +} + +func longRunningTask() { + var slice []int64 + for i := 0; i < 10000000; i++ { + slice = append(slice, int64(i)) + } +} + +``` + + +