我需要构建一个小模块,将SpEL编码的表达式转换为字符串。这是为了抛弃Jexl并访问应用程序上下文。
例如,如果我正确配置的属性文件包含
application.name=AppTest
company.name=ACME Inc.
我希望根据属性
翻译以下路径或类似于它的字符串/path/to/#{application.name}/#{company.name}
#或$与我无关
通常在Spring中你可以将属性注入bean中,我已经成功了。但是现在我希望用户输入一个模板字符串,该字符串可以使用Application Context中的所有属性进行翻译。目前我不需要访问bean属性,但我将来可能会有。以上是定义文件夹路径时最常见方案的简化变体。运行时参数(例如一天中的时间)增加了很少的复杂性,但我问的问题是一步一步地工作。
所以我试图在JUnit的帮助下通过玩它们来理解Spring Expressions。我编写了以下代码,但无法使其工作
final HashMap<String, Object> propertySource = new HashMap<String, Object>();
private final String FOLDER_PATTERN = "#tmp/appTest/#{company}_#{appname}/q1"; //#tmp is only token being replaced "hardcoded", not passed to Spring
@Before
public void setUp() throws Exception
{
propertySource.put("appname", APPNAME);
propertySource.put("company", COMPANY);
applicationContext.getEnvironment()
.getPropertySources()
.addLast(new MapPropertySource("test", propertySource));
}
@Test
public void playWithExpression()
{
ExpressionParser expParser = new SpelExpressionParser();
StandardEvaluationContext stdEvaluationContext = new StandardEvaluationContext();
stdEvaluationContext.setBeanResolver(new BeanFactoryResolver(applicationContext.getBeanFactory()));
// stdEvaluationContext.setVariables(propertySource);
final TemplateParserContext templateParserContext = new TemplateParserContext();
String folderPattern = FOLDER_PATTERN.replace("#tmp", SYSTEM_TEMP_DIR);
String realPath = expParser.parseExpression(folderPattern, templateParserContext)
.getValue(stdEvaluationContext, String.class);
String calculatedPath = folderPattern.replace("#{company}", COMPANY)
.replace("#{appname}", APPNAME);
assertEquals(calculatedPath, realPath);
}
部分复制并粘贴其他示例,我:
#{..}
表示法表达式已成功实例化,但getValue抛出异常
org.springframework.expression.spel.SpelEvaluationException: EL1007E:(pos 0): Property or field 'company' cannot be found on null
at org.springframework.expression.spel.ast.PropertyOrFieldReference.readProperty(PropertyOrFieldReference.java:220)
at org.springframework.expression.spel.ast.PropertyOrFieldReference.getValueInternal(PropertyOrFieldReference.java:94)
at org.springframework.expression.spel.ast.PropertyOrFieldReference.getValueInternal(PropertyOrFieldReference.java:81)
at org.springframework.expression.spel.ast.SpelNodeImpl.getTypedValue(SpelNodeImpl.java:131)
at org.springframework.expression.spel.standard.SpelExpression.getValue(SpelExpression.java:299)
at org.springframework.expression.common.CompositeStringExpression.getValue(CompositeStringExpression.java:89)
at org.springframework.expression.common.CompositeStringExpression.getValue(CompositeStringExpression.java:136)
at it.phoenix.web.data.managers.test.FoldersManagerTemplateTests.playWithExpression(FoldersManagerTemplateTests.java:80)
从错误判断,似乎Spring正在尝试将company
标记解析为根对象的属性,在这种情况下会丢失。
如果我使用#{@...}
表示法,那么Spring会尝试将该标记解释为bean标识符。不是我的情况。
如何从表达式中的AppContext的PropertySource
解析属性?
答案 0 :(得分:2)
可能有一个更优雅的解决方案,但
private final String FOLDER_PATTERN = "#tmp/appTest/#{environment.getProperty('company')}"
+ "_#{environment.getProperty('appname')}/q1"; // #tmp is only token being replaced
// "hardcoded", not passed to Spring
与
String realPath = expParser.parseExpression(folderPattern, templateParserContext)
.getValue(stdEvaluationContext, applicationContext, String.class);
有效(即使用应用程序上下文作为评估中的根对象)。
修改强>
稍微好一些......
private final String FOLDER_PATTERN = "#tmp/appTest/#{getProperty('company')}"
+ "_#{getProperty('appname')}/q1"; // #tmp is only token being replaced
// "hardcoded", not passed to Spring
与
String realPath = expParser.parseExpression(folderPattern, templateParserContext)
.getValue(stdEvaluationContext, applicationContext.getEnvironment(), String.class);