为什么GO不支持循环引用
作者:煎鱼 发布时间:2024-02-08 15:10:01
学习 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 程序员更多地考虑程序的依赖关系。
保持依赖关系图的简洁。
快速的程序构建。
如果支持循环引用:很容易会造成懒惰、不良的依赖性管理和缓慢的构建。这是设计者不希望看见的。
混乱的依赖关系。
缓慢的程序构建
因此考虑一开始就保持依赖图的正确 DAG
,Rob Pike
认为这是一个值得预先简化的领域。
在 Go 程序中去做导入循环这件事可能很方便,但背后的代价可能是灾难性的,会对 Go 的构建性能和依赖关系造成非常不利的影响。
所以在 Go 中被明确禁止支持。
3、总结
在程序中,如果我们频繁的出现模块与模块之间的循环引用,这时候我们是不是应该考虑一下,是不是设计的有些问题,要不要考虑调整?
但也并非所有的事都是二极管,Go 源码可能或多或少都有自己循环引用的案例,最重要的是想清楚。
来源:https://www.tuicool.com/articles/eeYF3um


猜你喜欢
- 本文的重点就是如何创建一个PHP类来实现两个数间的加减乘除,这种类似于极简单的计算器功能。实现起来是非常简单的,继续往下看:这里简单给大家介
- 前几天项目在运行的时候数据库数据插入突然报:ORA-01653: unable to extend table HSDC.ADDRESS b
- 最近因为要写一个项目的接口,需要远程的连接oracle数据库,刚开始的时候因为我本地只装了MySQL,所以用就连接了本地MySQL,接口大体
- python语言的一大优势:为科学计算提供了大量的支持功能,math模块提供了很多数学计算函数。math模块定义了一些数学模块,这个模块属于
- 代码如下:SELECT * FROM (SELECT TRUNC(SYSDATE, 'mm') + ROWNUM
- 主要涉及:JOIN 、JOIN 更新、GROUP BY HAVING 数据查重/去重1 INNER JOIN、LEFT JOIN、RIGHT
- 简介Simon Willison来自英国,是一名经验丰富的开发人员。曾工作于Yahoo,是Web开发框架Django的创始人之一,也是Ope
- 使用python的json模块序列化时间或者其他不支持的类型时会抛异常,例如下面的代码:# -*- coding: cp936 -*-fro
- 一、powershell中 python创建虚拟环境无法激活 二、管理员模式运行powershell,执行策略更改: S
- 先使用 Ms Access 压缩修复,然后再去相关的表把备注类型的字段里的“索引”去掉 备注型字段为什么不能超过1950字节?是由于备注型字
- 报错如下:TabError: inconsistent use of tabs and spaces in indentation我推荐一种
- 1.按列取、按索引/行取、按特定行列取import numpy as npfrom pandas import DataFrameimpor
- 首先,这次讲解的tansforms功能,通俗地讲,类似于在计算机视觉流程里的图像预处理部分的数据增强。transforms的原理:说明:图片
- PyCharm 具备一般 IDE 的功能,比如,调试、语法高亮、项目管理、代码跳转、智能提示、自动完成、单元测试、版本控制…另外,PyCha
- Edit:2016-5-11 修正了代码里面一些明显的错误,并发布在 ajaxjs 库之中,源码在这里。Edit:2016-5-24 加入
- git cherry-pick可以选择某一个分支中的一个或几个commit(s)来进行操作。例如,假设我们有个稳定版本的分支,叫v2.0,另
- pycharm一般安装完毕,就是默认是自动保存的,但是......但是....既然是程序,既然是软件,就难免出现bug。也许会 * 友出现头天
- 在金融领域中,我们的y值和预测得到的违约概率刚好是两个分布未知的两个分布。好的信用风控模型一般从准确性、稳定性和可解释性来评估模型。一般来说
- 注意:使用该方法,获取的数据总数目不能超过1万,否则出错#在python3上运行from elasticsearch import Elas
- 这段时间看了关于在SQL server 中通过日志和时间点来恢复数据。也看了一些网上的例子,看如何通过日志来恢复数据。 前提条件:数据库的故