Go 结构体(Struct)对象:方法、指针与值的选择

go 结构体(struct)对象:方法、指针与值的选择

本文旨在帮助初学者理解 Go 语言中结构体的使用,重点讲解方法中指针接收者与值接收者的区别,以及如何正确地修改结构体内部状态。通过一个汽车引擎启动的示例,深入剖析了使用值接收者导致状态修改失效的原因,并提供了使用指针接收者的正确解决方案,同时涉及结构体的初始化和最佳实践。

理解 Go 中的结构体和方法

Go 语言虽然不是严格意义上的面向对象编程(OOP)语言,但它支持使用结构体(Struct)来组织数据,并可以使用方法(Method)来操作这些数据,从而实现类似面向对象编程的效果。理解结构体和方法是掌握 Go 语言的关键一步。

结构体的定义和使用

结构体是一种用户自定义的类型,它可以包含多个不同类型的字段。例如,我们可以定义一个 Engine 结构体来表示汽车引擎:

package mainimport "fmt"type Engine struct {    cylinders int    started   bool}func main() {    var engine Engine    engine.cylinders = 4 // 设置气缸数为 4    fmt.Println("气缸数:", engine.cylinders)}

方法的定义和使用

方法是一种特殊的函数,它与特定的类型关联。在 Go 语言中,我们可以为结构体定义方法,从而实现对结构体的操作。方法的定义方式是在 func 关键字和方法名之间添加一个接收者(Receiver)。接收者可以是值类型或指针类型,这两种类型在使用上有着重要的区别。

指针接收者 vs 值接收者:一个关键的区别

在 Go 语言中,方法的接收者可以是值类型或指针类型。选择哪种类型取决于方法是否需要修改接收者的状态。

值接收者: 当使用值接收者时,方法操作的是接收者的一个副本。因此,对副本的修改不会影响原始的结构体。指针接收者: 当使用指针接收者时,方法操作的是接收者的原始值。因此,对接收者的修改会直接影响原始的结构体。

示例:汽车引擎启动问题

以下面的 Engine 结构体和 Start 方法为例,展示了使用值接收者导致的问题:

package mainimport (    "fmt")type Engine struct {    cylinders int    started   bool}// 值接收者func (engine Engine) Start() {    fmt.Println("Starting engine...")    engine.started = true // 修改的是副本    fmt.Println("Engine started:", engine.started)}func (engine Engine) IsStarted() bool {    return engine.started}func main() {    var engine Engine    fmt.Println("Engine started?", engine.IsStarted()) // 输出 false    engine.Start()                                      // 启动引擎    fmt.Println("Engine started?", engine.IsStarted()) // 仍然输出 false}

在这个例子中,Start 方法使用值接收者,因此 engine.started = true 修改的是 engine 的一个副本,而不是原始的 engine 结构体。因此,在 main 函数中,engine.IsStarted() 始终返回 false。

解决方案:使用指针接收者

要解决这个问题,我们需要使用指针接收者,如下所示:

package mainimport (    "fmt")type Engine struct {    cylinders int    started   bool}// 指针接收者func (engine *Engine) Start() {    fmt.Println("Starting engine...")    engine.started = true // 修改的是原始值    fmt.Println("Engine started:", engine.started)}func (engine *Engine) IsStarted() bool {    return engine.started}func main() {    var engine Engine    fmt.Println("Engine started?", engine.IsStarted()) // 输出 false    engine.Start()                                      // 启动引擎    fmt.Println("Engine started?", engine.IsStarted()) // 输出 true}

通过将 Start 方法的接收者改为 *Engine,我们就可以直接修改原始的 engine 结构体,从而使 engine.IsStarted() 返回 true。

总结

当方法需要修改结构体的内部状态时,必须使用指针接收者。否则,修改将只作用于接收者的副本,而不会影响原始的结构体。

结构体的初始化

Go 语言提供了多种初始化结构体的方式。

显式初始化

可以显式地指定结构体中每个字段的值:

engine := Engine{    cylinders: 4,    started:   false,}

简写初始化

如果按照结构体字段的定义顺序提供值,可以省略字段名:

engine := Engine{4, false}

使用 new 关键字

可以使用 new 关键字创建一个指向结构体的指针:

engine := new(Engine) // 返回 *Engineengine.cylinders = 4

这种方式会分配内存,并返回指向新分配的内存的指针。

最佳实践

选择正确的接收者类型: 如果方法需要修改结构体的状态,使用指针接收者;如果方法只需要读取结构体的状态,使用值接收者。考虑性能: 对于大型结构体,使用指针接收者可以避免复制结构体的开销,从而提高性能。保持一致性: 如果结构体的大部分方法都使用指针接收者,最好保持一致性,所有方法都使用指针接收者。

通过理解结构体、方法以及指针接收者和值接收者的区别,可以更好地利用 Go 语言的特性,编写出更高效、更可靠的代码。

以上就是Go 结构体(Struct)对象:方法、指针与值的选择的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1414118.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月16日 07:50:59
下一篇 2025年12月16日 07:51:13

相关推荐

  • i++和++i的区别及举例说明

    i++和++i的区别及举例说明 i++和++i命令的区别有: 1、赋值顺序不同 ++ i 是先加后赋值;i ++ 是先赋值后加;++i和i++都是分两步完成的。 因为++i 是后面一步才赋值的,所以它能够当作一个变量进行级联赋值,++i = a =b,即 ++i 是一个左值;i++ 的后面一步是自增…

    2025年12月17日
    000
  • scanf和getchar的区别

    scanf和getchar的区别 一、函数格式不同 scanf函数是格式输入函数,即按用户指定的格式从键盘上把数据输入到指定的变量中。 getchar函数是键盘输入函数,其功能是从键盘上输入一个字符。 二、读取方式不同 scanf函数在读取数字时会跳过空格、制表符和换行符。 getchar函数只能输…

    2025年12月17日
    000
  • #ifndef和#define的区别

    #ifndef和#define的区别 一、使用场景不同: #ifndef使用场景为: 1、头文件中使用,防止头文件被多重调用。 2、作为测试使用,省去注释代码的麻烦。 3、作为不同角色或者场景的判断使用。 #define使用场景: 宏定义 二、含义不同: #ifndef表示ifnotdefine。 …

    2025年12月17日
    000
  • printf和scanf的区别

    printf和scanf的区别 ● 这是两个功能完全不同的函数,printf向标准输出设备(一般是显示器)输出数据,scanf从标准输入设备(一般是键盘)输入数据。 ● printf是输出函数,scanf是输入函数。 拓展内容: printf()函数: 是格式化输出函数, 一般用于向标准输出设备按规…

    2025年12月17日
    000
  • 比较TCP与UDP之间的区别

    tcp(传输控制协议): 1)提供ip环境下的数据可靠传输(一台计算机发出的字节流会无差错的发往网络上的其他计算机,而且计算机a接收数据包的时候,也会向计算机b回发数据包,这也会产生部分通信量),有效流控,全双工操作(数据在两个方向上能同时传递),多路复用服务,是面向连接,端到端的传输; 2)面向连…

    好文分享 2025年12月17日
    000
  • 比较C#中值类型和引用类型的区别

    clr支持两种类型:值类型和引用类型,看起来fcl的大多数类型是引用类型,但用的最多的还是值类型。引用类型总是从托管堆中分配,在用new操作符实例一个对象,返回对象内存地址存放在一个变量中。在使用引用类型时要了解其四个心理因素:        1.内存必须从托管堆中分配        2.堆上分配的…

    好文分享 2025年12月17日
    000
  • 什么是XML Infoset

    XML Infoset是W3C定义的抽象数据模型,用于标准化XML文档解析后的信息表示。它定义了11种信息项(如文档、元素、属性等),屏蔽物理格式差异,确保不同解析器对XML内容的理解一致。DOM和SAX等解析技术均基于Infoset构建:DOM将其具象化为树结构,SAX则通过事件流式暴露信息项。I…

    2025年12月17日
    000
  • RSS订阅中的作者信息格式

    RSS和Atom中作者信息通过或标签标识,包含姓名、邮箱及网站链接,支持多作者;正确设置有助于提升内容可信度、便于追踪与SEO。 RSS订阅中的作者信息格式,主要用于标识文章的作者,让读者知道是谁写的,方便追踪特定作者的内容。格式通常包含作者姓名、邮箱,有时还会包含作者的网站链接。 作者信息的常见格…

    2025年12月17日
    000
  • XML中如何获取根节点属性_XML获取根节点属性的操作步骤

    XML根节点有且仅有一个,可包含属性;2. Python用ET.parse解析,root.get(“属性名”)获取属性值;3. JavaScript用DOMParser解析,xmlDoc.documentElement获取根节点,getAttribute读取属性;4. Jav…

    2025年12月17日
    000
  • XML中如何解压XML字符串_XML解压XML字符串的操作方法

    先解压再解析XML。C#用GZipStream解压字节流并转字符串,Java用GZIPInputStream或InflaterInputStream读取压缩数据,结合StreamReader或BufferedReader还原为明文XML后,交由XDocument或DocumentBuilder解析;…

    2025年12月17日
    000
  • XML中如何判断节点是否存在_XML判断节点存在性的技巧与方法

    使用XPath或find方法判断XML节点是否存在,若返回结果为空则节点不存在,结合attrib检查属性,并区分节点存在与文本内容是否为空。 在处理XML文档时,判断某个节点是否存在是一个常见需求。无论是解析配置文件、处理接口返回数据,还是进行数据校验,准确判断节点是否存在可以避免程序出错。以下是几…

    2025年12月17日
    000
  • XML中如何检查节点顺序_XML检查节点顺序的方法与技巧

    使用XPath、DOM解析、XSD约束和断言工具可检查XML节点顺序。首先通过XPath的position()函数验证节点位置,如//data/item[@type=’A’ and position()=1];其次用Python等语言解析DOM并比对实际与预期顺序;再者利用X…

    2025年12月17日
    000
  • XML与EXI压缩格式比较

    XML与EXI的核心区别在于:XML以人类可读性和互操作性为优先,适合开发调试和配置,但文件体积大、解析效率低;EXI作为W3C定义的二进制格式,牺牲可读性,通过二进制编码、字符串表、模式感知等技术实现高压缩比和高速解析,适用于带宽或资源受限场景。2. 两者并非替代关系,而是互补:XML用于数据定义…

    2025年12月17日
    000
  • RSS源如何实现内容推荐

    要实现RSS%ignore_a_1%,需在RSS数据基础上构建智能推荐系统。首先通过feedparser等工具抓取并解析RSS内容,提取标题、摘要、发布时间等信息,并存储到数据库中;对于仅提供片段的源,可结合Web Scraping技术获取全文。随后利用NLP技术对内容进行处理,包括分词、去停用词、…

    2025年12月17日
    000
  • 如何用XML表示时间序列数据

    XML通过层级结构和属性封装时间戳与数值,适合表示含丰富元数据和不规则采样的时间序列数据,便于跨系统交换;其优势在于自描述性、可扩展性和平台无关性,但存在冗余大、解析慢等问题,海量数据时不如二进制格式或专用数据库高效。 在XML中表示时间序列数据,核心在于利用其层级结构和属性来封装每个时间点的数据值…

    2025年12月17日
    000
  • RSS阅读器如何开发?核心功能有哪些?

    答案:开发RSS阅读器需实现订阅管理、内容抓取解析、展示与同步功能,采用Node.js或Python等技术栈,支持OPML导入、定时更新、离线缓存,并防范XXE攻击,提升用户体验。 RSS阅读器的开发核心在于抓取、解析和展示网站的RSS订阅源内容。这类工具帮助用户集中浏览多个网站的更新,无需逐个访问…

    2025年12月17日
    000
  • 如何验证XML文件的语法正确性?

    验证XML语法正确性需先检查其格式良好性,再验证有效性;格式良好性确保基本语法规则如标签闭合、根元素唯一等,由解析器在解析时自动检测;有效性则通过XSD或DTD确认文档符合预定义结构,包括元素顺序、数据类型等;常用工具包括lxml(Python)、JAXP(Java)、xmllint命令行工具及ID…

    2025年12月17日
    000
  • RSS中的skipHours元素作用

    skipHours是RSS中用于优化更新频率的元素,发布者可通过它指定某些小时段让订阅客户端暂停检查更新,以减少无效请求、降低服务器负载。 RSS中的skipHours元素,说白了,就是发布者在告诉订阅者(或者说,订阅客户端):在某些特定的小时段里,你暂时不用来检查我的更新了。它提供了一种精细化的机…

    2025年12月17日
    000
  • 什么是OpenTravel标准

    OpenTravel标准是旅游行业通用的XML消息格式,由OpenTravel Alliance维护,通过定义如OTA_AirAvailRQ/RS等消息类型,实现航空公司、酒店、旅行社等系统间的数据互通;它简化集成、降低成本,并支持自动化预订与查询;尽管JSON在轻量性和解析速度上占优,但OpenT…

    2025年12月17日
    000
  • XML中如何修改节点值_XML修改节点值的实用方法与注意事项

    使用DOM、XPath或流式处理可修改XML节点值,推荐小文件用DOM+XPath、大文件用流式处理,注意编码、空节点、格式保留及备份验证。 在处理XML数据时,修改节点值是一个常见需求。无论是配置文件更新、数据转换,还是接口报文调整,掌握正确的方法至关重要。下面介绍几种实用的XML节点值修改方式,…

    2025年12月17日
    000

发表回复

登录后才能评论
关注微信