如何防止在我的scala代码中使用特定的隐含?
例如,我最近被https://github.com/scala/scala/blob/68bad81726d15d03a843dc476d52cbbaf52fb168/src/library/scala/io/Codec.scala#L76提供的默认Codec
所占据。
有没有办法确保任何需要implicit codec: Codec
的代码永远不会使用fallbackSystemCodec
提供的代码?
或者,是否可以阻止所有隐式编解码器?
这是使用scalafix应该可行的吗?
答案 0 :(得分:5)
Scalafix可以使用SemanticTree
inspect implicit arguments。 Here是通过定义自定义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
。
答案 1 :(得分:2)
您可以通过完全使用新类型来完成此操作;这样,没有人能够在您的依赖项中覆盖它。从本质上讲,这就是我发布到create an ambiguous low priority implicit
的答案但是,例如如果您不能更改类型,则可能不实际。