Go 中的数据竞争模型

发表于 2年以前  | 总阅读数:314 次

of code (and growing) and contains approximately 2,100 unique Go services (and growing).

Uber 把Golang(简称 Go)作为开发微服务的主要编程语言。我们的 Go monorepo 仓库中大约包含了 5000 万行代码(这个数量还在增长),其中大约包含了 2100 个独立的 Go 服务(这个数量同样也在增长)。

并发作为 Go 中的一等公民;在函数调用之前使用 go 关键字就会以异步的方式调用。这些异步函数调用在 Go 中被称为 goroutines。开发者通过创建 goroutines 来缩短延迟时间(比如在 IO 或 RPC 调用等场景)。多个 goroutine 之间可以通过 (channels 来传递消息或者使用 共享内存 ) 来进行通信。共享内存恰好是 Go 中最常用的通信方式。

Go 程序员可以随意使用 goroutines, 因为它们被认为是 “轻量级” 并且创建 goroutines 是一件很容易的事。最后,我们注意到,在使用 Go 编写的程序通常比使用其他语言编写的程序表现出更高的并发性能。例如,通过扫描运行我们数据中心的数十万个微服务实例,我们发现使用 Go 编写的微服务表现出的并发能力大约是 java 的 8 倍。更高的并发也意味着可能发生更多并发错误。数据竞争是两个或者多个 goroutines 同时以无序并且至少有一个是以写的方式访问同一个数据时产生的并发错误。数据竞争是潜在的 bug,必须 不惜一切代价避免。

我们使用动态数据竞争检测技术开发了一个系统来检测 Uber 的数据竞争。这个系统在六个月的时间里,在我们的 Go 代码库中检测到大约 2,000 个数据竞争,其中我们的开发人员已经修复了大约 1,100 个数据竞争。

在这篇博客中,我们将展示我们在 Go 程序中发现的各种数据竞争模型。这项研究是通过分析 210 位独特的开发人员在六个月内修复的 1,100 多个数据竞争进行的。总的来说,我们注意到,由于某些语言设计选择,使得 Go 更容易引入数据竞争。其中语言特性和数据竞争之间存在复杂的相互作用。

Go 中的数据竞争模型_

我们研究了由开发者修复的约 1,100 个数据竞争的问题,并将它们划分为不同的类别。我们对这些数据竞争的研究显示了导致 Go 数据竞争的一些常见模型和一些神秘的原因:

1. 在 Go 的设计中,goroutines 可以以引用的方式不加限制地捕捉自由引用变量是引发数据竞争的重要因素

嵌套函数(又称闭包),在 Go 中可以通过引用的方式捕获所有自由变量。程序员不需要明确指定在闭包语法中捕获哪些自由变量。

这种使用模式与 Java 和 C ++ 不同。Java lambdas 只能够捕捉值,他们有意识地采取了该设计选择来避免并发错误 [1, 2]。C ++ 要求开发者明确指定按值还是引用的方式指定捕获方式。

尤其是在大闭包的情况下, 开发者通常不知道闭包内部使用的变量是通过引用方式捕获的自由变量。很多时候,Go 开发者通常会使用闭包函数作为 goroutines。由于使用引用捕捉和并发执行 goroutine, 除非有显式的同步操作,否则 Go 程序最终会以无序访问这些自由变量。我们用下面三个示例证明了这一点:

示例 1:在循环中捕捉临时变量引起的数据竞争

图 1A 中的代码显示了迭代 Go jobs 切片并通过 ProcessJob 函数处理每个 job 变量。

在这里,开发者将耗时的 ProcessJob 函数包装在匿名函数中,并且在每次循环遍历的时候都以 goroutine 的方式启动。但是,在循环中是在 goroutine 内部通过引用的方式来捕捉变量 job。当第一个循环迭代启动的 goroutine 访问了变量 job 时,父 goroutine 中的 for 循环将通过 slice 前进并更新变量 job 的值,以指向 slice 中的第二个元素,从而导致数据竞争。这种类型的数据竞争发生在循环体中对传值和引用类型元素进行读和写访问(包括切片,数组和 map)。Go 推荐一个编码习惯来隐藏和私有化循环体中的循环索引变量,很遗憾,开发者并不总是遵守这些规范。

示例 2:由于习惯捕获错误而引发的数据竞争。

图 1B:由于习惯捕获错误而引发的数据竞争。

Go 提倡函数有多个返回值。实际上,如图 1B 所示,函数除了实际的返回值以外,还会返回一个用于指示是否发生错误的错误对象。仅当错误值为 nil 时,实际的返回值才会被认为是有意义的。一种常见的做法是将返回的错误对象分配给名称为 err 的变量,然后检查其零值。但是,由于可以在函数主体内调用多个返回错误的函数,因此每次都会对 err 变量进行多次赋值,然后每次进行空值检查。当开发者将此惯用方式与 goroutine 混合在一起时,在闭包中通过引用捕获 err 。最终,在闭包函数(或多个 goroutine 实例)中对相同的 err 变量同时读取并将其写入,这就会导致数据竞争。

示例 3:由于捕捉命名返回变量而引起的数据竞争。

图 1C:由于捕捉命名返回变量而引起的数据竞争。

Go 引入了一种 命名返回值 的语法糖。命名返回变量被看作是定义在函数顶部的变量,其作用范围超出了函数的主体。函数的返回分两种,没有参数的返回语句的裸返回和命名的返回值。在存在闭包的情况下,将正常(非裸返回)返回值与命名返回混合使用或在命名返回的函数中使用延迟返回是有风险的,因为它会引入数据竞争。

图 1C 中名为 NamedReturnCallee 的函数返回名为 result 的整型值。由于这个语法的特性,在函数体中都可以对 result 变量进行读写操作。如果该函数在第 4 行裸返回第 2 行的分配 result = 10 的结果,则第 13 行的 Caller 将看到返回值为 10。编译器将会复制 result 的值并赋值给变量 retVal。如第 9 行所示,命名的返回函数还可以使用标准返回语法。这种情况下,编译器会将 20 赋值给命名返回变量 result。第 6 行创建了一个 goroutine,该 goroutine 捕获了命名的返回变量 result。在编写这个 goroutine 时,即使是并发专家也可能会认为,第 7 行的读操作是安全的,因为没有其他地方对同一个变量进行写入操作。毕竟,第 9 行上的语句返回常量 20 ,并且似乎没有触及命名的返回变量结果。但是,像前面所说的一样,这份代码,将 return 20 这个语句转换为写入结果。现在我们突然对共享结果变量进行并发读取和写入,这是数据竞争的一种情况。

2.切片令人困惑的类型,产生出微妙而难以诊断的数据竞争

切片 是一种动态数组的引用类型。一个 切片 内部包含指向基础数组的指针,当前长度和基础数组可以扩展的最大容量。为了易于讨论,我们将这些变量称为切片的元字段。切片上的一个常见操作是通过追加数据操作而使它扩容。当大小达到容量时,进行了新的分配(例如,有可能是当前 len 的两倍),并更新了元字段的值。当多个 goroutines 同时访问切片时,自然可以通过 mutex 访问这些受保护的信息。

图 2:即使在使用锁后,切片也有可能产生数据竞争。

在图 2 中,开发者认为,第 6 行使用锁对切片进行 append 操作足以避免数据竞争。但是,当将切片作为参数传递给第 14 行的 goroutine 时,就会发生数据竞争,而这并未受到锁的保护。goroutine 的调用导致切片的元字段从调用方(第 14 行)拷贝到 Callee(第 11 行)。鉴于切片是一种引用类型,开发者假设其通过(拷贝)向 Callee 引起了数据竞争。但是,切片与指针类型不同(元字段按值拷贝),因此是微妙的数据竞争。

3.同时访问 Go 内置的线程不安全的 map 会导致频繁的数据竞争

图 3:由于并发访问 map 而引起的数据竞争。

哈希表(map)是 Go 中内置的非线程安全的数据结构。如果多个 goroutines 同时访问同一哈希表,其中至少一个试图修改哈希表(插入或删除其中的数据),就会出现数据竞争问题。我们观察到开发者做出了一个基本但普遍的假设(这是一个错误的假设),即哈希表中的不同条目可以同时访问。这种假设源于开发者用于哈希表访问的 table[key] 语法并将它们误解为访问不相交的元素。但是,和数组或切片不同,map (哈希表)是稀疏的数据结构,访问一个元素可能会导致访问另一个元素。如果在同一过程中发生另一个插入/删除,它将修改稀疏数据结构并引起数据竞争。我们甚至发现了更复杂的并发访问哈希表引起的数据竞争,原因是同一个哈希表被传递到深度调用路径,并且开发者忘记了这些调用路径通过异步的 goroutine 改变哈希表的事实。图 3 显示了此类数据竞争的示例。

虽然哈希表导致的数据竞争并不是 Go 独有的,但以下原因使其更容易在 Go 中发生数据竞争:

  1. Go 开发者比其他语言的开发者更频繁地使用 map,因为 map 是语言内置的数据结构。例如,在我们的 Java 存储库中,我们发现每个 MLoC 有 4,389 个 map 结构,而 Go 同样是每 MLoC 5,950,高出 1. 34 倍。
  2. 哈希表访问语法就像数组访问语法(与 Java 的 get/put API 不同)一样,使其易于使用,这种访问语法不出意外地混淆了对数据结构的随机访问。在 Go 中,可以使用 table[key] 语法轻松查询不存在的 map 元素,该语法只需返回默认值而不会产生任何错误。这种容错性让开发者在 Go 中使用 map 时感到很满意。

4. Go 开发者经常在传递值(或在方法上传值)方面犯错,这可能导致非比寻常的数据竞争

按值传递语义是 Go 推荐 ,因为它简化了逃逸分析并为变量提供了更好的在栈上分配的机会,这减少了垃圾回收的压力。

和 Java 中所有对象都是引用类型不同的是,在 Go 中,对象可以是值类型(struct)或引用类型(接口)。这两者在语法方面没有差异,但是这会导致不正确使用同步结构体,例如 sync.RWMutex,在是 Go 中的值类型(结构)。如果一个函数创建了一个互斥体结构并以传值的方式给多个 goroutine 调用,这些并发执行的 goroutine 在不共享内部状态的互斥对象上执行操作。如下图 4 所示,这会破坏对受保护的共享内存区域的互斥访问。

图 4A:由于方法调用按引用或按指针引起的数据竞争。

图 4B:Sync.MutexLock/Unlock 签名。

由于 Go 语法对于在指针和值上调用方法是相同的,因此开发者不太关注 m.Lock() 使用的是拷贝而不是指针的问题。调用者仍然可以在互斥量值上调用这些 API,并且编译器会透明地使用传地址的方式调用。如果不存在这种透明度,则该错误可能会被编译器检测为类型不匹配的错误。

当开发者意外地实现了一个方法,其中接收者是指向结构的指针而不是结构的值拷贝时,就会发生与这种情况相反的情况。在这钟情况下,调用这个方法的多个 goroutine 最终会意外地共享结构的相同内部状态,而可能不是开发者开发者想要的。在这里,调用者也没有意识到值类型在接收者被透明地转换为指针类型。

5.混合使用(通道)消息传递和共享内存会使代码变得复杂并且容易受到数据竞争的影响

图 5:将消息传递与共享内存同时使用引起的数据竞争。

图 5 显示的是一个由开发者使用 channel 的信号和等待机制实现的一个泛型的 future。这个 Future 可以通过调用 Start() 方法来启动,并且可以通过调用 Future 的 Wait() 方法来阻塞直到 Future 的完成。Start() 方法创建一个 goroutine,它执行注册到 Future 的函数并记录其返回值(响应和错误)。goroutine 通过在通道 ch 上发送一条消息来向 Wait() 方法发出 future 完成的信号,如第 6 行所示。对称地,Wait() 方法阻塞以从 channel 中获取消息(第 11 行)。

Go 中的 Contexts 在横跨 API 和进程之间携带截止日期、取消信号和其他请求相关的值。这种为任务设置截止时间的做法在微服务开发中很常见。因此,Wait() 在上下文被取消(第 13 行)或 阻塞直到 future 完成(第 11 行)。此外,Wait() 包含在 select 语句(第 10 行)中,这个语句会阻塞,直到至少有一个分支是 ready 状态的。

如果上下文超时,则相应的案例在第 14 行将 Future 的 err 字段记录为 ErrCancelled。这个对 err 的写入与第 5 行对 future 相同变量的写入竞争。

6. sync.WaitGroup 为 Go 在其组同步结构中提供了更多选择,但 Add/Done 方法的不正确使用也会导致数据竞争

图 6A:由于 WaitGroup.Add() 使用不正确导致的数据竞争。

sync.WaitGroup 结构是 Go 中的组同步结构。不同于 C++ 屏障,pthreads 或 Java 屏障 或 latch 结构,WaitGroup 中的参与者数量不是在构造时确定的,而是动态更新的。在 WaitGroup 对象上允许进行三种操作 - Add(int)Done()Wait()Add() 增加参与者的 count 值,并且 Wait() 阻塞直到 Done() 被调用 count 次(通常每个参与者都会调用一次)。 WaitGroup 在 Go 中被广泛使用。如表 1 所示,Go 中的组同步比 Java 高 1.9 倍。

在图 6 中,开发者打算创建与切片 itemId 中的元素数量一样多的 goroutine 来同时处理这些 item。每个 goroutine 在不同索引的结果切片中记录其成功或失败状态,并在第 12 行记录父功能块,直到所有 goroutine 完成。然后它依次访问结果的所有元素以计算成功处理的数量。

为了使此代码正常工作,当在第 12 行调用 Wait() 时,注册参与者的数量必须已经等于 itemIds 的长度。仅当 wg.Add(1) 在调用 wg.Wait() 之前执行的次数与 itemIds 的长度一样多时才有可能,这意味着 wg.Add(1) 应该在每个 goroutine 之前放置在第 5 行调用。但是,开发者在第 7 行错误地将 wg.Add(1) 放置在 goroutine 的主体中,这不能保证在外部函数 WaitGrpExample 调用 Wait() 时已经执行。因此,调用 Wait() 时,注册到 WaitGroupitemId 的长度可能会少于此长度。出于这个原因,Wait() 可以提前解除阻塞,并且 WaitGrpExample 函数可以开始从切片结果中读取(第 13 行),同时一些 goroutine 正在同时写入同一个切片。

我们还发现过早将调用 Waitgroup 上的 wg.Done() 方法会导致数据竞争。如下图 B 所示,过早地调用 wg.Done() 与 Go 的 defer 语句交互的结果的微妙版本。当遇到多个 defer 语句时,按照后进先出的顺序执行。在这里,第 9 行的 wg.Wait()doCleanup() 运行之前完成。父 goroutine 在第 10 行访问 locationErr,而子 goroutine 可能仍在延迟的 doCleanup() 函数内写入 locationErr(为简洁起见并未显示)。

图 6B:由于 defer 语句调用WaitGroup.Done() 顺序导致的数据竞争错误。

7.为 Go 的表驱动测试套件用法并行运行测试通常会导致生产或测试代码中的数据竞争

测试 是 Go 中的内置功能。带有后缀_test.go 的文件中的前缀测试的任何功能都可以通过 Go 构建系统作为测试运行。如果测试代码调用 API testing.T.Parallel() ,Go 会并发运行这些测试用例。我们发现,由于这种并发执行的测试引发了大量的数据竞争。这些数据竞争的根本原因有时是在测试代码中,有时是在生产代码中。另外,在单个测试函数中,Go 开发者通常会编写许多子测验,并通过 Go 提供的测试工具软件包执行它们。Go 建议使用表格驱动测试习惯 来编写和运行一个测试套件。我们的系统并行地执行我们开发者编写的数十个或者数百个自测试时。这个习惯用法成为测试套件问题的根源,开发人员要么假设串行测试执行,要么忘记在大型复杂测试套件中使用共享对象。当生产 API 编写时没有线程安全(可能是因为不需要它),但被并行调用时,也会出现问题,这违反了假设。

总结

我们分析了修复的数据竞争问题,以对其背后的原因进行分类。这些问题被列为如下。这些标签并不是互斥的。

图 7:数据竞争的摘要。

从此链接获得了每个数据竞争模式的一个示例。

总而言之,根据观察到的(包括已经修复的)数据竞争,我们详细阐述了在 Go 程序中更加容易引入数据竞争问题的范式。我们希望我们在 Go 中进行数据竞争的经验帮助 Go 开发者更多地关注编写巧妙的并发代码。未来的编程语言设计师应仔细权衡不同的语言功能和编码习惯,减少创建常见和隐蔽的并发问题。

不足

本文的讨论是基于我们在 Uber 的 Go Monorepo 中进行数据竞争的经验,并且可能错过了其他可能发生数据竞争的模式。同样,由于动态竞争检测并未检测到由于代码和交织的覆盖而无法检测到所有有可能引起的数据竞争问题,因此我们的讨论可能错过了一些数据竞争模式。尽管对我们结果的普遍性有这些不足,但本文和部署经验中有关模式的讨论独立存在。

这是有关我们在 GO 代码中数据竞争的经验的两部分博客文章系列中的第二个。我们的经验的详尽版本将出现在 ACM Sigplan 编程语言设计与实施(PLDI),2022 年。在博客系列的第一部分中,我们讨论了与以 GO 代码为大规模部署动态竞争检测有关的学习。

本文由哈喽比特于2年以前收录,如有侵权请联系我们。
文章来源:https://mp.weixin.qq.com/s/UiVWvDRYcU209xCtdjYdWg

 相关推荐

刘强东夫妇:“移民美国”传言被驳斥

京东创始人刘强东和其妻子章泽天最近成为了互联网舆论关注的焦点。有关他们“移民美国”和在美国购买豪宅的传言在互联网上广泛传播。然而,京东官方通过微博发言人发布的消息澄清了这些传言,称这些言论纯属虚假信息和蓄意捏造。

发布于:1年以前  |  808次阅读  |  详细内容 »

博主曝三大运营商,将集体采购百万台华为Mate60系列

日前,据博主“@超能数码君老周”爆料,国内三大运营商中国移动、中国电信和中国联通预计将集体采购百万台规模的华为Mate60系列手机。

发布于:1年以前  |  770次阅读  |  详细内容 »

ASML CEO警告:出口管制不是可行做法,不要“逼迫中国大陆创新”

据报道,荷兰半导体设备公司ASML正看到美国对华遏制政策的负面影响。阿斯麦(ASML)CEO彼得·温宁克在一档电视节目中分享了他对中国大陆问题以及该公司面临的出口管制和保护主义的看法。彼得曾在多个场合表达了他对出口管制以及中荷经济关系的担忧。

发布于:1年以前  |  756次阅读  |  详细内容 »

抖音中长视频App青桃更名抖音精选,字节再发力对抗B站

今年早些时候,抖音悄然上线了一款名为“青桃”的 App,Slogan 为“看见你的热爱”,根据应用介绍可知,“青桃”是一个属于年轻人的兴趣知识视频平台,由抖音官方出品的中长视频关联版本,整体风格有些类似B站。

发布于:1年以前  |  648次阅读  |  详细内容 »

威马CDO:中国每百户家庭仅17户有车

日前,威马汽车首席数据官梅松林转发了一份“世界各国地区拥车率排行榜”,同时,他发文表示:中国汽车普及率低于非洲国家尼日利亚,每百户家庭仅17户有车。意大利世界排名第一,每十户中九户有车。

发布于:1年以前  |  589次阅读  |  详细内容 »

研究发现维生素 C 等抗氧化剂会刺激癌症生长和转移

近日,一项新的研究发现,维生素 C 和 E 等抗氧化剂会激活一种机制,刺激癌症肿瘤中新血管的生长,帮助它们生长和扩散。

发布于:1年以前  |  449次阅读  |  详细内容 »

苹果据称正引入3D打印技术,用以生产智能手表的钢质底盘

据媒体援引消息人士报道,苹果公司正在测试使用3D打印技术来生产其智能手表的钢质底盘。消息传出后,3D系统一度大涨超10%,不过截至周三收盘,该股涨幅回落至2%以内。

发布于:1年以前  |  446次阅读  |  详细内容 »

千万级抖音网红秀才账号被封禁

9月2日,坐拥千万粉丝的网红主播“秀才”账号被封禁,在社交媒体平台上引发热议。平台相关负责人表示,“秀才”账号违反平台相关规定,已封禁。据知情人士透露,秀才近期被举报存在违法行为,这可能是他被封禁的部分原因。据悉,“秀才”年龄39岁,是安徽省亳州市蒙城县人,抖音网红,粉丝数量超1200万。他曾被称为“中老年...

发布于:1年以前  |  445次阅读  |  详细内容 »

亚马逊股东起诉公司和贝索斯,称其在购买卫星发射服务时忽视了 SpaceX

9月3日消息,亚马逊的一些股东,包括持有该公司股票的一家养老基金,日前对亚马逊、其创始人贝索斯和其董事会提起诉讼,指控他们在为 Project Kuiper 卫星星座项目购买发射服务时“违反了信义义务”。

发布于:1年以前  |  444次阅读  |  详细内容 »

苹果上线AppsbyApple网站,以推广自家应用程序

据消息,为推广自家应用,苹果现推出了一个名为“Apps by Apple”的网站,展示了苹果为旗下产品(如 iPhone、iPad、Apple Watch、Mac 和 Apple TV)开发的各种应用程序。

发布于:1年以前  |  442次阅读  |  详细内容 »

特斯拉美国降价引发投资者不满:“这是短期麻醉剂”

特斯拉本周在美国大幅下调Model S和X售价,引发了该公司一些最坚定支持者的不满。知名特斯拉多头、未来基金(Future Fund)管理合伙人加里·布莱克发帖称,降价是一种“短期麻醉剂”,会让潜在客户等待进一步降价。

发布于:1年以前  |  441次阅读  |  详细内容 »

光刻机巨头阿斯麦:拿到许可,继续对华出口

据外媒9月2日报道,荷兰半导体设备制造商阿斯麦称,尽管荷兰政府颁布的半导体设备出口管制新规9月正式生效,但该公司已获得在2023年底以前向中国运送受限制芯片制造机器的许可。

发布于:1年以前  |  437次阅读  |  详细内容 »

马斯克与库克首次隔空合作:为苹果提供卫星服务

近日,根据美国证券交易委员会的文件显示,苹果卫星服务提供商 Globalstar 近期向马斯克旗下的 SpaceX 支付 6400 万美元(约 4.65 亿元人民币)。用于在 2023-2025 年期间,发射卫星,进一步扩展苹果 iPhone 系列的 SOS 卫星服务。

发布于:1年以前  |  430次阅读  |  详细内容 »

𝕏(推特)调整隐私政策,可拿用户发布的信息训练 AI 模型

据报道,马斯克旗下社交平台𝕏(推特)日前调整了隐私政策,允许 𝕏 使用用户发布的信息来训练其人工智能(AI)模型。新的隐私政策将于 9 月 29 日生效。新政策规定,𝕏可能会使用所收集到的平台信息和公开可用的信息,来帮助训练 𝕏 的机器学习或人工智能模型。

发布于:1年以前  |  428次阅读  |  详细内容 »

荣耀CEO谈华为手机回归:替老同事们高兴,对行业也是好事

9月2日,荣耀CEO赵明在采访中谈及华为手机回归时表示,替老同事们高兴,觉得手机行业,由于华为的回归,让竞争充满了更多的可能性和更多的魅力,对行业来说也是件好事。

发布于:1年以前  |  423次阅读  |  详细内容 »

AI操控无人机能力超越人类冠军

《自然》30日发表的一篇论文报道了一个名为Swift的人工智能(AI)系统,该系统驾驶无人机的能力可在真实世界中一对一冠军赛里战胜人类对手。

发布于:1年以前  |  423次阅读  |  详细内容 »

AI生成的蘑菇科普书存在可致命错误

近日,非营利组织纽约真菌学会(NYMS)发出警告,表示亚马逊为代表的电商平台上,充斥着各种AI生成的蘑菇觅食科普书籍,其中存在诸多错误。

发布于:1年以前  |  420次阅读  |  详细内容 »

社交媒体平台𝕏计划收集用户生物识别数据与工作教育经历

社交媒体平台𝕏(原推特)新隐私政策提到:“在您同意的情况下,我们可能出于安全、安保和身份识别目的收集和使用您的生物识别信息。”

发布于:1年以前  |  411次阅读  |  详细内容 »

国产扫地机器人热销欧洲,国产割草机器人抢占欧洲草坪

2023年德国柏林消费电子展上,各大企业都带来了最新的理念和产品,而高端化、本土化的中国产品正在不断吸引欧洲等国际市场的目光。

发布于:1年以前  |  406次阅读  |  详细内容 »

罗永浩吐槽iPhone15和14不会有区别,除了序列号变了

罗永浩日前在直播中吐槽苹果即将推出的 iPhone 新品,具体内容为:“以我对我‘子公司’的了解,我认为 iPhone 15 跟 iPhone 14 不会有什么区别的,除了序(列)号变了,这个‘不要脸’的东西,这个‘臭厨子’。

发布于:1年以前  |  398次阅读  |  详细内容 »
 相关文章
Android插件化方案 5年以前  |  237227次阅读
vscode超好用的代码书签插件Bookmarks 2年以前  |  8063次阅读
 目录