当前位置:首页 > 文章列表 > Golang > Go教程 > Go通过不变性优化程序详解

Go通过不变性优化程序详解

来源:脚本之家 2023-01-07 12:14:22 0浏览 收藏

怎么入门Golang编程?需要学习哪些知识点?这是新手们刚接触编程时常见的问题;下面golang学习网就来给大家整理分享一些知识点,希望能够给初学者一些帮助。本篇文章就来介绍《Go通过不变性优化程序详解》,涉及到不变性、程序优化,有需要的可以收藏一下

正文

不变性的概念非常简单,在您创建结构体后,就永远无法修改它。这个概念听起来非常简单,但您的程序想利用它从中收益并不是那么容易。接下来我们在 Go 中,使用不变性概念,来让您的代码更具有可读性和稳定性。

减少对全局或外部状态的依赖

当我们使用相同的参数,执行相同的函数两次,我们的预期,应该得到相同的结果。但是当我们的函数中依赖外部状态或全局变量时,函数可能会输出不同的结果。我们最好避免这种情况。

函数的参数总是给定的,那我们调用,总是可以返回相同的函数。如果您有一个共享全局变量用于函数内部的某些内容,请考虑将该变量作为参数传递,而不是直接函数内部使用它。

这可以让您的函数返回值更加可预测,并且更加易于测试,整个代码的可读性也会得到提高,因为调用者会知道,哪些值会影响函数的行为,参数的作用不就是会影响返回值的吗?

让我们看一个例子。

package main
import (
   "fmt"
   "math/rand"
   "time"
)
var randNum int
func main() {
   s1 := rand.NewSource(time.Now().UnixNano())
   r1 := rand.New(s1)
   randNum = r1.Intn(100)
   fmt.Println(Add(1, 1))
}
func Add(a, b int) int {
   return a + b + randNum
}

Add 函数中使用了全局变量 randNum 作为计算的一部分,从函数签名中并没有体现这一点。更好的方法是,全局变量 randNum 应该作为参数传递,如下所示。

func Add(a, b, randNum int) int {
   return a + b + randNum
}

这样更具有可预测性,而且我们如果需要修改入参,影响的作用域也仅在 Add 函数中。

仅导出结构体的函数,而不是成员变量

我们知道,Go 结构体中的成员变量,如果首字母为大写,那么该成员变量对外可见(这是编译器决定的)。回到我们的博客,仅导出结构体函数,而不是成员变量,目的是希望成员变量的数据被保护,保证成员变量的有效的状态!因为这可以让您的代码更加可靠,您不必维护每个修改该成员变量的操作,因为这些操作都将无效。

举一个例子

ackage main
import (
	"fmt"
)
type AK47 struct {
	bullet int
}
func NewAK47(bullet int) AK47 {
	return AK47{bullet: bullet}
}
func (a AK47) GetBullet() int {
	return a.bullet
}
func (a AK47) SetBullet(bullet int) {
	a.bullet = bullet
}
func main() {
	ak47 := NewAK47(30)
	fmt.Println(ak47.GetBullet())
	ak47.SetBullet(20)
	fmt.Println(ak47.GetBullet())
}

我们定义了一个结构体 AK47,这把枪有一个成员变量 bullet 子弹数,它是非导出字段,我们还定义了一个构造函数 NewAK47 和一个 GetBullet 函数。

一旦创建了 AK47,就无法更改它的成员变量 bullet 了。此时您可能会有疑惑,如果我们需要修改成员变量呢?别急,您可以试试下面的方法。

在函数中使用复制值,而不是使用指针

在上一个副标题中,我们提到了一个概念,在创建结构体后永远不要更改它。然而在实际中,我们经常需要修改结构体中的成员变量。

我们在使用不变性的同时,仍然可以维护实例化结构体的多个状态,这并不意味着我们打破了结构体创建后不要更改它,我们更改的是它的副本,也就是复制后的结构体。复制后的结构体?难道我们需要去实现很多复制结构体每个字段的函数吗?

当然不,我们可以利用 Go 的特性,在调用函数时,入参是复制值的行为。对于需要修改结构体中成员变量的操作,我们可以创建一个函数,该函数接收结构体为参数,并且返回一个修改后的结构体副本。

我们可以在不改变调用方结构体的情况下,修改该副本的任何内容,这意味着对于原结构体没有任何副作用,并且该结构体的值仍然是可预测的。

不知道您有没有用过 Go 标准库的 Slice 切片,其中的 append 函数就使用了这个方法。让我们接着用 AK47 来实现这个方法

代码如下

package main
import (
	"fmt"
)
type AK47 struct {
	bullet int
}
func NewAK47(bullet int) AK47 {
	return AK47{bullet: bullet}
}
func (a AK47) GetBullet() int {
	return a.bullet
}
func (a AK47) AddBullet(ak47 AK47) AK47 {
	newAK47 := NewAK47(a.GetBullet() + ak47.GetBullet())
	return newAK47
}
func main() {
	ak47 := NewAK47(30)
	add := NewAK47(20)
	fmt.Println(ak47.GetBullet())
	ak47 = ak47.AddBullet(add)
	fmt.Println(ak47.GetBullet())
}

如您所见,我们通过 AddBullet 函数增加枪的子弹,但实际上并没有更改传入的结构体中的任何成员变量。最后,返回了一个带有更新字段的新 AK47 结构体。

与复制值相比,指针更有优势,尤其是当您的结构体成员变量、内容非常大时时,这种方法,通过复制的方式修改数据,可能会导致性能问题。您应该问自己,这么做是否值得,例如您正在编写并发代码?

总结

您在使用不变量时,请务必先权衡利弊。实现本篇博客中所描述的方法,需要大量的代码。但是,如果我们在编写并发代码时,不考虑共享变量的不可变性,往往会出现与预期不符的情况,例如内存竞态问题?其实我想说的就是线程安全问题 : - )

实现不变性,也可能出现严重的性能问题!这是一把双刃剑。请不要过早的优化代码。

今天关于《Go通过不变性优化程序详解》的内容就介绍到这里了,是不是学起来一目了然!想要了解更多关于golang的内容请关注golang学习网公众号!

版本声明
本文转载于:脚本之家 如有侵犯,请联系study_golang@163.com删除
Go语言使用goroutine及通道实现并发详解Go语言使用goroutine及通道实现并发详解
上一篇
Go语言使用goroutine及通道实现并发详解
Go chassis云原生微服务开发框架应用编程实战
下一篇
Go chassis云原生微服务开发框架应用编程实战
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    543次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    514次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    499次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • SEO  AI Mermaid 流程图:自然语言生成,文本驱动可视化创作
    AI Mermaid流程图
    SEO AI Mermaid 流程图工具:基于 Mermaid 语法,AI 辅助,自然语言生成流程图,提升可视化创作效率,适用于开发者、产品经理、教育工作者。
    512次使用
  • 搜获客笔记生成器:小红书医美爆款内容AI创作神器
    搜获客【笔记生成器】
    搜获客笔记生成器,国内首个聚焦小红书医美垂类的AI文案工具。1500万爆款文案库,行业专属算法,助您高效创作合规、引流的医美笔记,提升运营效率,引爆小红书流量!
    505次使用
  • iTerms:一站式法律AI工作台,智能合同审查起草与法律问答专家
    iTerms
    iTerms是一款专业的一站式法律AI工作台,提供AI合同审查、AI合同起草及AI法律问答服务。通过智能问答、深度思考与联网检索,助您高效检索法律法规与司法判例,告别传统模板,实现合同一键起草与在线编辑,大幅提升法律事务处理效率。
    533次使用
  • TokenPony:AI大模型API聚合平台,一站式接入,高效稳定高性价比
    TokenPony
    TokenPony是讯盟科技旗下的AI大模型聚合API平台。通过统一接口接入DeepSeek、Kimi、Qwen等主流模型,支持1024K超长上下文,实现零配置、免部署、极速响应与高性价比的AI应用开发,助力专业用户轻松构建智能服务。
    577次使用
  • 迅捷AIPPT:AI智能PPT生成器,高效制作专业演示文稿
    迅捷AIPPT
    迅捷AIPPT是一款高效AI智能PPT生成软件,一键智能生成精美演示文稿。内置海量专业模板、多样风格,支持自定义大纲,助您轻松制作高质量PPT,大幅节省时间。
    501次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码