Scala Отражение для создания объекта-компаньона и вызова метода apply - PullRequest
0 голосов
/ 22 апреля 2020

Вот проблема.

trait TestTrait[T, R] extends (T => R) 

// Class implementing TestTrait. This is one class, there are a number of class implementing TestTrait
class TestClass(val input: Map[String, String])extends ConfigurableDomainExtractor[String, String] {
  override def apply(value: String): String = ???
}

// Companion Object
object TestClass {
 def apply(properties: Map[String, String]): TestClass = TestClass(properties)
}

я хочу определить метод в классе утилит, скажем,

class CreateInstance {
  def getInstance(fullyQualifiedClassName: String, properties: Map[String, String]): TestTrait = ???

// fullyQualifiedClassName is the name of the class that needs to be instantiated using its companion object. It can be TestClass or any class implementing TestTrait
// properties is the map that needs to be pass to the apply method of the companion object.


}

, который будет генерироваться при передаче имени класса объект того же класса, сначала получая объект-компаньон, а затем вызывая apply для объекта-компаньона, передающего карту, чтобы сгенерировать экземпляр класса.

Я знаю, что у нас есть отражения в scala, и я пробовал несколько вещей, но но безрезультатно. Вот несколько вещей, которые я попробовал.

  import scala.reflect.runtime.universe

def getInstance(fullyQualifiedClassName: String, properties: Map[String, String]): TestTrait = {
val runtimeMirror = universe.runtimeMirror(getClass.getClassLoader)
    val module        = runtimeMirror.staticModule(fullyQualifiedClassName)
    val obj           = runtimeMirror.reflectModule(module)

      obj.instance
        .asInstanceOf[TestTrait[String, String]]
        .apply(properties)
        .asInstanceOf[TestTrait[String, String]]

}

Может кто-нибудь помочь мне завершить def getInstance метод.

1 Ответ

1 голос
/ 22 апреля 2020

Попробуйте Scala отражение

import scala.reflect.runtime.universe
import scala.reflect.runtime.universe._

def getInstance(fullyQualifiedClassName: String, properties: Map[String, String]): TestTrait[String, String] = {
  val runtimeMirror = universe.runtimeMirror(getClass.getClassLoader)
  val objSymbol = runtimeMirror.staticModule(fullyQualifiedClassName)
  val objMirror = runtimeMirror.reflectModule(objSymbol)
  val obj = objMirror.instance
  val objTyp = objSymbol.typeSignature
  val methodSymbol = objTyp.decl(TermName("apply")).asMethod
  val instanceMirror = runtimeMirror.reflect(obj)
  val methodMirror = instanceMirror.reflectMethod(methodSymbol)
  methodMirror(properties).asInstanceOf[TestTrait[String, String]]
}

getInstance("pckg.App.TestClass", Map("a" -> "b"))

или Java отражение

def getInstance(fullyQualifiedClassName: String, properties: Map[String, String]): TestTrait[String, String] = {
  val clazz = Class.forName(fullyQualifiedClassName)
  val moduleField = clazz.getDeclaredField("MODULE$")
  val obj = moduleField.get(null) // field MODULE$ is static
  val applyMethod = clazz.getDeclaredMethod("apply", classOf[Map[_, _]])
  applyMethod.invoke(obj, properties).asInstanceOf[TestTrait[String, String]]
}

getInstance("pckg.App$TestClass$", Map("a" -> "b"))
...