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


100

我正在尝试将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。我不能instanceof在Scala中使用这种方式吗?

Answers:


114

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

您可能还喜欢观看ScalaTest速成课程


6
好吧,这并不能真正回答问题。ScalaTest具有对类型检查的内置支持。请参阅@ martin-g的答案
maasg 2016年

如果“类型”是特征,该怎么办?
路易(Lobo),

不知道我是否理解正确,但是应该相同:isInstanceOf[TraitName]
agilesteel

88

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

anInstance mustBe a[SomeClass]

4
这是最新版本的ScalaTests的推荐方法
maasg

6
也可以使用,a[Type]因此您在语法上是正确的;)
塞缪尔

我一直在寻找!:)
阿泰斯(Atais)'17

22
tiger shouldBe a [Tiger]是当前语法scalatest.org/at_a_glance/FlatSpec
jhegedus

2
mustBe如果您要为FreeSpec 使用doc.scalatest.org/3.0.1/#org.scalatest.MustMatchers,则@jhegedus 也正确。
Tobi

30

如果您想减少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看起来非常好,甚至对于非编程人员来说也是可以理解的(假设他们知道实例是什么:-)。
helpermethod 2011年

如果您使用语法糖,则可以通过一些重构将house.door应该写为(madeOf [Wood])或house.door应该写为(madeOf [Bamboo])。
Guillaume Belrose


16

当前关于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
Rich Dougherty

3

将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
  )
}

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.