Есть ли какая-либо функция 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