热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Go通过不变性优化程序详解【golang基础】

这篇文章主要为大家介绍了Go通过不变性优化程序实例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多

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通过不变性优化程序详解的详细内容,更多关于Go 程序不变性的资料请关注编程笔记其它相关文章!


推荐阅读
  • 本文讨论了clone的fork与pthread_create创建线程的不同之处。进程是一个指令执行流及其执行环境,其执行环境是一个系统资源的集合。在调用系统调用fork创建一个进程时,子进程只是完全复制父进程的资源,这样得到的子进程独立于父进程,具有良好的并发性。但是二者之间的通讯需要通过专门的通讯机制,另外通过fork创建子进程系统开销很大。因此,在某些情况下,使用clone或pthread_create创建线程可能更加高效。 ... [详细]
  • 本文介绍了在多平台下进行条件编译的必要性,以及具体的实现方法。通过示例代码展示了如何使用条件编译来实现不同平台的功能。最后总结了只要接口相同,不同平台下的编译运行结果也会相同。 ... [详细]
  • 先看官方文档TheJavaTutorialshavebeenwrittenforJDK8.Examplesandpracticesdescribedinthispagedontta ... [详细]
  • 如何用JNI技术调用Java接口以及提高Java性能的详解
    本文介绍了如何使用JNI技术调用Java接口,并详细解析了如何通过JNI技术提高Java的性能。同时还讨论了JNI调用Java的private方法、Java开发中使用JNI技术的情况以及使用Java的JNI技术调用C++时的运行效率问题。文章还介绍了JNIEnv类型的使用方法,包括创建Java对象、调用Java对象的方法、获取Java对象的属性等操作。 ... [详细]
  • 本文介绍了操作系统的定义和功能,包括操作系统的本质、用户界面以及系统调用的分类。同时还介绍了进程和线程的区别,包括进程和线程的定义和作用。 ... [详细]
  • Java编程实现邻接矩阵表示稠密图的方法及实现类介绍
    本文介绍了Java编程如何实现邻接矩阵表示稠密图的方法,通过一个名为AMWGraph.java的类来构造邻接矩阵表示的图,并提供了插入结点、插入边、获取邻接结点等功能。通过使用二维数组来表示结点之间的关系,并通过元素的值来表示权值的大小,实现了稠密图的表示和操作。对于对稠密图的表示和操作感兴趣的读者可以参考本文。 ... [详细]
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • 本文介绍了使用Java实现大数乘法的分治算法,包括输入数据的处理、普通大数乘法的结果和Karatsuba大数乘法的结果。通过改变long类型可以适应不同范围的大数乘法计算。 ... [详细]
  • 开发笔记:加密&json&StringIO模块&BytesIO模块
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了加密&json&StringIO模块&BytesIO模块相关的知识,希望对你有一定的参考价值。一、加密加密 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • 使用Ubuntu中的Python获取浏览器历史记录原文: ... [详细]
  • eclipse学习(第三章:ssh中的Hibernate)——11.Hibernate的缓存(2级缓存,get和load)
    本文介绍了eclipse学习中的第三章内容,主要讲解了ssh中的Hibernate的缓存,包括2级缓存和get方法、load方法的区别。文章还涉及了项目实践和相关知识点的讲解。 ... [详细]
  • 如何提高PHP编程技能及推荐高级教程
    本文介绍了如何提高PHP编程技能的方法,推荐了一些高级教程。学习任何一种编程语言都需要长期的坚持和不懈的努力,本文提醒读者要有足够的耐心和时间投入。通过实践操作学习,可以更好地理解和掌握PHP语言的特异性,特别是单引号和双引号的用法。同时,本文也指出了只走马观花看整体而不深入学习的学习方式无法真正掌握这门语言,建议读者要从整体来考虑局部,培养大局观。最后,本文提醒读者完成一个像模像样的网站需要付出更多的努力和实践。 ... [详细]
  • 设计模式——模板方法模式的应用和优缺点
    本文介绍了设计模式中的模板方法模式,包括其定义、应用、优点、缺点和使用场景。模板方法模式是一种基于继承的代码复用技术,通过将复杂流程的实现步骤封装在基本方法中,并在抽象父类中定义模板方法的执行次序,子类可以覆盖某些步骤,实现相同的算法框架的不同功能。该模式在软件开发中具有广泛的应用价值。 ... [详细]
  • java drools5_Java Drools5.1 规则流基础【示例】(中)
    五、规则文件及规则流EduInfoRule.drl:packagemyrules;importsample.Employ;ruleBachelorruleflow-group ... [详细]
author-avatar
素描淡写的快乐_855
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有