当前位置: 首页 > news >正文

Go 语言一些常用语法编写和优化指南

Go 语言以其简洁的语法和强大的并发性能而受到开发者的喜爱。然而,为了充分利用 Go 的潜力,我们需要了解如何优化 Go 程序。本文将介绍一些常见的 Go 语言优化技巧,并通过实际例子进行说明。

推荐系列
来来来,老铁们,男人女人都需要的技术活 拿去不谢:远程调试,发布网站到公网演示,远程访问内网服务,
福利链接

1. 利用 sync.Pool 减少内存分配

在 Go 中,频繁的内存分配和释放可能会导致性能问题。sync.Pool 可以用于存储和复用临时对象,从而减少内存分配和垃圾回收的开销。

var bufPool = sync.Pool{New: func() interface{} {return make([]byte, 1024)},
}buf := bufPool.Get().([]byte)
// 使用 buf...
bufPool.Put(buf)

在这个例子中,我们创建了一个 sync.Pool 来存储字节切片。当我们需要一个字节切片时,我们首先尝试从池中获取,如果池中没有可用的对象,那么 New 函数就会被调用来创建一个新的字节切片。使用完字节切片后,我们将其放回池中,以便后续的复用。

2. 使用缓冲通道进行异步操作

Go 的通道(channel)是一种在 goroutine 之间进行通信的机制。缓冲通道可以用于异步操作,从而提高程序的并发性能。

ch := make(chan int, 100) // 创建一个缓冲大小为100的通道go func() {for i := 0; i < 100; i++ {ch <- i // 向通道发送数据}close(ch)
}()for i := range ch { // 从通道接收数据fmt.Println(i)
}

在这个例子中,我们创建了一个缓冲大小为 100 的通道。然后我们启动了一个 goroutine 来向通道发送数据,主 goroutine 从通道接收数据。由于通道是缓冲的,所以发送者和接收者可以并行工作,从而提高了程序的并发性能。

3. 利用 pprof 进行性能分析

Go 标准库中的 net/http/pprof 包提供了一种方便的方式来分析 Go 程序的性能。我们可以通过添加一些简单的代码来启动一个 HTTP 服务器,然后通过 pprof 工具来获取和分析性能数据。

import _ "net/http/pprof"go func() {log.Println(http.ListenAndServe("localhost:6060", nil))
}()

在这个例子中,我们启动了一个运行在 localhost:6060 的 HTTP 服务器。然后我们可以通过 go tool pprof http://localhost:6060/debug/pprof/profile 命令来获取 CPU profile,或者通过 go tool pprof http://localhost:6060/debug/pprof/heap 命令来获取内存 profile。

4. 使用 strings.Builder 进行字符串拼接

在 Go 中,字符串是不可变的,这意味着每次字符串拼接操作都会创建一个新的字符串。如果你需要进行大量的字符串拼接操作,这可能会导致大量的内存分配和垃圾回收。strings.Builder 是 Go 语言中用于高效字符串拼接的工具。

var builder strings.Builderfor i := 0; i < 1000; i++ {builder.WriteString("Hello, World!")
}result := builder.String()

在这个例子中,我们使用 strings.Builder 来进行 1000 次字符串拼接操作。与直接使用 ++= 进行字符串拼接相比,strings.Builder 可以显著提高性能。

5. 利用 time.After 避免 goroutine 泄露

在 Go 中,如果一个 goroutine 在完成任务后没有被正确地关闭,那么它可能会一直占用内存,这被称为 goroutine 泄露。time.After 是一种常用的防止 goroutine 泄露的技巧。

func doSomethingWithTimeout(timeout time.Duration) {done := make(chan bool)go func() {// 做一些耗时的操作...done <- true}()select {case <-done:// 操作成功完成case <-time.After(timeout):// 操作超时}
}

在这个例子中,我们启动了一个 goroutine 来执行一些耗时的操作,然后使用 select 语句等待操作的完成或超时。如果操作在超时时间内完成,那么 done 通道会接收到一个值,select 语句会退出。如果操作在超时时间内没有完成,那么 time.After 会发送一个值,select 语句会退出,goroutine 会被正确地关闭。

6. 使用 strconv 而不是 fmt 进行字符串转换

在 Go 中,fmt.Sprintf 是一种常用的将其他类型的值转换为字符串的方法。然而,fmt.Sprintf 的性能通常不如 strconv 包中的函数。

s := fmt.Sprintf("%d", 123) // 不推荐s := strconv.Itoa(123) // 推荐

在这个例子中,我们比较了 fmt.Sprintfstrconv.Itoa 两种将整数转换为字符串的方法。虽然 fmt.Sprintf 更灵活,但 strconv.Itoa 的性能更好。

7. 使用索引访问切片元素

在 Go 中,使用 range 循环遍历切片是一种常见的做法。然而,如果你只需要访问切片的元素,而不需要元素的索引,那么使用索引访问元素通常会有更好的性能。

for i := range slice {_ = slice[i] // 推荐
}for _, v := range slice {_ = v // 不推荐
}

在这个例子中,我们比较了使用 range 循环和使用索引访问切片元素的两种方法。虽然使用 range 循环更简洁,但使用索引访问元素的性能更好。

8. 避免在循环中创建 goroutine

在 Go 中,go 关键字可以用于创建新的 goroutine。然而,如果你在循环中创建 goroutine,那么可能会导致大量的 goroutine 被创建,从而消耗大量的内存。

for _, v := range slice {go func(v int) {// 处理 v...}(v)
}

在这个例子中,我们在循环中为每个元素创建了一个新的 goroutine。虽然这样可以并行处理元素,但如果切片的大小很大,那么可能会创建大量的 goroutine,从而消耗大量的内存。因此,我们应该避免在循环中创建 goroutine,或者使用一些技术(如使用 sync.WaitGroup 或者使用通道)来限制 goroutine 的数量。

Go 语言优化指南(续)

在前几篇文章中,我们已经介绍了一些常见的 Go 语言优化技巧。在这篇文章中,我们将继续探讨更多的优化技巧,并通过实际例子进行说明。

9. 使用 sync.Map 进行并发安全的映射操作

在 Go 中,内置的 map 类型不是并发安全的,这意味着你不能在多个 goroutine 中同时对同一个 map 进行读写操作。sync.Map 是 Go 语言中用于并发安全的映射操作的工具。

var m sync.Mapm.Store("hello", "world") // 存储键值对value, ok := m.Load("hello") // 加载键值对
if ok {fmt.Println(value)
}

在这个例子中,我们使用 sync.Map 来存储和加载键值对。与内置的 map 相比,sync.Map 的性能可能稍微差一些,但它可以在多个 goroutine 中安全地使用。

10. 利用 context 包进行超时和取消操作

在 Go 中,context 包提供了一种在 API 边界之间传递超时、取消信号以及其他请求范围的值的机制。

ctx, cancel := context.WithTimeout(context.Background(), time.Second)
defer cancel()select {
case <-time.After(2 * time.Second):fmt.Println("overslept")
case <-ctx.Done():fmt.Println(ctx.Err())
}

在这个例子中,我们创建了一个会在一秒后自动取消的 context。然后我们等待两秒或 context 被取消。由于 context 会在一秒后被取消,所以 ctx.Err() 会返回一个错误,表明 context 已经被取消。

11. 使用 atomic 包进行并发安全的操作

在 Go 中,sync/atomic 包提供了一些原子操作函数,可以用于实现并发安全的计数器、标志等。

var counter int64go func() {for {atomic.AddInt64(&counter, 1)time.Sleep(time.Millisecond)}
}()go func() {for {fmt.Println(atomic.LoadInt64(&counter))time.Sleep(time.Second)}
}()

在这个例子中,我们创建了一个并发安全的计数器。一个 goroutine 每毫秒将计数器加一,另一个 goroutine 每秒打印计数器的当前值。由于我们使用了 atomic 包中的函数,所以这个计数器在多个 goroutine 中是安全的。

12. 利用 reflect 包进行动态操作

在 Go 中,reflect 包提供了一种在运行时动态操作对象的机制,包括获取对象的类型和值、调用方法等。

v := reflect.ValueOf(123)
t := reflect.TypeOf(123)fmt.Println(v.Int()) // 输出:123
fmt.Println(t.Name()) // 输出:int

在这个例子中,我们使用 reflect 包获取了一个整数的值和类型。虽然 reflect 包非常强大,但它的性能通常不如静态类型的操作,所以我们应该谨慎使用。

13. 使用 sort 包进行高效排序

Go 语言的 sort 包提供了一系列函数用于对切片和自定义数据结构进行排序。

nums := []int{3, 1, 4, 1, 5, 9, 2, 6, 5, 3, 5}
sort.Ints(nums)
fmt.Println(nums) // 输出:[1 1 2 3 3 4 5 5 5 6 9]

在这个例子中,我们使用 sort.Ints 函数对一个整数切片进行排序。sort 包还提供了其他函数,如 sort.Float64ssort.Strings 等,用于对特定类型的切片进行排序。

14. 利用 encoding/json 包进行 JSON 操作

Go 语言的 encoding/json 包提供了一系列函数用于处理 JSON 数据。

type Person struct {Name string `json:"name"`Age  int    `json:"age"`
}jsonStr := `{"name":"John","age":30}`
var p Person
json.Unmarshal([]byte(jsonStr), &p)
fmt.Println(p) // 输出:{John 30}

在这个例子中,我们定义了一个 Person 结构体,并使用 json.Unmarshal 函数将一个 JSON 字符串解析到这个结构体中。encoding/json 包还提供了其他函数,如 json.Marshal 等,用于将 Go 数据结构转换为 JSON 字符串。

推荐系列

来来来,老铁们,男人女人都需要的技术活 拿去不谢:远程调试,发布网站到公网演示,远程访问内网服务,
福利链接

相关文章:

Go 语言一些常用语法编写和优化指南

Go 语言以其简洁的语法和强大的并发性能而受到开发者的喜爱。然而&#xff0c;为了充分利用 Go 的潜力&#xff0c;我们需要了解如何优化 Go 程序。本文将介绍一些常见的 Go 语言优化技巧&#xff0c;并通过实际例子进行说明。 推荐系列 来来来,老铁们,男人女人都需要的技术活…...

Golang 语法系列:结构体

结构体&#xff1a;相当于"类" 1.结构体声明 type [name] struct {[field_name] [field_type][field_name] [field_type]... }//例子&#xff1a;type Person struct {name stringage int }其中field_name可以省略 2.结构体的使用 1) 格式1 var person Person p…...

关于iPad中的密码和触控ID的使用,看这篇文章就差不多了

序言 许多苹果iPad型号都有熟悉的密码系统和触控ID,这需要指纹扫描才能解锁设备。本指南向你展示如何使用iPad Air 2或更高版本、iPad Mini 3或更新版本以及iPad Pro设置或更改密码和触控ID指纹。 一些iPad Pro型号支持面部识别,并配备了面容ID而不是触控ID作为安全功能。面…...

Vue3之ref与reactive的基本使用

ref可以创建基本类型、对象类型的响应式数据 reactive只可以创建对象类型的响应式数据 接下来让我为大家介绍一下吧&#xff01; 在Vue3中&#xff0c;我们想让数据变成响应式数据&#xff0c;我们需要借助到ref与reactive 先为大家介绍一下ref如何使用还有什么注意点 我们需…...

wsl内置Ubuntu使用 Dinky 与 Flink 集成

Dinky 与 Flink 集成 说明 本文档介绍 Dinky 与 Flink 集成的使用方法, 如果您是 Dinky 的新用户, 请先阅读 本文档, 以便更好的搭建 Dinky 环境 如果您已经熟悉 Dinky 并已经部署了 Dinky, 请跳过本文档的前置要求部分, 直接阅读 Dinky 与 Flink 集成部分 注意: 本文档基…...

”戏说“ 交换机 与 路由器

一般意义上说 老哥 这文章发表 的 东一榔头 西一锤 呵呵&#xff0c; 想到哪里就啰嗦到哪里 。 交换机&#xff1a; 其实就是在通道交换 路由器&#xff1a; 不光是在通道交换还要在协议上交换 下图你看懂了吗&#xff1f; &#xff08;仅仅数据交换-交换机 协议…...

Linux pageset

1. 引言 在用户进程发生缺页异常时&#xff0c;Linux内核需要分配所需物理页面以及建立也表映射&#xff0c;来维持进程的正常内存使用需求。而对于分配物理页面仅依赖于buddy系统&#xff0c;对于小order页面的分配效率较低。因此Linux通过在每个cpu维护一个page链表&#xff…...

【C++之语法篇003】

C学习笔记---003 C知识开篇1、内联函数1.1、什么是内联函数?1.2、解决外部头文件&#xff0c;重复定义问题1.3、内联函数的总结 2、auto关键字2.1、auto的作用2.2、auto的总结 3、范围for3.1、什么是范围for&#xff1f;3.2、范围for的循环应用 4、指针空值关键字nullptr4.1、…...

Github代码仓库SSH配置流程

作者&#xff1a; Herman Ye Auromix 测试环境&#xff1a; Ubuntu20.04 更新日期&#xff1a; 2024/02/21 注1&#xff1a; Auromix 是一个机器人爱好者开源组织。 注2&#xff1a; 由于笔者水平有限&#xff0c;以下内容可能存在事实性错误。 相关背景 在为Github代码仓库配…...

Arrays工具类的常见方法总结

一、Arrays.asList( ) 1.作用&#xff1a;Arrays.asList( )方法的作用是将数组转换成List&#xff0c;将List中的全部集合对象添加至ArrayList集合中 2.参数&#xff1a;动态参数 (T... a) 3.返回值&#xff1a;List 集合 List<T> 4.举例&#xff1a; package com…...

物联网和人工智能的融合

物联网和人工智能的融合 1. 物联网和人工智能的融合2. 芯片技术的进步3. 安全和隐私保护挑战4. 软件开发和调试技术的创新5. 自动化和智能化趋势 1. 物联网和人工智能的融合 随着物联网和人工智能技术的快速发展&#xff0c;嵌入式系统将更多地与物联网设备和人工智能算法相结…...

【微信小程序】wxss 和 css 、wxml 和 html 区别

wxss 和 css 区别 wxss 支持小程序特有的选择器和 样式属性 scroll-into-view cover-view 等wxss 引入了 rpx 单位&#xff0c;可以根据屏幕宽度进行自适应&#xff0c;使得开发者可以更方便的处理不同尺寸屏幕的适配问题。wxss 背景图片只能引入外链&#xff0c;不能使用本地…...

python统计分析——使用AIC进行模型选择

参考资料&#xff1a;用python动手学统计学 1、导入库 # 导入库 # 用于数值计算的库 import numpy as np import pandas as pd import scipy as sp from scipy import stats # 用于绘图的库 import matplotlib.pyplot as plt import seaborn as sns sns.set() # 用于估计统计…...

Android 11以上获取不到第三方app是否安装

开年第一篇&#xff0c;处理了一下年前的小问题。 问题&#xff1a;本地app跳转到第三方app地图进行导航&#xff0c;获取不到第三方地图是否安装。 解决&#xff1a; 1.添加包名 This can be done by adding a <queries> element in the Android manifest.在app下的…...

Java的编程之旅24——private私有方法

1.private的介绍 在面向对象编程中&#xff0c;private是一种访问修饰符&#xff0c;用于限制成员的访问范围。私有成员只能在所属的类内部访问&#xff0c;对外部的类或对象是不可见的。 private的使用可以带来以下几个好处&#xff1a; 封装实现细节&#xff1a;私有成员可…...

为什么在MOS管开关电路设计中使用三极管容易烧坏?

MOS管作为一种常用的开关元件&#xff0c;具有低导通电阻、高开关速度和低功耗等优点&#xff0c;因此在许多电子设备中广泛应用。然而&#xff0c;在一些特殊情况下&#xff0c;我们需要在MOS管控制电路中加入三极管来实现一些特殊功能。然而&#xff0c;不同于MOS管&#xff…...

CSS的注释:以“ /* ”开头,以“ */ ”结尾

CSS的注释:以“ /* ”开头&#xff0c;以“*/”结尾 CSS的注释: 以“ /* ”开头&#xff0c;以“ */ ”结尾 在CSS中&#xff0c;注释是一种非常重要的工具&#xff0c;它们可以帮助开发者记录代码的功能、用法或其他重要信息。这些信息对于理解代码、维护代码以及与他人合作都…...

MySQL中常见的几种日志类型【重点】

在MySQL中&#xff0c;有几种不同类型的日志&#xff0c;用于记录数据库的活动和操作&#xff0c;以便于故障排查、性能调优和数据恢复等目的。以下是MySQL中常见的几种日志类型&#xff1a; 错误日志&#xff08;Error Log&#xff09;&#xff1a; 错误日志记录了MySQL服务器…...

odoo16-API(Controller)带有验证访问的接口

odoo16-API&#xff08;Controller&#xff09;带有验证访问的接口 目前我使用odoo原生的登录token来验证登陆的有效性 废话不多说直接上代码 # 测试获取session_id import requests class GetOdooData(http.Controller):def getOdooToken(self):# http://localhost:8123访问…...

Eclipse项目间的引用

我们在开发的时候&#xff0c;有时候需要把一个大的项目打散&#xff0c;尤其是现在微服务的架构很流行&#xff0c;一个大的项目往往被拆成很多小的项目&#xff0c;而有的项目作为公共工程被独立出来&#xff0c;比如有个工程专门提供各种Util工具类&#xff0c;有的工程专门…...

深度学习在微纳光子学中的应用

深度学习在微纳光子学中的主要应用方向 深度学习与微纳光子学的结合主要集中在以下几个方向&#xff1a; 逆向设计 通过神经网络快速预测微纳结构的光学响应&#xff0c;替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…...

大数据学习栈记——Neo4j的安装与使用

本文介绍图数据库Neofj的安装与使用&#xff0c;操作系统&#xff1a;Ubuntu24.04&#xff0c;Neofj版本&#xff1a;2025.04.0。 Apt安装 Neofj可以进行官网安装&#xff1a;Neo4j Deployment Center - Graph Database & Analytics 我这里安装是添加软件源的方法 最新版…...

Linux云原生安全:零信任架构与机密计算

Linux云原生安全&#xff1a;零信任架构与机密计算 构建坚不可摧的云原生防御体系 引言&#xff1a;云原生安全的范式革命 随着云原生技术的普及&#xff0c;安全边界正在从传统的网络边界向工作负载内部转移。Gartner预测&#xff0c;到2025年&#xff0c;零信任架构将成为超…...

MySQL账号权限管理指南:安全创建账户与精细授权技巧

在MySQL数据库管理中&#xff0c;合理创建用户账号并分配精确权限是保障数据安全的核心环节。直接使用root账号进行所有操作不仅危险且难以审计操作行为。今天我们来全面解析MySQL账号创建与权限分配的专业方法。 一、为何需要创建独立账号&#xff1f; 最小权限原则&#xf…...

保姆级【快数学会Android端“动画“】+ 实现补间动画和逐帧动画!!!

目录 补间动画 1.创建资源文件夹 2.设置文件夹类型 3.创建.xml文件 4.样式设计 5.动画设置 6.动画的实现 内容拓展 7.在原基础上继续添加.xml文件 8.xml代码编写 (1)rotate_anim (2)scale_anim (3)translate_anim 9.MainActivity.java代码汇总 10.效果展示 逐帧…...

前端高频面试题2:浏览器/计算机网络

本专栏相关链接 前端高频面试题1&#xff1a;HTML/CSS 前端高频面试题2&#xff1a;浏览器/计算机网络 前端高频面试题3&#xff1a;JavaScript 1.什么是强缓存、协商缓存&#xff1f; 强缓存&#xff1a; 当浏览器请求资源时&#xff0c;首先检查本地缓存是否命中。如果命…...

Linux安全加固:从攻防视角构建系统免疫

Linux安全加固:从攻防视角构建系统免疫 构建坚不可摧的数字堡垒 引言:攻防对抗的新纪元 在日益复杂的网络威胁环境中,Linux系统安全已从被动防御转向主动免疫。2023年全球网络安全报告显示,高级持续性威胁(APT)攻击同比增长65%,平均入侵停留时间缩短至48小时。本章将从…...

React核心概念:State是什么?如何用useState管理组件自己的数据?

系列回顾&#xff1a; 在上一篇《React入门第一步》中&#xff0c;我们已经成功创建并运行了第一个React项目。我们学会了用Vite初始化项目&#xff0c;并修改了App.jsx组件&#xff0c;让页面显示出我们想要的文字。但是&#xff0c;那个页面是“死”的&#xff0c;它只是静态…...

如何把工业通信协议转换成http websocket

1.现状 工业通信协议多数工作在边缘设备上&#xff0c;比如&#xff1a;PLC、IOT盒子等。上层业务系统需要根据不同的工业协议做对应开发&#xff0c;当设备上用的是modbus从站时&#xff0c;采集设备数据需要开发modbus主站&#xff1b;当设备上用的是西门子PN协议时&#xf…...

qt+vs Generated File下的moc_和ui_文件丢失导致 error LNK2001

qt 5.9.7 vs2013 qt add-in 2.3.2 起因是添加一个新的控件类&#xff0c;直接把源文件拖进VS的项目里&#xff0c;然后VS卡住十秒&#xff0c;然后编译就报一堆 error LNK2001 一看项目的Generated Files下的moc_和ui_文件丢失了一部分&#xff0c;导致编译的时候找不到了。因…...