gpt4 book ai didi

module - 两个模块,都导出相同的名称

转载 作者:行者123 更新时间:2023-12-02 03:53:10 24 4
gpt4 key购买 nike

我想使用两个包:CorpusLoaders.jl ,和WordNet.jl

  • CorpusLoaders.SemCor 导出 sensekey(::SenseTaggedWord)
  • WordNet 导出 sensekey(::DB,::Synset,::Lemma)

我想使用这两种 sensekey 方法。

例如

对于一些混合的项目列表:mixedlist::Vector{Union{Tuple{SenseTaggedWord},Tuple{DB, Synset,Lemma}}。即列表中的项目是 SenseTaggedWord 的 1 元组和 DBSynsetLemma< 的 3 元组的混合。/.

for item in mixedlist
println(sensekey(item...)
end

应该可以工作。这个例子有点滑稽,因为我为什么要这样混合它们。但是,希望它能够说明一般情况下的问题。

尝试使用 CorpusLoaders.SemCor、WordNet 引入两个结果,警告:WordNet 和 Semcor 都导出“sensekey”;在 Main 模块中使用它必须经过限定。

手动导入两者:import CorpusLoaders.SemCor.sensekey; import WordNet.sensekey 导致警告:忽略将 Semcor.sensekey 导入到 Main 中的冲突

可以做什么?我想要它们,并且由于多次调度,它们并不真正冲突。

<小时/>

鉴于 CorpusLoaders.jl 是我正在编写的一个包,我确实有更多选择,因为我可以使我的 CorpusLoaders.jl 依赖于 WordNet.jl。如果我这样做了,那么我可以在 CorpusLoaders.jl 中说

 import WordNet
function WordNet.sensekey(s::SenseTaggedWord)...

这将使它们都工作。但这意味着需要 WordNet 作为 CorpusLoaders 的依赖项。

我想知道如何为包的使用者解决问题——而不是作为包的创建者。

最佳答案

tl;dr 通过模块命名空间在脚本中使用函数时限定函数,即 CorpusLoader.sensekey()WordNet.sensekey()

<小时/>

说明

编辑后我对您的问题的理解(感谢您的澄清)是:

  • 您编写了一个名为 CorpusLoaders.jl 的包,它导出函数 sensekey(::SenseTaggedWord)
  • 有一个名为 WordNet.jl 的外部包,它导出函数 sensekey(::DB,::Synset,::Lemma)
  • 您有一个使用这两个模块的脚本。

并且您担心使用模块或直接“导入”函数可能会在脚本中产生歧义和/或错误,请询问

  1. 如何编写我的 CorpusLoaders 包以防止与其他包发生潜在冲突,以及
  2. 如何编写脚本来清楚地消除两个函数之间的歧义,同时仍然允许使用它们?

我认为这源于对 usingimport 之间的不同以及模块如何创建 namespace 的轻微混淆。文档 here 对此进行了很好的解释。 .

本质上,答案是:

  1. 您不必担心从模块中导出的内容会与其他模块发生冲突。这就是模块的用途:您正在创建一个 namespace ,它将“限定”所有导出的变量,例如CorpusLoaders.sensekey(::SenseTaggedWord)

  2. 当您输入 using CorpusLoaders 时,您对 julia 说的是“导入模块本身,以及从其命名空间限定符中剥离的所有导出变量,并将它们带入 Main”。请注意,这意味着您现在可以直接从 Main 访问 sensekey 作为函数,无需命名空间限定符,作为 CorpusLoaders.sensekey(),因为您还导入了模块作为可以使用的变量。

如果您随后尝试使用模块WordNet,julia 会非常合理地发出警告,其实质内容是:

"You've imported two functions that have the same name. I can't just strip their namespace off because that could create problems in some scenarios (even though in your case it wouldn't because they have different signatures, but I couldn't possibly know this in general). If you want to use either of these functions, please do so using their appropriate namespace qualifier".

因此,2.的解决方案是:

  • 你要么这样做

    using CorpusLoaders;
    using WordNet;

    ,忽略警告,照常在主命名空间中导入所有其他导出的变量,并通过其模块直接访问这些特定函数,例如 CorpusLoaders.sensekey()WordNet.sensekey () 每次您需要在脚本中使用它们时,或者

  • 通过这样做,您可以始终清楚地消除两个模块的歧义

    import CorpusLoaders;
    import WordNet;

    并适本地限定所有变量,或者

  • 在函数签名不冲突的特殊情况下,如果您真的希望能够在没有命名空间限定符的情况下使用该函数,而是依赖于多个分派(dispatch),你可以做像 FengYang 建议的那样:

     import CorpusLoaders;
    import WordNet;
    sensekey(a::SenseTaggedWord) = CorpusLoader.sensekey(a);
    sensekey(a::DB, b::Synset, c::Lemma) = WordNet.sensekey(a, b, c);

    本质上是一个new函数,在模块 Main 上定义,充当两个命名空间限定函数的包装器。

最后,这一切都归结为针对您的特定代码使用 usingimport 和命名空间。 :)

<小时/>作为附录,使用 CorpusLoaderWordNet 等长命名空间限定符,代码可能会变得非常笨拙。 julia 没有类似 python 的 import numpy as np 的功能,但同时模块会成为工作区中的简单变量,因此为它们创建别名很简单。所以你可以这样做:

import CorpusLoaders; const cl = CorpusLoaders;
import Wordnet; const wn = WordNet;
# ... code using both cl.sensekey() and wn.sensekey()

关于module - 两个模块,都导出相同的名称,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39554601/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com