Java:如何在没有实现接口的情况下独立评估EL表达式?

15

我想在我的应用程序中使用EL,但是我找不到任何操作说明。通常情况下,我最终需要一些接口,而我没有实现它们。

我有一个对象映射表,并且我想要类似于“Hello,${person.name}”的字符串表达式被评估为字符串。

我该如何实现这一点,使用Commons EL、javax.el、OGNL或类似库?必须是独立的库。

我已经知道Java:在J2EE之外使用EL,也看过JSTL/JSP EL(Expression Language)在非JSP(独立)上下文中。那不是我要找的东西。

我正在寻找的是一个示例,演示要添加哪个依赖项,然后如何初始化解析器,该解析器将具有:

private static String evaluateEL( String expr, Map<String, String> properties );

并允许我做:

String greet = evaluateEL("Hello ${person.name}", 
     new HashMap(){{
       put("person", new Person("Ondra"));
     }}
);

我希望它使用一些合理的值,例如在null上使用""而不是抛出NPE之类的错误。


1
虽然不完全是EL,FreeMarker具有类似的语法和功能集合。 - fge
Velocity也有与EL类似的语法。 - Uwe Plonus
OGNL可能是这个,但他们的网站现在无法使用。 - Ondra Žižka
2个回答

22

目前有很多EL引擎,其中大部分实现了Java表达式语言API。

来源

目前我使用了BeanUtils编写了这段代码——虽然丑陋但可行。

import java.lang.reflect.InvocationTargetException;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.StringTokenizer;
import org.apache.commons.beanutils.PropertyUtils;
import org.apache.commons.lang.StringUtils;
import org.slf4j.LoggerFactory;

public static class SimpleEvaluator implements IExprLangEvaluator {
    private static final org.slf4j.Logger log = LoggerFactory.getLogger( SimpleEvaluator.class );

    @Override
    public String evaluateEL( String template, Map<String, String> properties ) {

        StringTokenizer st = new StringTokenizer( template );
        String text = st.nextToken("${");
        StringBuilder sb = new StringBuilder();

        // Parse the template: "Hello ${person.name} ${person.surname}, ${person.age}!"
        do{
            try {
                sb.append(text);
                if( ! st.hasMoreTokens() )
                    break;

                // "${foo.bar[a]"
                String expr  = st.nextToken("}");
                // "foo.bar[a].baz"
                expr = expr.substring(2);
                // "foo"
                String var = StringUtils.substringBefore( expr, ".");

                Object subject = properties.get( var );

                // "bar[a].baz"
                String propPath = StringUtils.substringAfter( expr, ".");

                sb.append( resolveProperty( subject, propPath ) );

                text = st.nextToken("${");
                text = text.substring(1);
            } catch( NoSuchElementException ex ){
                // Unclosed ${
                log.warn("Unclosed ${ expression, missing } : " + template);
            }
        } while( true );

        return sb.toString();
    }

    // BeanUtils
    private String resolveProperty( Object subject, String propPath ) {
        if( subject == null ) return "";

        if( propPath == null || propPath.isEmpty() ) return subject.toString();

        try {
            return "" + PropertyUtils.getProperty( subject, propPath );
        } catch(     IllegalAccessException | InvocationTargetException | NoSuchMethodException ex ) {
            log.warn("Failed resolving '" + propPath + "' on " + subject + ":\n    " + ex.getMessage(), ex);
            return "";
        }
    }

}// class SimpleEvaluator

5
我在http://juel.sourceforge.net/guide/start.html找到了一个,虽然不是完美的一行代码,但非常接近。
ExpressionFactory factory = new de.odysseus.el.ExpressionFactoryImpl();
de.odysseus.el.util.SimpleContext context = new de.odysseus.el.util.SimpleContext();
context.setVariable("foo", factory.createValueExpression("bar", String.class));
ValueExpression e = factory.createValueExpression(context, "Hello ${foo}!", String.class);
System.out.println(e.getValue(context)); // --> Hello, bar!

Maven依赖:

    <!-- Expression language -->
    <dependency>
        <groupId>de.odysseus.juel</groupId>
        <artifactId>juel-api</artifactId>
        <version>2.2.7</version>
    </dependency>
    <dependency>
        <groupId>de.odysseus.juel</groupId>
        <artifactId>juel-impl</artifactId>
        <version>2.2.7</version>
        <type>jar</type>
    </dependency>

2
请使用<artifactId>juel-impl</artifactId>,否则您将拥有两个冲突的EL-API。 - chris
根据评论,我已将artifactIdjuel更改为juel-impl,但没有测试它。 - Ondra Žižka
我已经更新了版本。看起来JUEL实现已将“groupId”更改为“de.odysseus.juel”。 - Ondra Žižka

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接