Go语言重构 go语言重构知乎

调试Go语言的核心转储(Core Dumps)英文原文链接【Go, the unwritten parts】发表于2017/05/22 作者JBD是Go语言开发小组成员
检查程序的执行路径和当前状态是非常有用的调试手段 。核心文件(core file)包含了一个运行进程的内存转储和状态 。它主要是用来作为事后调试程序用的 。它也可以被用来查看一个运行中的程序的状态 。这两个使用场景使调试文件转储成为一个非常好的诊断手段 。我们可以用这个方法来做事后诊断和分析线上的服务(production services) 。
在这篇文章中,我们将用一个简单的hello world网站服务作为例子 。在现实中 , 我们的程序很容易就会变得很复杂 。分析核心转储给我们提供了一个机会去重构程序的状态并且查看只有在某些条件/环境下才能重现的案例 。
作者注 : 这个调试流程只在Linux上可行 。我不是很确定它是否在其它Unixs系统上工作 。macOS对此还不支持 。Windows现在也不支持 。
在我们开始前,需要确保核心转储的ulimit设置在合适的范围 。它的缺省值是0,意味着最大的核心文件大小是0 。我通常在我的开发机器上将它设置成unlimited 。使用以下命令:
接下来,你需要在你的机器上安装 delve。
下面我们使用的 main.go 文件 。它注册了一个简单的请求处理函数(handler)然后启动了HTTP服务 。
让我们编译并生产二进制文件 。
现在让我们假设,这个服务器出了些问题,但是我们并不是很确定问题的根源 。你可能已经在程序里加了很多辅助信息 , 但还是无法从这些调试信息中找出线索 。通常在这种情况下,当前进程的快照会非常有用 。我们可以用这个快照深入查看程序的当前状态 。
有几个方式来获取核心文件 。你可能已经熟悉了奔溃转储(crash dumps) 。它们是在一个程序奔溃的时候写入磁盘的核心转储 。Go语言在缺省设置下不会生产奔溃转储 。但是当你把 GOTRACEBACK 环境变量设置成“crash”,你就可以用 Ctrl backslash 才触发奔溃转储 。如下图所示:
上面的操作会使程序终止,将堆栈跟踪(stack trace)打印出来,并把核心转储文件写入磁盘 。
另外个方法可以从一个运行的程序获得核心转储而不需要终止相应的进程 。gcore 可以生产核心文件而无需使运行中的程序退出 。
根据上面的操作,我们获得了转储而没有终止对应的进程 。下一步就是把核心文件加载进delve并开始分析 。
【Go语言重构 go语言重构知乎】 差不多就这些 。delve的常用操作都可以使用 。你可以backtrace,list , 查看变量等等 。有些功能不可用因为我们使用的核心转储是一个快照而不是正在运行的进程 。但是程序执行路径和状态全部可以访问 。
为什么许多原本的 Java 项目都试图用 go 进行重写开源?项目推倒重构是项目开发大忌 , 一方面我们要尽量避免做项目推倒重构,尽量在前期就规划好,另一方面 , 我们又希望项目能常做小重构 , 这对项目可持续性开发是很有帮助的 。而语言的重构,把Java项目用Go语言重写一遍,无疑是一次重大的推倒重来 。
一、Go语言的优势在哪里
Go语言领先于Java的最大优势,就在于快 。Go语言会被编译成机器代码,直接执行;Java语言则使用JVM运行其代码,这比Go语言要慢了很多 。另外 , Java语言的内存管理,相比于Go语言,也复杂得多,而内存管理,不管对于程序运行,还是对程序员的开发,都极为重要 。最后,Go语言没有引用只有指针,这比Java语言处处引用,又领先了一个身位 。
二、Go语言为什么更适合开源
开源 , 也就是开放源代码 , 最大的好处在于,可以利用全世界的程序员资源,来帮助你完善你的产品,开发新需求,或者修复产品BUG 。这对产品的可持续发展,是非常有帮助的,很多企业纷纷将自己的产品开源,其实就是这个道理 。而Go语言更易学,更易避错,更易阅读等特点,就决定了它更适合用来做开源项目 。
三、Java语言的优势
Java语言是目前软件开发中使用率最广泛,也是最重要的程序之一 , 它的地位,绝对不是目前Go语言可以比拟的 。Java在WEB应用的开发中,有着很重要的地位 。但是,Java语言相对复杂的并发设计,相当庞大的项目体系,使其在开发、测试阶段都略为复杂,在某些方面已经逐步落后于其他语言 。
讲讲go语言的结构体作为C语言家族的一员,go和c一样也支持结构体 。可以类比于java的一个POJO 。
在学习定义结构体之前,先学习下定义一个新类型 。
新类型 T1 是基于 Go 原生类型 int 定义的新自定义类型,而新类型 T2 则是 基于刚刚定义的类型 T1 , 定义的新类型 。
这里要引入一个底层类型的概念 。
如果一个新类型是基于某个 Go 原生类型定义的, 那么我们就叫 Go 原生类型为新类型的底层类型
在上面的例子中,int就是T1的底层类型 。
但是T1不是T2的底层类型 , 只有原生类型才可以作为底层类型,所以T2的底层类型还是int
底层类型是很重要的,因为对两个变量进行显式的类型转换,只有底层类型相同的变量间才能相互转换 。底层类型是判断两个类型本质上是否相同的根本 。
这种类型定义方式通常用在 项目的渐进式重构,还有对已有包的二次封装方面
类型别名表示新类型和原类型完全等价,实际上就是同一种类型 。只不过名字不同而已 。
一般我们都是定义一个有名的结构体 。
字段名的大小写决定了字段是否包外可用 。只有大写的字段可以被包外引用 。
还有一个点提一下
如果换行来写
Age: 66,后面这个都好不能省略
还有一个点,观察e3的赋值
new返回的是一个指针 。然后指针可以直接点号赋值 。这说明go默认进行了取值操作
e3.Age等价于(*e3).Age
如上定义了一个空的结构体Empty 。打印了元素e的内存大小是0 。
有什么用呢?
基于空结构体类型内存零开销这样的特性,我们在日常 Go 开发中会经常使用空 结构体类型元素,作为一种“事件”信息进行 Goroutine 之间的通信
这种以空结构体为元素类建立的 channel,是目前能实现的、内存占用最小的 Goroutine 间通信方式 。
这种形式需要说的是几个语法糖 。
语法糖1:
对于结构体字段 , 可以省略字段名,只写结构体名 。默认字段名就是结构体名
这种方式称为 嵌入字段
语法糖2:
如果是以嵌入字段形式写的结构体
可以省略嵌入的Reader字段 , 而直接访问ReaderName
此时book是一个各个属性全是对应类型零值的一个实例 。不是nil 。这种情况在Go中称为零值可用 。不像java会导致npe
结构体定义时可以在字段后面追加标签说明 。
tag的格式为反单引号
tag的作用是可以使用[反射]来检视字段的标签信息 。
具体的作用还要看使用的场景 。
比如这里的tag是为了帮助encoding/json标准包在解析对象时可以利用的规则 。比如omitempty表示该字段没有值就不打印出来 。
Go语言重构的介绍就聊到这里吧,感谢你花时间阅读本站内容,更多关于go语言重构知乎、Go语言重构的信息别忘了在本站进行查找喔 。

    推荐阅读