[英]How does the JSF EL Field Referencing From Getter Setter work?
我想将其复制到我自己的项目中,而不是xhtml或其他xml结构化文件。 我想在Java文件中复制这种功能。 可能吗?
#{Object.field}
示例类
class Object
{
public String getField()
{
return "field";
}
public void setField()
{
}
}
它以某种方式将getter和setter的后缀转换为像实际的字段引用一样工作。 但是我希望开发人员只能将实际字段放进去。 因此,当他们不输入实际字段的名称(这是getter和setter函数的后缀)时,它将在编译时显示错误。
我从字面上不想使用EL表达式。 “#{Object.field}”。 我只想复制系统,以使开发人员不必使用循环,getter和setter来创建表。 如果可能的话,我希望他们能够像这样编码
createTable(new Field[]{Object.field1, Object.field4, Object.field6}, new Object[]{object1, object2, object3});
//new Field[]{Object.field1, Object.field4} <-- selected fields
//new Object[]{object1, object2, object3} <-- objects that is going to be shown in the table
我发现可以通过使用Java反射来做到这一点。 但是要获得一个特定的“字段”有很多麻烦。 如果我可以像“ Object.fieldName”一样简单地获取“ Field”,那就太好了。
我认为您需要的是EL Resolver。 只要有任何EL字符串,就需要从javax&实现ELResolver,只需使用自己的EL Resolver即可解决它。
下面是示例代码。 它将无法正常工作,因此请尝试删除不必要的内容,例如函数映射器,变量映射器等,然后运行它。
Maven依赖
<dependency>
<groupId>javax.el</groupId>
<artifactId>el-api</artifactId>
<version>2.2</version>
</dependency>
<dependency>
<groupId>org.apache.tomcat</groupId>
<artifactId>jasper-el</artifactId>
<version>6.0.18</version>
</dependency>
定制EL旋转变压器
package ravi.tutorial.el;
import java.beans.FeatureDescriptor;
import java.util.Iterator;
import java.util.Map;
import javax.el.ELContext;
import javax.el.ELResolver;
import javax.el.MapELResolver;
public class DemoELResolver extends ELResolver {
private ELResolver delegate = new MapELResolver();
private Map<Object, Object> userMap;
public DemoELResolver(Map<Object, Object> userMap) {
this.userMap = userMap;
}
@Override
public Object getValue(ELContext context, Object base, Object property) {
if (base == null) {
base = userMap;
}
return delegate.getValue(context, base, property);
}
@Override
public Class<?> getCommonPropertyType(ELContext context, Object base) {
if (base == null) {
base = userMap;
}
return delegate.getCommonPropertyType(context, base);
}
@Override
public Iterator<FeatureDescriptor> getFeatureDescriptors(ELContext context,
Object base) {
if (base == null) {
base = userMap;
}
return delegate.getFeatureDescriptors(context, base);
}
@Override
public Class<?> getType(ELContext context, Object base, Object property) {
if (base == null) {
base = userMap;
}
return delegate.getType(context, base, property);
}
@Override
public boolean isReadOnly(ELContext context, Object base, Object property) {
if (base == null) {
base = userMap;
}
return delegate.isReadOnly(context, base, property);
}
@Override
public void setValue(ELContext context, Object base, Object property,
Object value) {
if (base == null) {
base = userMap;
}
delegate.setValue(context, base, property, value);
}
}
使用ELResolver
ExpressionFactory expressionFactory;
if (args == null || args.length == 0) {
System.out
.println("Enter command line argument 1=Apache Jasper 2=Sourceforge JUEL.");
return;
}
if (args[0].equals("1")) {
expressionFactory = new org.apache.el.ExpressionFactoryImpl();
System.out.println("Choosing org.apache.el.ExpressionFactoryImpl");
} else if (args[0].equals("2")) {
expressionFactory = new de.odysseus.el.ExpressionFactoryImpl();
System.out.println("Choosing de.odysseus.el.ExpressionFactoryImpl");
} else {
System.out.println("Wrong argument");
return;
}
// create a map with some variables in it
Map<Object, Object> userMap = new HashMap<Object, Object>();
userMap.put("x", new Integer(123));
userMap.put("y", new Integer(456));
// get the method for ${myprefix:hello(string)}
Method sayHello = DemoEL.class.getMethod("sayHello",
new Class[] { String.class });
// create the context
ELResolver demoELResolver = new DemoELResolver(userMap);
final VariableMapper variableMapper = new DemoVariableMapper();
final DemoFunctionMapper functionMapper = new DemoFunctionMapper();
functionMapper.addFunction("myprefix", "hello", sayHello);
final CompositeELResolver compositeELResolver = new CompositeELResolver();
compositeELResolver.add(demoELResolver);
compositeELResolver.add(new ArrayELResolver());
compositeELResolver.add(new ListELResolver());
compositeELResolver.add(new BeanELResolver());
compositeELResolver.add(new MapELResolver());
ELContext context = new ELContext() {
@Override
public ELResolver getELResolver() {
return compositeELResolver;
}
@Override
public FunctionMapper getFunctionMapper() {
return functionMapper;
}
@Override
public VariableMapper getVariableMapper() {
return variableMapper;
}
};
// create and resolve a value expression
String sumExpr = "${x+y}";
ValueExpression ve = expressionFactory.createValueExpression(context,
sumExpr, Object.class);
Object result = ve.getValue(context);
System.out.println("Result=" + result);
// call a function
String fnExpr = "#{myprefix:hello('Dave')}";
ValueExpression fn = expressionFactory.createValueExpression(context,
fnExpr, Object.class);
fn.getValue(context);
参考: http : //weblogs.java.net/blog/felipeal/archive/2008/10/adding_el_suppo.html
这些表达式是字符串,只能在运行时求值。 字段名称以get / set为前缀,以形成方法名称,然后使用Reflection API在对象上调用该方法名称。
我认为,如果有可能在编译时执行类似的操作,那么他们肯定会这样做。
Object.field1
无效,除非您实际上在该类中有一个类似的公共字段。 它必须是字符串"Object.field1"
。 而且,您在编译时实际上无法检查该字符串的内容。
因此,不可能在编译时强制执行该操作。 最好的办法是抛出运行时异常。
我认为您不需要构建像EL-API一样复杂的系统。 只需学习Reflection API 。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.