Есть ли какая-либо функция Scala, которая позволяет вам вызывать метод, имя которого хранится в строке?

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

Вроде как Руби send параметр.

4 ответа

Решение

Вы можете сделать это с помощью отражения в Java:

class A {
  def cat(s1: String, s2: String) = s1 + " " + s2
}
val a = new A
val hi = "Hello"
val all = "World"
val method = a.getClass.getMethod("cat",hi.getClass,all.getClass)
method.invoke(a,hi,all)

И если вы хотите, чтобы в Scala это было легко, вы можете создать класс, который сделает это за вас, плюс неявный для преобразования:

case class Caller[T>:Null<:AnyRef](klass:T) {
  def call(methodName:String,args:AnyRef*):AnyRef = {
    def argtypes = args.map(_.getClass)
    def method = klass.getClass.getMethod(methodName, argtypes: _*)
    method.invoke(klass,args: _*)
  }
}
implicit def anyref2callable[T>:Null<:AnyRef](klass:T):Caller[T] = new Caller(klass)
a call ("cat","Hi","there")

Однако подобные действия преобразуют ошибки времени компиляции в ошибки времени выполнения (т. Е., По сути, обходят систему типов), поэтому используйте их с осторожностью.

(Отредактируйте: и посмотрите использование NameTransformer в ссылке выше - добавление, которое поможет, если вы попытаетесь использовать операторы.)

Да. Это называется отражением. Вот ссылка на один способ, используя некоторые экспериментальные вещи. Однако вы должны помнить, что Scala не является динамическим языком и может не иметь возможности легко делать некоторые вещи, которые могут делать языки сценариев. Возможно, вам лучше выполнить сопоставление строки, а затем вызвать правильный метод.

Да, ты можешь! Вам нужно .invoke() метод объекта метода. Простой пример ниже:

 case class MyCaseClass(i: String) {
   def sayHi = {
     println(i)
   }
 }
 val hiObj = MyCaseClass("hi")
 val mtdName = "sayHi"
 // Method itself as an object
 val mtd = hiObj.getClass.getMethod(mtdName)
 mtd.invoke(hiObj)
scala> val commandExecutor = Map("cleanup" -> {()=> println("cleanup successfully")} )
commandExecutor: scala.collection.immutable.Map[String,() => Unit] = Map(cleanup -> <function0>)

scala> val command="cleanup"
command: String = cleanup

scala> commandExecutor(command).apply
cleanup successfully
Другие вопросы по тегам