Go語言context?test源碼分析詳情
1.測試例子分析
example_test.go,展示了With-系列的4個例子
func ExampleWithCancel() { ? gen := func(ctx context.Context) <-chan int { ? ? dst := make(chan int) ? ? n := 1 ? ? go func() { ? ? ? for { ? ? ? ? select { ? ? ? ? case <-ctx.Done(): ? ? ? ? ? return // returning not to leak the goroutine ? ? ? ? case dst <- n: ? ? ? ? ? n++ ? ? ? ? } ? ? ? } ? ? }() ? ? return dst ? } ? ctx, cancel := context.WithCancel(context.Background()) ? defer cancel() // cancel when we are finished consuming integers ? for n := range gen(ctx) { ? ? fmt.Println(n) ? ? if n == 5 { ? ? ? break ? ? } ? } ? // Output: ? // 1 ? // 2 ? // 3 ? // 4 ? // 5 }
結(jié)構(gòu)分析,gen
是一個函數(shù),返回值是一個信道, for range channel
是有特殊意義的, for會循環(huán)從channel讀數(shù)據(jù),直到channel被close(),不然就是無限循環(huán).
gen內(nèi)部的協(xié)程就是典型的閉包,for range會不斷觸發(fā)讀,gen內(nèi)部的for select 會不斷觸發(fā)寫,主協(xié)程讀5次之后,會結(jié)束main函數(shù),會觸發(fā)defer函數(shù), 也就是取消操作對應(yīng)的回調(diào),此時done信道會被close,gen內(nèi)部的協(xié)程會正常退出.
這個例子是測試支持取消信號的上下文,取消函數(shù)的調(diào)用放在了main
的defer
函數(shù)中.
const shortDuration = 1 * time.Millisecond func ExampleWithDeadline() { ? d := time.Now().Add(shortDuration) ? ctx, cancel := context.WithDeadline(context.Background(), d) ? // Even though ctx will be expired, it is good practice to call its ? // cancellation function in any case. Failure to do so may keep the ? // context and its parent alive longer than necessary. ? defer cancel() ? select { ? case <-time.After(1 * time.Second): ? ? fmt.Println("overslept") ? case <-ctx.Done(): ? ? fmt.Println(ctx.Err()) ? } ? // Output: ? // context deadline exceeded }
deadline
的這個例子,在main
的defer
中也有主動調(diào)用取消函數(shù)的. 實際上通過打印可以顯示deadline是否按預(yù)期工作.
func ExampleWithTimeout() { ? ctx, cancel := context.WithTimeout(context.Background(), shortDuration) ? defer cancel() ? select { ? case <-time.After(1 * time.Second): ? ? fmt.Println("overslept") ? case <-ctx.Done(): ? ? fmt.Println(ctx.Err()) // prints "context deadline exceeded" ? } ? // Output: ? // context deadline exceeded }
timeout
只是deadline
的一種簡寫.
func ExampleWithValue() { ? type favContextKey string ? f := func(ctx context.Context, k favContextKey) { ? ? if v := ctx.Value(k); v != nil { ? ? ? fmt.Println("found value:", v) ? ? ? return ? ? } ? ? fmt.Println("key not found:", k) ? } ? k := favContextKey("language") ? ctx := context.WithValue(context.Background(), k, "Go") ? f(ctx, k) ? f(ctx, favContextKey("color")) ? // Output: ? // found value: Go ? // key not found: color }
context.WithValue
和Context.Value()
是存取操作, 取的時候,如果key沒找到,會返回nil.
2.單元測試
context_text.go,x_test.go
是單元測試, example_test.go
是示例,benchmark_test.go是基準(zhǔn)測試, net_test.go展示了deadline對net包的支持.
先看單元測試的context_text.go.
type testingT interface {} type otherContext struct {} func quiescent(t testingT) time.Duration {} func XTestBackground(t testingT) {} func XTestTODO(t testingT) {} func XTestWithCancel(t testingT) {} func contains(m map[canceler]struct{}, key canceler) bool {} func XTestParentFinishesChild(t testingT) {} func XTestChildFinishesFirst(t testingT) {} func testDeadline(c Context, name string, t testingT) {} func XTestDeadline(t testingT) {} func XTestTimeout(t testingT) {} func XTestCanceledTimeout(t testingT) {} func XTestValues(t testingT) {} func XTestAllocs(t testingT, testingShort func() bool, testingAllocsPerRun func(int, func()) float64) {} func XTestSimultaneousCancels(t testingT) {} func XTestInterlockedCancels(t testingT) {} func XTestLayersCancel(t testingT) {} func XTestLayersTimeout(t testingT) {} func XTestCancelRemoves(t testingT) {} func XTestWithCancelCanceledParent(t testingT) {} func XTestWithValueChecksKey(t testingT) {} func XTestInvalidDerivedFail(t testingT) {} func recoveredValue(fn func()) (v interface{}) {} func XTestDeadlineExceededSupportsTimeout(t testingT) {} type myCtx struct {} type myDoneCtx struct {} func (d *myDoneCtx) Done() <-chan struct{} {} func XTestCustomContextGoroutines(t testingT) {}
這暴露的大多測試函數(shù)的參數(shù)類型是testingT接口類型,但這個源文件中沒有實現(xiàn)testingT
接口的,
func TestBackground(t *testing.T) ? ? ? ? ? ? ? ? ? ? ?{ XTestBackground(t) } func TestTODO(t *testing.T) ? ? ? ? ? ? ? ? ? ? ? ? ? ?{ XTestTODO(t) } func TestWithCancel(t *testing.T) ? ? ? ? ? ? ? ? ? ? ?{ XTestWithCancel(t) } func TestParentFinishesChild(t *testing.T) ? ? ? ? ? ? { XTestParentFinishesChild(t) } func TestChildFinishesFirst(t *testing.T) ? ? ? ? ? ? ?{ XTestChildFinishesFirst(t) } func TestDeadline(t *testing.T) ? ? ? ? ? ? ? ? ? ? ? ?{ XTestDeadline(t) } func TestTimeout(t *testing.T) ? ? ? ? ? ? ? ? ? ? ? ? { XTestTimeout(t) } func TestCanceledTimeout(t *testing.T) ? ? ? ? ? ? ? ? { XTestCanceledTimeout(t) } func TestValues(t *testing.T) ? ? ? ? ? ? ? ? ? ? ? ? ?{ XTestValues(t) } func TestAllocs(t *testing.T) ? ? ? ? ? ? ? ? ? ? ? ? ?{ XTestAllocs(t, testing.Short, testing.AllocsPerRun) } func TestSimultaneousCancels(t *testing.T) ? ? ? ? ? ? { XTestSimultaneousCancels(t) } func TestInterlockedCancels(t *testing.T) ? ? ? ? ? ? ?{ XTestInterlockedCancels(t) } func TestLayersCancel(t *testing.T) ? ? ? ? ? ? ? ? ? ?{ XTestLayersCancel(t) } func TestLayersTimeout(t *testing.T) ? ? ? ? ? ? ? ? ? { XTestLayersTimeout(t) } func TestCancelRemoves(t *testing.T) ? ? ? ? ? ? ? ? ? { XTestCancelRemoves(t) } func TestWithCancelCanceledParent(t *testing.T) ? ? ? ?{ XTestWithCancelCanceledParent(t) } func TestWithValueChecksKey(t *testing.T) ? ? ? ? ? ? ?{ XTestWithValueChecksKey(t) } func TestInvalidDerivedFail(t *testing.T) ? ? ? ? ? ? ?{ XTestInvalidDerivedFail(t) } func TestDeadlineExceededSupportsTimeout(t *testing.T) { XTestDeadlineExceededSupportsTimeout(t) } func TestCustomContextGoroutines(t *testing.T) ? ? ? ? { XTestCustomContextGoroutines(t) }
這是x_test.go
的內(nèi)容,直接是用testing.T
類型來實現(xiàn)testingT接口.
那先分析一下testing.T對testingT接口的實現(xiàn).
type T struct { ? common ? isParallel bool ? context ? ?*testContext } func (t *T) Deadline() (deadline time.Time, ok bool) { ? deadline = t.context.deadline ? return deadline, !deadline.IsZero() }
注意:testing.T.context不是context.Context的實現(xiàn)類型, Deadline()返回了t.context中存儲的deadline信息.
testing.T內(nèi)嵌了testing.common,大部分方法集都來至common:
Error(args ...interface{}) Errorf(format string, args ...interface{}) Fail() FailNow() Failed() bool Fatal(args ...interface{}) Fatalf(format string, args ...interface{}) Helper() Log(args ...interface{}) Logf(format string, args ...interface{}) Name() string Skip(args ...interface{}) SkipNow() Skipf(format string, args ...interface{}) Skipped() bool
Parallel()
是由testing.T
實現(xiàn),某個測試用例多次重復(fù)執(zhí)行時, 可啟用并發(fā)參數(shù).
到此這篇關(guān)于Go語言context test源碼分析詳情的文章就介紹到這了,更多相關(guān)Go語言context test源碼分析內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
golang實現(xiàn)瀏覽器導(dǎo)出excel文件功能
這篇文章主要介紹了golang實現(xiàn)瀏覽器導(dǎo)出excel文件功能,文章通過golang導(dǎo)出excel文件返回給web,實現(xiàn)瀏覽器導(dǎo)出excel文件功能,具有一定的參考價值,需要的小伙伴可以參考一下2022-03-03詳解如何在Go中循環(huán)中使用Defer關(guān)鍵字示例詳解
這篇文章主要為大家介紹了詳解如何在Go中循環(huán)中使用Defer關(guān)鍵字示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-09-09Go語言調(diào)用Shell與可執(zhí)行文件的實現(xiàn)
這篇文章主要介紹了Go語言調(diào)用Shell與可執(zhí)行文件的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-10-10