当前位置:首页 > 文章列表 > Golang > Go问答 > go/packages.Load() 返回多种类型

go/packages.Load() 返回多种类型

来源:stackoverflow 2024-02-18 22:48:27 0浏览 收藏

本篇文章主要是结合我之前面试的各种经历和实战开发中遇到的问题解决经验整理的,希望这篇《go/packages.Load() 返回多种类型》对你有很大帮助!欢迎收藏,分享给更多的需要的朋友学习~

问题内容

我试图确定两种类型是否与 go/types.identical 相同,令人惊讶的是,不同 packages 返回的同一代码段的类型总是不同的。load 调用总是不同的。

我对这些 api 做出了错误的假设吗?

package main

import (
    "fmt"
    "go/types"

    "golang.org/x/tools/go/packages"
)

func getTimeTime() *types.Named {
    pkgs, err := packages.Load(&packages.Config{
        Mode: packages.NeedImports | packages.NeedSyntax | packages.NeedTypes | packages.NeedDeps | packages.NeedTypesInfo,
        Overlay: map[string][]byte{
            "/t1.go": []byte(`package t
            import "time"
            var x time.Time`),
        },
    }, "file=/t1.go")
    if err != nil {
        panic(err)
    }
    for _, v := range pkgs[0].TypesInfo.Types {
        return v.Type.(*types.Named) // named type of time.Time
    }
    panic("unreachable")
}

func main() {
    t1, t2 := getTimeTime(), getTimeTime()
    if !types.Identical(t1, t2) {
        fmt.Println(t1, t2, "are different")
    }
}

正确答案


显然,有一个隐藏文档解释了所有这些(它不附加任何内容,因此不在 godoc 上):https://cs.opensource.google/go/x/tools/+/master:go/packages/doc.go;l=75

动机和设计考虑因素

新包装的设计解决了两个现有的问题 packages:go/build,它定位并描述包,以及 golang.org/x/tools/go/loader,加载、解析和类型检查 他们。 go/build.Package 结构编码了太多的 'go build' 组织项目的方式,让我们需要一种数据类型 描述了独立于 Go 源代码的包 底层构建系统。我们想要同样有效的东西 使用 go build 和 vgo,以及其他构建系统,例如 Bazel 和 Blaze,使得构建适用于所有环境的分析工具成为可能 这些环境。 errcheck 和 staticcheck 等工具 Google 的 Go 社区基本上无法使用,并且某些 Google 的 Go 内部工具无法在外部使用。这个新的 package提供了统一的方式通过查询获取包元数据 这些构建系统中的每一个都可以选择支持它们的首选 包的命令行符号,以便工具可以整齐地集成 与用户的构建环境。元数据查询函数执行 适合当前工作区的外部查询工具。

加载包总是返回完整的导入图“一路 down”,即使您想要的只是有关单个包裹的信息, 因为我们目前所有构建系统的查询机制 支持({go,vgo} 列表和 blaze/bazel 基于方面的查询)不能 提供有关一个包的详细信息,而无需访问所有包 它也有依赖性,因此没有额外的渐近成本 提供传递信息。 (此属性可能不正确 假设的第五构建系统。)

在对 TypeCheck 的调用中,所有初始包以及任何 传递依赖于其中之一,必须从源加载。 考虑 A->B->C->D->E:如果 A,C 是初始的,则 A,B,C 必须从加载 来源; D 可以从导出数据加载,E 可以不加载 全部(尽管 D 的导出数据可能提到了它,所以 types.Package 可以为其创建并公开。)

旧的加载程序具有抑制函数类型检查的功能 基于每个包的主体,主要是为了减少工作量 获取导入包的类型信息。现在进口 导出数据满意,优化似乎不再 必要的。

尽管进行了一些早期尝试,旧的加载程序并未利用导出 数据,而不是始终使用相当于 WholeProgram 模式。这 是由于混合源数据包和导出数据包的复杂性造成的 (现在通过上面提到的向上遍历来解决),并且因为 导出数据文件几乎总是丢失或过时。现在就去吧 build'支持缓存,所有底层构建系统都可以 保证在合理(摊销)时间内生成出口数据。

现在报告由构建系统合成的测试“主”包 作为一流的套餐,避免了客户的需要(例如 go/ssa)来重新发明这个生成逻辑。

go/packages 比旧的加载器更简单的一种方式是它的 包装内测试的处理。包内测试是那些 包含被测库的所有文件以及测试 文件。旧的加载程序通过两阶段构建包内测试 突变的过程称为“增强”:首先它会构建 并类型检查所有普通库包并类型检查 依赖于它们的包;然后它会添加更多(测试)文件到 再次对包装进行类型检查。这种两阶段方法有四个 主要问题:1)在处理测试时,加载器修改了 库包,客户端应用程序无法看到 测试包和库包;一个会变异 进入另一个。 2)因为测试文件可以声明额外的方法 对于包的库部分中定义的类型, 库部分中方法调用的调度受到 测试文件的存在。这应该是一个线索 包在逻辑上是不同的。 3)这种“增强”模型 假设每个库包最多有一个包内测试,即 对于使用“go build”的项目来说是这样,但其他构建系统则不然。 4) 由于测试处理的两阶段性质,所有包 导入库包必须在扩充之前进行处理,
强制使用“一次性”API 并阻止客户端调用 Load
现在在 WholeProgram 模式下可以按顺序进行多次。 (TypeCheck 模式对于不同的情况具有类似的一次性限制 原因。)

该软件包的早期草案支持“多重拍摄”操作。 尽管它允许客户端进行一系列调用(或并发调用) 调用)到 Load,逐步构建包图,它 具有边际价值:它使 API 变得复杂(因为它允许一些 选项会因调用而变化,但不会因其他调用而变化),这使 实现,它不能在类型模式下工作,如所解释的 如上所述,它的效率低于进行一次组合调用(当 这个有可能)。在我们考察过的客户中,没有一个 多次调用加载但不能轻松且令人满意 修改为仅进行一次调用。但是,应用程序可能会发生变化 需要。例如,ssadump 命令加载用户指定的 包以及运行时包。这是很诱人的 只需将“运行时”附加到用户提供的列表中,但这并不 如果用户指定了一个临时包,例如 [a.go b.go],则可以工作。 相反,ssadump 不再请求运行时包,而是寻找它 在用户指定的包的依赖项之间,并发出一个 如果没有找到则出错。

好了,本文到此结束,带大家了解了《go/packages.Load() 返回多种类型》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多Golang知识!

版本声明
本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
如何使用Ajax进行前端和后端通信如何使用Ajax进行前端和后端通信
上一篇
如何使用Ajax进行前端和后端通信
Java在大数据和人工智能领域的应用解析
下一篇
Java在大数据和人工智能领域的应用解析
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    542次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    508次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    497次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • 笔灵AI生成答辩PPT:高效制作学术与职场PPT的利器
    笔灵AI生成答辩PPT
    探索笔灵AI生成答辩PPT的强大功能,快速制作高质量答辩PPT。精准内容提取、多样模板匹配、数据可视化、配套自述稿生成,让您的学术和职场展示更加专业与高效。
    24次使用
  • 知网AIGC检测服务系统:精准识别学术文本中的AI生成内容
    知网AIGC检测服务系统
    知网AIGC检测服务系统,专注于检测学术文本中的疑似AI生成内容。依托知网海量高质量文献资源,结合先进的“知识增强AIGC检测技术”,系统能够从语言模式和语义逻辑两方面精准识别AI生成内容,适用于学术研究、教育和企业领域,确保文本的真实性和原创性。
    41次使用
  • AIGC检测服务:AIbiye助力确保论文原创性
    AIGC检测-Aibiye
    AIbiye官网推出的AIGC检测服务,专注于检测ChatGPT、Gemini、Claude等AIGC工具生成的文本,帮助用户确保论文的原创性和学术规范。支持txt和doc(x)格式,检测范围为论文正文,提供高准确性和便捷的用户体验。
    38次使用
  • 易笔AI论文平台:快速生成高质量学术论文的利器
    易笔AI论文
    易笔AI论文平台提供自动写作、格式校对、查重检测等功能,支持多种学术领域的论文生成。价格优惠,界面友好,操作简便,适用于学术研究者、学生及论文辅导机构。
    50次使用
  • 笔启AI论文写作平台:多类型论文生成与多语言支持
    笔启AI论文写作平台
    笔启AI论文写作平台提供多类型论文生成服务,支持多语言写作,满足学术研究者、学生和职场人士的需求。平台采用AI 4.0版本,确保论文质量和原创性,并提供查重保障和隐私保护。
    41次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码