8

如何防止在我的 scala 代码中使用特定的隐式?

例如,我最近被https://github.com/scala/scala/blob/68bad81726d15d03a843dc476d52cbbaf52fb168/src/library/scala/io/Codec.scala#L76Codec提供的默认设置所困扰。有没有办法确保任何需要 an 的代码永远不会使用由 提供的代码?或者,是否可以阻止所有隐式编解码器?implicit codec: CodecfallbackSystemCodec

这是使用scalafix应该可行的吗?

4

2 回答 2

6

Scalafix可以使用SemanticTree. 是通过定义自定义 scalafix 规则的示例解决方案。

给定

import scala.io.Codec

object Hello {
  def foo(implicit codec: Codec) = 3
  foo
}

我们可以定义一个自定义规则

class ExcludedImplicitsRule(config: ExcludedImplicitsRuleConfig)
    extends SemanticRule("ExcludedImplicitsRule") {

...

  override def fix(implicit doc: SemanticDocument): Patch = {
    doc.tree.collect {
      case term: Term if term.synthetic.isDefined => // TODO: Use ApplyTree(func, args)
        val struct = term.synthetic.structure
        val isImplicit = struct.contains("implicit")
        val excludedImplicit = config.blacklist.find(struct.contains)
        if (isImplicit && excludedImplicit.isDefined)
          Patch.lint(ExcludedImplicitsDiagnostic(term, excludedImplicit.getOrElse(config.blacklist.mkString(","))))
        else
          Patch.empty
    }.asPatch
  }

}

和相应的.scalafix.conf

rule = ExcludedImplicitsRule
ExcludedImplicitsRuleConfig.blacklist = [
  fallbackSystemCodec
]

应该能够sbt scalafix提高诊断

[error] /Users/mario/IdeaProjects/scalafix-exclude-implicits/example-project/scalafix-exclude-implicits-example/src/main/scala/example/Hello.scala:7:3: error: [ExcludedImplicitsRule] Attempting to pass excluded implicit fallbackSystemCodec to foo'
[error]   foo
[error]   ^^^
[error] (Compile / scalafix) scalafix.sbt.ScalafixFailed: LinterError

注意输出println(term.synthetic.structure)

Some(ApplyTree(
  OriginalTree(Term.Name("foo")),
  List(
    IdTree(SymbolInformation(scala/io/LowPriorityCodecImplicits#fallbackSystemCodec. => implicit lazy val method fallbackSystemCodec: Codec))
  )
))

显然,上述解决方案在搜索字符串时效率不高,但它应该给出一些方向。也许匹配上ApplyTree(func, args)会更好。

scalafix-exclude-implicits-example展示了如何配置项目以使用ExcludedImplicitsRule.

于 2019-03-30T03:20:51.250 回答
2

您可以通过完全使用新类型来做到这一点;这样,没有人能够在您的依赖项中覆盖它。这本质上是我发布的创建模棱两可的低优先级隐式的答案

但是,如果您无法更改类型,则它可能不实用。

于 2019-04-02T16:01:02.643 回答