Как избавиться: тип класса требуется, но T найден

Как решить эту ошибку компиляции:

trait Container {
  def getInts() : Seq[Int]
  def getStrings() : Seq[String]

  def put[T](t: T)
  def get[T] : Seq[T]
}

class MutableContainer extends Container {
  val entities = new mutable.HashMap[Class[_], mutable.Set[Any]]() with mutable.MultiMap[Class[_], Any]

  override def getStrings(): Seq[String] = entities.get(classOf[String]).map(_.toSeq).getOrElse(Seq.empty).asInstanceOf[Seq[String]] //strings
  override def getInts(): Seq[Int] = entities.get(classOf[Int]).map(_.toSeq).getOrElse(Seq.empty).asInstanceOf[Seq[Int]]

  override def get[T]: Seq[T] = entities.get(classOf[T]).map(_.toSeq).getOrElse(Seq.empty).asInstanceOf[Seq[T]]
  override def put[T](t: T): Unit = entities.addBinding(t.getClass, t)
}

Вот ошибка:

[error] Container.scala:23: class type required but T found
[error]       override def get[T]: Seq[T] = entities.get(classOf[T]).map(_.toSeq).getOrElse(Seq.empty).asInstanceOf[Seq[T]]

Ответы

Ответ 1

T не тип класса, а параметр типа. Запросить ClassTag:

import scala.reflect._

override def get[T](implicit ct: ClassTag[T]): Seq[T] =
  entities.get(ct.runtimeClass)
          .map(_.toSeq)
          .getOrElse(Seq.empty)
          .asInstanceOf[Seq[T]]

Но это приносит еще одну проблему; это не переопределение!

Итак, вы должны изменить базовый класс, чтобы объявить get следующим образом:

def get[T: ClassTag]: Seq[T]

Ответ 2

Решение без отражения

trait Container {
  def getInts() : Seq[Int]
  def getStrings() : Seq[String]
  def put[T](clazz: Class[T],t :T)
  def get[T](clazz: Class[T]) : Seq[T]
}

class MutableContainer extends Container {
  val entities = new mutable.HashMap[Class[_], mutable.Set[Any]]() with mutable.MultiMap[Class[_],Any]
  override def getStrings(): Seq[String] = entities.get(classOf[String]).map(_.toSeq).getOrElse(Seq.empty).asInstanceOf[Seq[String]] //strings
  override def getInts(): Seq[Int] = entities.get(classOf[Int]).map(_.toSeq).getOrElse(Seq.empty).asInstanceOf[Seq[Int]]
  override def get[T](clazz: Class[T]): Seq[T] = entities.get(clazz).map(_.toSeq).getOrElse(Seq.empty).asInstanceOf[Seq[T]]
  override def put[T](clazz: Class[T],t :T): Unit = entities.addBinding(clazz, t)
}