Cómo hacer una comprobación de instancia con Scala (Test)


Estoy tratando de incorporar ScalaTest en mi proyecto Java; reemplazando todas las pruebas JUnit con ScalaTests. En un momento, quiero comprobar si el inyector de Guice inyecta el tipo correcto. En Java, tengo una prueba como esta:

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

Pero tengo un problema haciendo lo mismo con 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)
        }
    }
}

Se queja de que el valor instanceof no es un miembro de Door/Window/Roof. ¿No puedo usar instanceof de esa manera en Scala?

Author: Toastrackenigma, 2011-12-19

6 answers

Scala no es Java. Scala simplemente no tiene el operador instanceof en su lugar tiene un método paramétrico llamado isInstanceOf[Type].

También puede disfrutar viendo un Curso acelerado ScalaTest.

 104
Author: agilesteel,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/ajaxhispano.com/template/agent.layouts/content.php on line 61
2018-08-26 10:25:06

Con Scalatest 2.2.x (tal vez incluso antes) se puede utilizar:

anInstance mustBe a[SomeClass]
 52
Author: martin-g,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/ajaxhispano.com/template/agent.layouts/content.php on line 61
2018-07-25 20:51:45

Si desea ser menos JUnit-esque y si desea utilizar los matchers de ScalaTest, puede escribir su propio matcher de propiedades que coincida con type (bar type erasure).

Encontré este hilo bastante útil: http://groups.google.com/group/scalatest-users/browse_thread/thread/52b75133a5c70786/1440504527566dea?#1440504527566dea

Luego puedes escribir aserciones como:

house.door should be (anInstanceOf[WoodenDoor])

En lugar de

assert(house.door instanceof WoodenDoor)
 29
Author: Guillaume Belrose,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/ajaxhispano.com/template/agent.layouts/content.php on line 61
2012-12-03 20:24:11

Las respuestas actuales sobre isInstanceOf[Type] y los consejos de junit son buenas, pero quiero agregar una cosa (para las personas que llegaron a esta página en una capacidad no relacionada con junit). En muchos casos, la coincidencia de patrones scala se adaptará a sus necesidades. Lo recomendaría en esos casos porque le da el encasillamiento de forma gratuita y deja menos espacio para el error.

Ejemplo:

OuterType foo = blah
foo match {
  case subFoo : SubType => {
    subFoo.thingSubTypeDoes // no need to cast, use match variable
  }
  case subFoo => {
    // fallthrough code
  }
}
 14
Author: alexbobp,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/ajaxhispano.com/template/agent.layouts/content.php on line 61
2014-05-01 22:24:24

Consolidando la referencia de discusión ScalaTest de Guillaume (y otra discusión vinculada a James Moore) en dos métodos, actualizados para ScalaTest 2.x y Scala 2.10 (para usar ClassTag en lugar de manifest):

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
  )
}
 2
Author: Raman,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/ajaxhispano.com/template/agent.layouts/content.php on line 61
2014-05-15 16:44:19

Utilizo 2.11.8 para hacer la aserción con colecciones. La sintaxis más reciente es la siguiente:

val scores: Map[String, Int] = Map("Alice" -> 10, "Bob" -> 3, "Cindy" -> 8)
scores shouldBe a[Map[_, _]] 
 0
Author: aristotll,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/ajaxhispano.com/template/agent.layouts/content.php on line 61
2017-08-24 23:07:44