java - 如何在freemarker模板中调用反射方法?

标签 java reflection code-generation freemarker

我们可以使用 java 中的一些代码来迭代类字段、注释:

Class<?> clazz;
for(Field field : clazz.getDeclaredFields()) {
    for(Annotation annotation : field.getDeclaredAnnotations()) {
        for(Method method : annotation.annotationType().getMethods()) {
            if(method.getName().equals("nullable")) {
                try {
                    // note this line
                    System.out.println(method.invoke(annotation, (Object[])null) );
                }
                catch(Exception e) {
                    System.out.println("Ex");
                }
            }
        }
    }
}

上面的代码想要检查字段注释之一是否具有nullable方法,记录它的值。假设这个字段是某个类:

public class someClass {
    @Column(nullable = false)
    private Long number;

    @ManyToOne(fetch = FetchType.LAZY)
    @JoinColumn(name = "entity_id", nullable = true)
    private Entity entity;
}

该类的上述代码记录 false,然后记录 true 作为字段注释中 nullable 属性的值。现在我想迭代 freemarker 模板中的类字段和注释,但问题是我无法调用 method.invoke(annotation, (Object[])null)

<#list fields as field>
    <#list field.annotations as annotation>
        <#if annotation??>
            <#if annotation.annotationType().getSimpleName() == "Column" || annotation.annotationType().getSimpleName() == "JoinColumn">
                <#list annotation.annotationType().getMethods() as annotationMethod>
                    <#attempt>
                        <#if annotationMethod.getName() == "nullable">
                            // this line get error 
                            ${annotationMethod.invoke(annotation, (Object[])null)}
                        </#if>
                        <#recover>
                    </#attempt>
                </#list>
                <#break>
            </#if>
        </#if>
    </#list>
</#list>

在 freemarker 模板中是否有其他方法可以实现此目标? 我想将这些代码用于代码生成器。

UDPATE

这是我给出的异常(exception):

Syntax error in template "entity-detail.ftl" in line 32, column 111:
Encountered "]", but was expecting one of:
    <STRING_LITERAL>
    <RAW_STRING>
    "false"
    "true"
    <INTEGER>
    <DECIMAL>
    "."
    "+"
    "-"
    "!"
    "["
    "("
    "{"
    <ID>
freemarker.core.ParseException: Syntax error in template "entity-detail.ftl" in line 32, column 111:
Encountered "]", but was expecting one of:
    <STRING_LITERAL>
    <RAW_STRING>
    "false"
    "true"
    <INTEGER>
    <DECIMAL>
    "."
    "+"
    "-"
    "!"
    "["
    "("
    "{"
    <ID>
    at freemarker.core.FMParser.generateParseException(FMParser.java:5749)
    at freemarker.core.FMParser.jj_consume_token(FMParser.java:5608)
    at freemarker.core.FMParser.UnaryExpression(FMParser.java:658)
    at freemarker.core.FMParser.MultiplicativeExpression(FMParser.java:768)
    at freemarker.core.FMParser.AdditiveExpression(FMParser.java:720)
    at freemarker.core.FMParser.RangeExpression(FMParser.java:900)
    at freemarker.core.FMParser.RelationalExpression(FMParser.java:848)
    at freemarker.core.FMParser.EqualityExpression(FMParser.java:811)
    at freemarker.core.FMParser.AndExpression(FMParser.java:967)
    at freemarker.core.FMParser.OrExpression(FMParser.java:989)
    at freemarker.core.FMParser.Expression(FMParser.java:548)
    at freemarker.core.FMParser.DynamicKey(FMParser.java:1341)
    at freemarker.core.FMParser.AddSubExpression(FMParser.java:1107)
    at freemarker.core.FMParser.PrimaryExpression(FMParser.java:607)
    at freemarker.core.FMParser.UnaryExpression(FMParser.java:653)
    at freemarker.core.FMParser.MultiplicativeExpression(FMParser.java:768)
    at freemarker.core.FMParser.AdditiveExpression(FMParser.java:720)
    at freemarker.core.FMParser.RangeExpression(FMParser.java:900)
    at freemarker.core.FMParser.RelationalExpression(FMParser.java:848)
    at freemarker.core.FMParser.EqualityExpression(FMParser.java:811)
    at freemarker.core.FMParser.AndExpression(FMParser.java:967)
    at freemarker.core.FMParser.OrExpression(FMParser.java:989)
    at freemarker.core.FMParser.Expression(FMParser.java:548)
    at freemarker.core.FMParser.Parenthesis(FMParser.java:616)
    at freemarker.core.FMParser.PrimaryExpression(FMParser.java:588)
    at freemarker.core.FMParser.UnaryExpression(FMParser.java:653)
    at freemarker.core.FMParser.MultiplicativeExpression(FMParser.java:768)
    at freemarker.core.FMParser.AdditiveExpression(FMParser.java:720)
    at freemarker.core.FMParser.RangeExpression(FMParser.java:900)
    at freemarker.core.FMParser.RelationalExpression(FMParser.java:848)
    at freemarker.core.FMParser.EqualityExpression(FMParser.java:811)
    at freemarker.core.FMParser.AndExpression(FMParser.java:967)
    at freemarker.core.FMParser.OrExpression(FMParser.java:989)
    at freemarker.core.FMParser.Expression(FMParser.java:548)
    at freemarker.core.FMParser.PositionalArgs(FMParser.java:2985)
    at freemarker.core.FMParser.MethodArgs(FMParser.java:1357)
    at freemarker.core.FMParser.AddSubExpression(FMParser.java:1111)
    at freemarker.core.FMParser.PrimaryExpression(FMParser.java:607)
    at freemarker.core.FMParser.UnaryExpression(FMParser.java:653)
    at freemarker.core.FMParser.MultiplicativeExpression(FMParser.java:768)
    at freemarker.core.FMParser.AdditiveExpression(FMParser.java:720)
    at freemarker.core.FMParser.RangeExpression(FMParser.java:900)
    at freemarker.core.FMParser.RelationalExpression(FMParser.java:848)
    at freemarker.core.FMParser.EqualityExpression(FMParser.java:811)
    at freemarker.core.FMParser.AndExpression(FMParser.java:967)
    at freemarker.core.FMParser.OrExpression(FMParser.java:989)
    at freemarker.core.FMParser.Expression(FMParser.java:548)
    at freemarker.core.FMParser.StringOutput(FMParser.java:1522)
    at freemarker.core.FMParser.MixedContentElements(FMParser.java:3721)
    at freemarker.core.FMParser.If(FMParser.java:1625)
    at freemarker.core.FMParser.FreemarkerDirective(FMParser.java:3384)
    at freemarker.core.FMParser.MixedContentElements(FMParser.java:3772)
    at freemarker.core.FMParser.Attempt(FMParser.java:1671)
    at freemarker.core.FMParser.FreemarkerDirective(FMParser.java:3511)
    at freemarker.core.FMParser.MixedContentElements(FMParser.java:3772)
    at freemarker.core.FMParser.List(FMParser.java:1746)
    at freemarker.core.FMParser.FreemarkerDirective(FMParser.java:3388)
    at freemarker.core.FMParser.MixedContentElements(FMParser.java:3772)
    at freemarker.core.FMParser.If(FMParser.java:1625)
    at freemarker.core.FMParser.FreemarkerDirective(FMParser.java:3384)
    at freemarker.core.FMParser.MixedContentElements(FMParser.java:3772)
    at freemarker.core.FMParser.If(FMParser.java:1625)
    at freemarker.core.FMParser.FreemarkerDirective(FMParser.java:3384)
    at freemarker.core.FMParser.MixedContentElements(FMParser.java:3772)
    at freemarker.core.FMParser.List(FMParser.java:1746)
    at freemarker.core.FMParser.FreemarkerDirective(FMParser.java:3388)
    at freemarker.core.FMParser.MixedContentElements(FMParser.java:3772)
    at freemarker.core.FMParser.Attempt(FMParser.java:1671)
    at freemarker.core.FMParser.FreemarkerDirective(FMParser.java:3511)
    at freemarker.core.FMParser.MixedContentElements(FMParser.java:3772)
    at freemarker.core.FMParser.List(FMParser.java:1746)
    at freemarker.core.FMParser.FreemarkerDirective(FMParser.java:3388)
    at freemarker.core.FMParser.MixedContentElements(FMParser.java:3772)
    at freemarker.core.FMParser.Root(FMParser.java:4411)
    at freemarker.template.Template.<init>(Template.java:252)
    at freemarker.cache.TemplateCache.loadTemplate(TemplateCache.java:549)
    at freemarker.cache.TemplateCache.getTemplateInternal(TemplateCache.java:439)
    at freemarker.cache.TemplateCache.getTemplate(TemplateCache.java:292)
    at freemarker.template.Configuration.getTemplate(Configuration.java:2654)
    at freemarker.template.Configuration.getTemplate(Configuration.java:2503)
    at com.vira.framework.common.generator.CodeGenerator.generate(CodeGenerator.java:70)
    at com.vira.framework.common.generator.CodeGeneratorTest.main(CodeGeneratorTest.java:7)

最佳答案

如果您只是从注释中获取“可为空”值。则不需要此代码

       <#list annotation.annotationType().getMethods() as annotationMethod>
                <#attempt>
                    <#if annotationMethod.getName() == "nullable">
                        // this line get error 
                        ${annotationMethod.invoke(annotation, (Object[])null)}
                    </#if>
                    <#recover>
                </#attempt>
            </#list>

就写这个!!

${annotation.nullable()}

关于java - 如何在freemarker模板中调用反射方法?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48871598/

相关文章:

c# - Java、C# 和 Prolog 反射机制

java - JFrame 中的背景图像覆盖了我的按钮

java - 如何获取存储在字段中的对象,然后获取其类,然后获取该类的字段等? (递归获取类的字段)

java - 是否有 Class.isAssignableFrom 与 Type 对象一起使用的替代方法?

java - 使用反射调用方法/将列表转换为 Var Args

c# - 使用反射实例化不可变对象(immutable对象)

java - 自定义 "hash table"实现: Why is it so slow?(字节码生成)

java hmac/sha512 生成

build - 如何使用 dart source_gen 生成文件到不同的目录

c++11 - 可变模板函数的显式实例化