为什么GO不支持循环引用

 更新时间:2021年10月27日 14:17:38   作者:煎鱼  
这篇文章主要介绍的是为什么GO不支持循环引用,学习 Go 语言的开发者越来越多了,很多小伙伴在使用时,就会遇到种种不理解的问题,其中一点就是包的循环引用的报错,下main文章我们一起来看看学习原因

学习 Go 语言的开发者越来越多了,很多小伙伴在使用时,就会遇到种种不理解的问题。

其中一点就是包的循环引用的报错:

package command-line-arguments
 imports github.com/eddycjy/awesome-project/a
 imports github.com/eddycjy/awesome-project/b
 imports github.com/eddycjy/awesome-project/a: import cycle not allowed


一下子就很懵逼了,为什么 Go 不支持包之间的循环引用呢,这就很不解了,难道还影响性能了?

如下图:

今天煎鱼将和大家一起了解背后的原因。

1、案例演示

这里我们做一个基本的案例 Demo,便于没接触过的同学建立初步认知。我们的程序分别有 2 个 package

package a 的代码如下:

import (
 "github.com/eddycjy/awesome-project/b"
)

func Hello(s string) {
 b.Print(s)
}

package b 的代码如下:

import (
 "fmt"

 "github.com/eddycjy/awesome-project/a"
)

func Hello() {
 a.Hello("脑子进煎鱼了")
}

func Print(s string) {
 fmt.Println(s)
}

再在 main.go 的文件中调用 a.Hello("脑子进煎鱼了") 方法。

一运行,就会出现如下错误提示:

package command-line-arguments
 imports github.com/eddycjy/awesome-project/a
 imports github.com/eddycjy/awesome-project/b
 imports github.com/eddycjy/awesome-project/a: import cycle not allowed

错误的本质原因是 package a 引用了 package b,而 package b 又引用了 package a,造成了循环引用。

这在 Go 语言中是明令禁止的,在编译时就会中断程序,导致编译失败。

2、原因分析

根据现在 Go 官方的统一意见来看,package 循环导入几乎不可能出现,即使是 Go2,也被明确拒绝了。

因为 Go2 可能是很多核心问题的破变的关键节点,有许多人提了类似《proposal: Go 2: allow import cycle》的提案,希望解决循环引入的问题。

Go 语言之父 Rob Pike 亲自回答了这个问题,原因如下:

没有支持循环引用:目的是迫使 Go 程序员更多地考虑程序的依赖关系。

  • 保持依赖关系图的简洁。
  • 快速的程序构建。

如果支持循环引用:很容易会造成懒惰、不良的依赖性管理和缓慢的构建。这是设计者不希望看见的。

  • 混乱的依赖关系。
  • 缓慢的程序构建

因此考虑一开始就保持依赖图的正确 DAGRob Pike 认为这是一个值得预先简化的领域。

在 Go 程序中去做导入循环这件事可能很方便,但背后的代价可能是灾难性的,会对 Go 的构建性能和依赖关系造成非常不利的影响。

所以在 Go 中被明确禁止支持。

3、总结

在程序中,如果我们频繁的出现模块与模块之间的循环引用,这时候我们是不是应该考虑一下,是不是设计的有些问题,要不要考虑调整?

但也并非所有的事都是二极管,Go 源码可能或多或少都有自己循环引用的案例,最重要的是想清楚。

到此这篇关于为什么GO不支持循环引用的文章就介绍到这了,更多相关GO不支持循环引用内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • go-zero 组件布隆过滤器使用示例详解

    go-zero 组件布隆过滤器使用示例详解

    这篇文章主要为大家介绍了go-zero组件介绍之布隆过滤器使用示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-05-05
  • 基于go实例网络存储协议详解

    基于go实例网络存储协议详解

    这篇文章主要为大家介绍了基于go实例网络存储协议详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-03-03
  • Golang内存泄漏场景以及解决方案详析

    Golang内存泄漏场景以及解决方案详析

    golang中内存泄露的发现与排查一直是来是go开发者头疼的一件事,下面这篇文章主要给大家介绍了关于Golang内存泄漏场景以及解决的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2023-01-01
  • Go语言开发必知的一个内存模型细节

    Go语言开发必知的一个内存模型细节

    这篇文章主要为大家介绍了Go语言开发必知的一个内存模型细节详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-07-07
  • Go 文件读取和写入操作全面讲解

    Go 文件读取和写入操作全面讲解

    这篇文章主要为大家介绍了Go文件的读取和写入操作示例的全面详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-11-11
  • go micro集成链路跟踪的方法和中间件原理解析

    go micro集成链路跟踪的方法和中间件原理解析

    这篇文章主要介绍了go-micro集成链路跟踪的方法和中间件原理,关于Http或者说是Restful服务的链路跟踪,go-micro的httpClient支持CallWrapper,可以用WrapCall来添加链路跟踪的CallWrapper,本文给大家介绍的非常详细,需要的朋友参考下吧
    2022-05-05
  • golang结构化日志log/slog包之LogValuer的用法简介

    golang结构化日志log/slog包之LogValuer的用法简介

    这篇文章主要为大家详细介绍了golang结构化日志log/slog包中 LogValuer 和日志记录函数的正确包装方法,感兴趣的小伙伴可以跟随小编一起了解一下
    2023-10-10
  • 使用go实现常见的数据结构

    使用go实现常见的数据结构

    这篇文章主要介绍了使用go实现常见的数据结构,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-03-03
  • Go语言日志内聚复用及gjson踩坑记录分享

    Go语言日志内聚复用及gjson踩坑记录分享

    这篇文章主要为大家介绍了Go语言日志内聚复用及gjson踩坑记录分享,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-06-06
  • Go语言Handler详细说明

    Go语言Handler详细说明

    这篇文章主要介绍了Go语言Handler详细说明,Handler用于处理请求并给予响应。更严格地说,用来读取请求体、并将请求对应的响应字段(respones header)写入ResponseWriter中,需要的朋友可以参考下
    2022-04-04

最新评论