如何使用 Scala 进行 instanceof 检查(测试)

Posted

技术标签:

【中文标题】如何使用 Scala 进行 instanceof 检查(测试)【英文标题】:How to do an instanceof check with Scala(Test) 【发布时间】:2012-01-23 14:48:08 【问题描述】:

我正在尝试将 ScalaTest 合并到我的 Java 项目中;用 ScalaTests 替换所有 JUnit 测试。在某一时刻,我想检查 Guice 的 Injector 是否注入了正确的类型。在 Java 中,我有一个这样的测试:

public class InjectorBehaviour 
    @Test
    public void shouldInjectCorrectTypes() 
        Injector injector = Guice.createInjector(new ModuleImpl());
        House house = injector.getInstance(House.class);

        assertTrue(house.door() instanceof WoodenDoor);
        assertTrue(house.window() instanceof BambooWindow);
        assertTrue(house.roof() instanceof SlateRoof);
    

但我在使用 ScalaTest 时遇到了问题:

class InjectorSpec extends Spec 
    describe("An injector") 
        it("should inject the correct types") 
            val injector = Guice.createInjector(new ModuleImpl)
            val house = injector.getInstance(classOf[House])

            assert(house.door instanceof WoodenDoor)
            assert(house.window instanceof BambooWindow)
            assert(house.roof instanceof SlateRoof)
        
    

它抱怨值instanceof 不是Door/Window/Roof 的成员。我不能在 Scala 中那样使用instanceof 吗?

【问题讨论】:

【参考方案1】:

Scala 不是 Java。 Scala 只是没有运算符instanceof,而是有一个称为isInstanceOf[Type] 的参数方法。

您可能还喜欢看ScalaTest Crash Course。

【讨论】:

好吧,这并不能真正回答问题。 ScalaTest 内置了对类型检查的支持。查看@martin-g 的答案 如果“类型”是一个特征怎么办? 不确定我是否理解正确,但应该是一样的:isInstanceOf[TraitName]【参考方案2】:

使用 Scalatest 2.2.x(可能更早),您可以使用:

anInstance mustBe a[SomeClass]

【讨论】:

这是 ScalaTests 最新版本的推荐方法 也可以使用a[Type],这样你就可以在语法上正确;) 我在找那个! :) tiger shouldBe a [Tiger] 是当前语法scalatest.org/at_a_glance/FlatSpec @jhegedus mustBe 也是正确的,如果您使用 FreeSpec 所需的 doc.scalatest.org/3.0.1/#org.scalatest.MustMatchers。【参考方案3】:

如果你想少一点 JUnit 风格,如果你想使用 ScalaTest 的匹配器,你可以编写自己的匹配类型的属性匹配器(条形擦除)。

我发现这个帖子很有用:http://groups.google.com/group/scalatest-users/browse_thread/thread/52b75133a5c70786/1440504527566dea?#1440504527566dea

然后您可以编写如下断言:

house.door should be (anInstanceOf[WoodenDoor])

而不是

assert(house.door instanceof WoodenDoor)

【讨论】:

+1 这看起来很不错,对于非编程人员来说甚至可以理解(假设他们知道实例是什么:-))。 如果语法糖是您所追求的,通过一些重构,您可能可以编写 house.door should be (madeOf[Wood]) 或 house.door should be (madeOf[Bamboo])。 另见groups.google.com/d/topic/scalatest-users/HeOKgs5PC2o/…【参考方案4】:

关于 isInstanceOf[Type] 和 junit 建议的当前答案很好,但我想补充一件事(对于以非 junit 相关身份访问此页面的人)。在许多情况下,scala 模式匹配将满足您的需求。在这些情况下,我会推荐它,因为它可以免费为您提供类型转换,并减少出错的空间。

例子:

OuterType foo = blah
foo match 
  case subFoo : SubType => 
    subFoo.thingSubTypeDoes // no need to cast, use match variable
  
  case subFoo => 
    // fallthrough code
  

【讨论】:

在 ScalaTest 中测试模式匹配的推荐方法是使用 inside(foo) 而不是 `foo match)。见scalatest.org/user_guide/using_matchers#matchingAPattern【参考方案5】:

将 Guillaume 的 ScalaTest 讨论参考(以及 James Moore 链接的另一个讨论)合并为两种方法,针对 ScalaTest 2.x 和 Scala 2.10 进行了更新(使用 ClassTag 而不是清单):

import org.scalatest.matchers._
import scala.reflect._

def ofType[T:ClassTag] = BeMatcher  obj: Any =>
  val cls = classTag[T].runtimeClass
  MatchResult(
    obj.getClass == cls,
    obj.toString + " was not an instance of " + cls.toString,
    obj.toString + " was an instance of " + cls.toString
  )


def anInstanceOf[T:ClassTag] = BeMatcher  obj: Any =>
  val cls = classTag[T].runtimeClass
  MatchResult(
    cls.isAssignableFrom(obj.getClass),
    obj.getClass.toString + " was not assignable from " + cls.toString,
    obj.getClass.toString + " was assignable from " + cls.toString
  )

【讨论】:

【参考方案6】:

我使用 2.11.8 对集合进行断言。较新的语法如下:

val scores: Map[String, Int] = Map("Alice" -> 10, "Bob" -> 3, "Cindy" -> 8)
scores shouldBe a[Map[_, _]] 

【讨论】:

由于擦除,您无法检查Map 的类型参数。你写的和写scores shouldBe a[Map[_, _]]一样。这里提到了这一点:scalatest.org/user_guide/using_matchers#checkingAnObjectsClass

以上是关于如何使用 Scala 进行 instanceof 检查(测试)的主要内容,如果未能解决你的问题,请参考以下文章

如何使用 Scala 在 Spark 中进行滑动窗口排名?

如何在 switch 语句中使用 instanceof

instanceof

如何对Scala中集合进行排序?

Scala,如何使用函数式编程在复杂序列之间进行一种连接操作?

如何使用 Spark 数据框列上的函数或方法使用 Scala 进行转换