Динамически добавлять свойство или метод к объекту в Groovy - PullRequest
36 голосов
/ 24 января 2011

Можно ли динамически добавить свойство или метод к объекту в Groovy? Это то, что я пробовал до сих пор:

class Greet {
  def name
  Greet(who) { name = who[0].toUpperCase() + [1..-1] }
  def salute() { println "Hello $name!" }
}

g = new Greet('world')  // create object
g.salute()              // Output "Hello World!"

g.bye = { println "Goodbye, $name" }
g.bye()

Но я получаю следующее исключение:

Hello World!
Caught: groovy.lang.MissingPropertyException: No such property: bye for class: Greet
Possible solutions: name
    at test.run(greet.groovy:11)

1 Ответ

59 голосов
/ 24 января 2011

Если вы просто хотите добавить метод bye() к единственному экземпляру g класса Greet, вам нужно сделать:

g.metaClass.bye = { println "Goodbye, $name" }
g.bye()

В противном случае, чтобы добавить bye() ко всем экземплярам Greet (с этого момента), вызовите

Greet.metaClass.bye = { println "Goodbye, $name" }

Но вам нужно сделать это, прежде чем создавать экземпляр Greet класса

Вот страница метакласса для каждого экземпляра

А вот и общая страница MetaClasses


Также есть ошибка в вашем конструкторе. Вы пропускаете who в передней части вашего [1..-1], и если конструктору передается String длиной менее 2 символов, он выдаст исключение

Лучшая версия может быть:

Greet( String who ) { 
  name = who.inject( '' ) { String s, String c ->
    s += s ? c.toLowerCase() : c.toUpperCase()
  }
}

Как отмечено в комментариях,

Greet( String who ) { 
  name = who.capitalize()
}

это правильный путь

...