Groovy динамически вызываемый класс и метод find не работает?
Я пытаюсь построить динамический запрос, похожий на:
def domain = DomainName
def ids = 1
def domainClass = "$domain" as Class
domainClass.find("from ${domain} as m where m.job = ${ids} ").id
Но это не работает.
Если я пытаюсь это, все в порядке:
def domain = DomainName
def ids = 1
DomainName.find("from ${domain} as m where m.job = ${ids} ").id
Как я могу использовать динамическое имя класса домена с find?
4 ответа
Самый простой способ - это использовать getDomainClass
метод:
String domainClassName = 'com.foo.bar.Person'
def ids = 1
def domainClass = grailsApplication.getDomainClass(domainClassName).clazz
domainClass.find("from $domainClassName as m where m.job = ${ids} ").id
Обратите внимание, что если вы пытаетесь получить один экземпляр по идентификатору, используйте get
:
long id = 1234
def person = domainClass.get(id)
и если вы хотите получить несколько экземпляров и у вас есть список идентификаторов, вы можете использовать getAll
def ids = [1,2,3,4,5]
def people = domainClass.getAll(ids)
Также ДЕЙСТВИТЕЛЬНО плохая идея использовать GStrings со встроенными значениями свойств - Google 'SQL Injection'
Например, чтобы найти человека по имени пользователя:
String username = 'foo'
def person = domainClass.find(
"from $domainClassName as m where m.username=:username",
[username: username])
Вы должны быть в состоянии сделать это, явно используя GroovyClassLoader:
def domain = "DomainName"
def c = new GroovyClassLoader().loadClass(domain)
c.find('...').id
Лучший способ получить класс Domain динамически - использовать объект GrailsApplication. Пример:
import org.codehaus.groovy.grails.commons.ApplicationHolder
def domainName = "full.package.DomainName"
def domainGrailsClass = ApplicationHolder.application.getArtefact("Domain", domainName)
def domainClass = domainGrailsClass.getClazz()
domainClass.find("from ${domainGrailsClass.name} as m where m.job = ${ids}").id
Вы также можете использовать Class.forName()
так же, как вы бы в Java. Используйте версию с 3 параметрами и передайте загрузчик класса контекста текущего потока:
import grails.util.GrailsNameUtils
def domainName = "full.package.DomainName"
def domainClass = Class.forName(domainName, true, Thread.currentThread().getContextClassLoader())
domainClass.find("from ${GrailsNameUtils.getShortName(domainName)} as m where m.job = ${ids}").id
Загрузчики классов - ужасная тема в Java и JVM-фреймворках. В Grails вы почти всегда хотите использовать загрузчик классов контекста потока. Но еще лучше - использовать интерфейс GrailsApplication и вообще избежать этой проблемы.
Использовать GrailsClassUtils
GrailsClassUtils.getShortName(DomainName)
чтобы получить название класса, так что это должно работать... если я понял вопрос
def domainClassName = GrailsClassUtils.getShortName(DomainName)
def ids = 1
DomainName.find("from ${domainClassName} as m where m.job = ${ids} ").id