浅谈Go语言的高效编码细节

 更新时间:2023年01月09日 15:07:27   作者:阿兵云原生  
这篇文章主要介绍了浅谈Go语言的高效编码细节,我们都知道golang是天生的高并发,高效的编译型语言,可我们也都可知道,工具再好,用法不对,全都白费,我们来举2个常用路径来感受一下

xdm,我们都知道 golang 是天生的高并发,高效的编译型语言

可我们也都可知道,工具再好,用法不对,全都白费,我们来举 2 个常用路径来感受一下

struct和map用谁呢

计算量很小的时候,可能看不出使用 临时 struct 和 map 的耗时差距,但是数量起来了,差距就明显了,且会随着数量越大,差距越发明显

当我们遇到键和值都可以是固定的时候,我们选择 struct 比 选择 map 的方式 高效多了

  • 我们模拟循环计算 1 亿 次,看看使用各自的数据结构会耗时多少
  • 循环前计算一下当前时间
  • 循环后计算一下当前时间
  • 最后计算两个时间的差值,此处我们使用 毫秒为单位
func main() {
	t1 :=time.Now().UnixNano()/1e6
	for i := 0; i < 100000000; i++ {
		var test struct {
			Name  string
			hobby string
		}
		test.Name = "xiaomotong"
		test.hobby = "program"
	}
	t2 :=time.Now().UnixNano()/1e6
	fmt.Println("t2 - t1 == ", t2-t1)
}

程序运行查看效果:

# go run main.go
t1 ==  1634377149185
t2 ==  1634377149221
t2 - t1 ==  36

使用 struct 的方式,耗时 36 ms ,大家感觉这个时间如何?

我们一起来看看使用 map 的方式吧

func main() {
	t1 :=time.Now().UnixNano()/1e6
	fmt.Println("t1 == ", t1)
	for i := 0; i < 100000000; i++ {
		var test = map[string]interface{}{}
		test["name"] = "xiaomotong"
		test["hobby"] = "program"
	}
	t2 :=time.Now().UnixNano()/1e6
	fmt.Println("t2 == ", t2)
	fmt.Println("t2 - t1 == ", t2-t1)
}

程序运行查看效果:

# go run main.go
t1 ==  1634377365927
t2 ==  1634377373525
t2 - t1 ==  7598

使用 struct 的方式,耗时 7598 ms

使用 map 和 使用 struct 的方式,完成同样数据处理,耗时相差 212 倍 , 就这,我们平时编码的时候,对于上述的场景,你会选择哪种数据结构呢?

为什么上述差距会那么大,原因是

在我们可以确定字段的情况下,我们使用 临时的 Struct 在运行期间是不需要动态分配内容的,

可是 map 就不一样,map 还要去检查索引,这一点就非常耗时了

字符串如何拼接是好

工作中编码 xdm 遇到字符串拼接的情况,都是如何实现的呢?我们的工具暂时提供如下几种:

  • 使用 + 的方式
  • 使用 fmt.Sprintf() 的方式
  • 使用 strings.Join 的方式
  • 使用 buffer 的方式

看到这里,也许我们各有各的答案,不过我们还是来实操一遍,看看他们在相同字符串拼接情况下,各自的处理耗时如何

用 + 的方式

我们来计算循环追加 50 万 次字符串,看看耗时多少

func main() {
	t1 := time.Now().UnixNano() / 1e6
	fmt.Println("t1 == ", t1)
	s := "xiao"
	for i := 0; i < 500000; i++ {
		s += "motong"
	}
	t2 := time.Now().UnixNano() / 1e6
	fmt.Println("t2 == ", t2)
	fmt.Println("t2 - t1 == ", t2-t1)
}

程序运行查看效果:

# go run main.go
t1 ==  1634378595642
t2 ==  1634378743119
t2 - t1 ==  147477

看到这个数据 xdm 有没有惊呆了,居然这么慢,耗时 147477 ms 那可是妥妥的 2分27秒呀

Go语言 中使用+处理字符串是很消耗性能的,通过数据我们就可以看出来

使用 fmt.Sprintf() 的方式

func main() {
	t1 := time.Now().UnixNano() / 1e6
	fmt.Println("t1 == ", t1)
	s := "xiao"
	for i := 0; i < 500000; i++ {
		s = fmt.Sprintf("%s%s",s,"motong")
	}
	t2 := time.Now().UnixNano() / 1e6
	fmt.Println("t2 == ", t2)
	fmt.Println("t2 - t1 == ", t2-t1)
}

程序运行查看效果:

# go run main.go
t1 ==  1634378977361
t2 ==  1634379240292
t2 - t1 ==  262931

看到这个数据,咱们也惊呆了,居然耗时 262931 ms,合计 4 分 22秒 ,xdm 是不是没有想到 使用 fmt.Sprintf 比 使用 + 还慢

使用 strings.Join 的方式

func main() {
   t1 := time.Now().UnixNano() / 1e6
   fmt.Println("t1 == ", t1)
   s := []string{}
   s = append(s,"xiao")
   for i := 0; i < 500000; i++ {
      s = append(s ,"motong")
   }
   strings.Join(s,"")
   t2 := time.Now().UnixNano() / 1e6
   fmt.Println("t2 == ", t2)
   fmt.Println("t2 - t1 == ", t2-t1)
}

程序运行查看效果:

# go run main.go
t1 ==  1634570001216
t2 ==  1634570001294
t2 - t1 ==  78

耗时 142923 ms ,合计** 78 ms**

使用 buffer 的方式

使用 buffer 的方式 应该说是最好的方式,

func main() {
	t1 := time.Now().UnixNano() / 1e6
	fmt.Println("t1 == ", t1)
	s := bytes.NewBufferString("xiao")
	for i := 0; i < 500000; i++ {
		s.WriteString("motong")
	}
	t2 := time.Now().UnixNano() / 1e6
	fmt.Println("t2 == ", t2)
	fmt.Println("t2 - t1 == ", t2-t1)
}

# go run main.go
t1 ==  1634378506021
t2 ==  1634378506030
t2 - t1 ==  9

通过上面的数据,我们看到,拼接同样 50 万次的数据

  • 第一种,使用 + 的方式 ,需要 147477 ms
  • 第二种,使用 fmt.Sprintf() 的方式,需要 262931 ms
  • 第三种,使用 strings.Join 的方式,需要 78 ms
  • 第四种,使用 buffer 的方式 ,需要 9ms

使用 buffer 的方式

是 第一种的 16,386 倍 ,是第二种的 29,214 倍 ,是第三种的 8 倍多

xdm ,如果是遇到上面的场景,你会选择使用哪一种方式呢,评论区可以一起讨论一下,是否还有更高效的方式

到此这篇关于浅谈Go语言的高效编码细节的文章就介绍到这了,更多相关Go高效编码内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:

相关文章

  • Go语言测试库testify使用学习

    Go语言测试库testify使用学习

    这篇文章主要为大家介绍了Go语言测试库testify的使用学习示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-07-07
  • Golang实现短网址/短链服务的开发笔记分享

    Golang实现短网址/短链服务的开发笔记分享

    这篇文章主要为大家详细介绍了如何使用Golang实现短网址/短链服务,文中的示例代码讲解详细,具有一定的学习价值,感兴趣的小伙伴可以了解一下
    2023-05-05
  • golang用melody搭建轻量的websocket服务的示例代码

    golang用melody搭建轻量的websocket服务的示例代码

    在Go中,可以使用gin和melody库来搭建一个轻量级的WebSocket服务,gin是一个流行的Web框架,而melody是一个用于处理WebSocket的库,本文给大家演示如何使用gin和melody搭建WebSocket服务,感兴趣的朋友一起看看吧
    2023-10-10
  • Go中的go.mod使用详解

    Go中的go.mod使用详解

    这篇文章主要介绍了Go中的go.mod使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-12-12
  • Go语音开发中常见Error类型处理示例详解

    Go语音开发中常见Error类型处理示例详解

    这篇文章主要为大家介绍了Go语音开发中常见Error类型处理示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-12-12
  • Golang Gin框架实现多种数据格式返回结果详解

    Golang Gin框架实现多种数据格式返回结果详解

    这篇文章主要介绍了Golang Gin框架实现多种数据格式返回结果,我们都知道,一个完整的请求包含请求和处理请求以及结果返回三个步骤,在服务器端对请求处理完成以后,会将结果返回给客户端,在gin框架中,支持返回多种请求数据格式,下面我们一起来看看
    2023-05-05
  • golang jsoniter extension 处理动态字段的实现方法

    golang jsoniter extension 处理动态字段的实现方法

    这篇文章主要介绍了golang jsoniter extension 处理动态字段的实现方法,我们使用实例级别的 extension, 而非全局,可以针对不同业务逻辑有所区分,jsoniter 包提供了比较完善的定制能力,通过例子可以感受一下扩展性,需要的朋友可以参考下
    2023-04-04
  • Golang实现自己的Redis(有序集合跳表)实例探究

    Golang实现自己的Redis(有序集合跳表)实例探究

    这篇文章主要为大家介绍了Golang实现自己的Redis(有序集合跳表)实例探究,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2024-01-01
  • Go 1.21新内置函数min、max和clear的用法详解

    Go 1.21新内置函数min、max和clear的用法详解

    Go 1.21 版本已经正式发布,它带来了许多新特性和改进,其中引入了的三个新内置函数:max、min 和 clear,接下来我们就来看看这些函数的用途和特点吧
    2023-08-08
  • go generate代码自动生成指南

    go generate代码自动生成指南

    这篇文章主要介绍了go generate代码自动生成指南,本文将探讨 go generate 命令的使用方法、原理以及一些实际应用场景,希望读者能够更好地理解和运用这个强大的工具
    2024-01-01

最新评论