Go语言context test源码分析详情
来源:脚本之家
时间:2023-02-24 16:44:58 472浏览 收藏
来到golang学习网的大家,相信都是编程学习爱好者,希望在这里学习Golang相关编程知识。下面本篇文章就来带大家聊聊《Go语言context test源码分析详情》,介绍一下test、语言context、源码,希望对大家的知识积累有所帮助,助力实战开发!
1.测试例子分析
example_test.go,展示了With-系列的4个例子
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 | func ExampleWithCancel() { gen := func(ctx context.Context) <p>结构分析,<code>gen< /code >是一个函数,返回值是一个信道, <code> for range channel< /code >是有特殊意义的, for 会循环从channel读数据,直到channel被close(),不然就是无限循环.< /p > <p>gen内部的协程就是典型的闭包, for range会不断触发读,gen内部的 for select 会不断触发写,主协程读5次之后,会结束main函数,会触发defer函数, 也就是取消操作对应的回调,此时 done 信道会被close,gen内部的协程会正常退出.< /p > <p>这个例子是测试支持取消信号的上下文,取消函数的调用放在了<code>main< /code >的<code>defer< /code >函数中.< /p > <pre class= "brush:bash;" >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 <p><code>deadline< /code >的这个例子,在<code>main< /code >的<code>defer< /code >中也有主动调用取消函数的. 实际上通过打印可以显示deadline是否按预期工作.< /p > <pre class= "brush:bash;" >func ExampleWithTimeout() { ctx, cancel := context.WithTimeout(context.Background(), shortDuration) defer cancel() select { case <p><code>timeout< /code >只是<code>deadline< /code >的一种简写.< /p > <pre class= "brush:bash;" >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 }< /pre > <p><code>context.WithValue< /code >和<code>Context.Value()< /code >是存取操作, 取的时候,如果key没找到,会返回nil.< /p > <h2>2.单元测试< /h2 > <p><code>context_text.go,x_test.go< /code >是单元测试, ex<code>ample_test.go< /code >是示例,benchmark_test.go是基准测试, net_test.go展示了deadline对net包的支持.< /p > <p><strong>先看单元测试的context_text.go.< /strong >< /p > <pre class= "brush:bash;" > 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() <p>这暴露的大多测试函数的参数类型是testingT接口类型,但这个源文件中没有实现<code>testingT< /code >接口的,< /p > <pre class= "brush:bash;" >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) }< /pre > <p>这是<code>x_test.go< /code >的内容,直接是用<code>testing.T< /code >类型来实现testingT接口.< /p > <p><strong>那先分析一下testing.T对testingT接口的实现.< /strong >< /p > <pre class= "brush:bash;" > 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() }< /pre > <blockquote><p><strong>注意:< /strong >testing.T.context不是context.Context的实现类型, Deadline()返回了t.context中存储的deadline信息.< /p >< /blockquote > <p><strong>testing.T内嵌了testing.common,大部分方法集都来至common:< /strong >< /p > <pre class= "brush:bash;" >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< /pre > <p><code>Parallel()< /code >是由<code>testing.T< /code >实现,某个测试用例多次重复执行时, 可启用并发参数.< /p > <p>今天关于《Go语言context test 源码分析详情》的内容介绍就到此结束,如果有什么疑问或者建议,可以在golang学习网公众号下多多回复交流;文中若有不正之处,也希望回复留言以告知!< /p >< /pre >< /pre >< /pre > |
声明:本文转载于:脚本之家 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
-
157 收藏
-
289 收藏
-
475 收藏
-
237 收藏
-
270 收藏
最新阅读
更多>
-
384 收藏
-
244 收藏
-
158 收藏
-
483 收藏
-
328 收藏
-
169 收藏
-
172 收藏
-
348 收藏
-
417 收藏
-
357 收藏
-
345 收藏
-
203 收藏
课程推荐
更多>
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习