我将一些参数传递给通过查询字符串实现ModelDriven<Transporter>
的动作类。
<s:form namespace="/admin_side" action="Test" id="dataForm" name="dataForm">
<s:url id="editURL" action="EditTest" escapeAmp="false">
<s:param name="transporterId" value="1"/>
<s:param name="transporterName" value="'DHL'"/>
</s:url>
<s:a href="%{editURL}">Click</s:a>
</s:form>
动作类如下。
@Namespace("/admin_side")
@ResultPath("/WEB-INF/content")
@ParentPackage(value = "struts-default")
public final class TestAction extends ActionSupport
implements Serializable, Preparable, ModelDriven<Transporter>
{
private static final long serialVersionUID = 1L;
private Transporter transporter = new Transporter();
@Action(value = "Test",
results = {
@Result(name = ActionSupport.SUCCESS, location = "Test.jsp"),
@Result(name = ActionSupport.INPUT, location = "Test.jsp")},
interceptorRefs = {
@InterceptorRef(value = "paramsPrepareParamsStack",
params = {"params.acceptParamNames", "transporterId, transporterName"})})
public String load() throws Exception {
return ActionSupport.SUCCESS;
}
@Action(value = "EditTest",
results = {
@Result(name = ActionSupport.SUCCESS, location = "Test.jsp"),
@Result(name = ActionSupport.INPUT, location = "Test.jsp")},
interceptorRefs = {
@InterceptorRef(value = "paramsPrepareParamsStack",
params = {"params.acceptParamNames", "transporterId, transporterName"})})
public String edit() {
System.out.println(transporter.getTransporterId()
+ " : " + transporter.getTransporterName());
return ActionSupport.SUCCESS;
}
@Override
public Transporter getModel() {
return transporter;
}
@Override
public void prepare() throws Exception {}
}
服务器终端显示以下消息。
Jan 09, 2014 4:06:32 PM com.opensymphony.xwork2.interceptor.ParametersInterceptor error
SEVERE: Developer Notification (set struts.devMode to false to disable this message):
Unexpected Exception caught setting 'transporterId' on 'class actions.TestAction: Error setting expression 'transporterId' with value ['1', ]
Jan 09, 2014 4:06:32 PM com.opensymphony.xwork2.interceptor.ParametersInterceptor error
SEVERE: Developer Notification (set struts.devMode to false to disable this message):
Unexpected Exception caught setting 'transporterName' on 'class actions.TestAction: Error setting expression 'transporterName' with value ['DHL', ]
即使日志级别为SEVERE
,这些参数的值也可在操作类中使用
System.out.println(transporter.getTransporterId()
+ " : " + transporter.getTransporterName());
在edit()
方法中。
如果paramsPrepareParamsStack
被defaultStack
替换,那么这些消息就会消失。
像['DHL', ]
这样的表达式表示一个数组。模型中的transporterId
和transporterName
分别为Long
和String
类型。
我做错了什么?
答案 0 :(得分:5)
此处不涉及数组问题(即使看起来像这样):这种异常意味着Struts无法为您的参数找到Setter:
来自ParametersInterceptor documentation:
缺少参数的警告
当给定参数名称没有setter时,会显示一条警告消息 如下所示将记录在devMode中:
SEVERE: Developer Notification (set struts.devMode to false to disable this message): Unexpected Exception caught setting 'search' on 'class demo.ItemSearchAction: Error setting expression 'search' with value ['search', ] Error setting expression 'search' with value ['search', ] - [unknown location] at com.opensymphony.xwork2.ognl.OgnlValueStack.handleRuntimeException(OgnlValueStack.java:201) at com.opensymphony.xwork2.ognl.OgnlValueStack.setValue(OgnlValueStack.java:178) at com.opensymphony.xwork2.ognl.OgnlValueStack.setParameter(OgnlValueStack.java:152)
因此,预期行为允许开发人员发现缺少的setter 或参数名称或设置器中的拼写错误。
通过在JSP中放置一个Action中不存在的元素,可以轻松地重现此错误。
由于您的属性(使用其Setters)存在于模型中,并且您使用的是ModelDriven
和paramsPrepareParamsStack
,我认为正在进行的是:
ModelDriven Interceptor
被委派处理Model对象; Parameters Interceptor
时,ModelDriven Interceptor
尚未运行; 但如果这是真的,那么你应该 NOT 能够在prepare()
方法中检索这些参数(这就是你使用这个堆栈的原因......):请尝试,并在此处发布结果。
我想要解决此问题的第一件事就是将ModelDriven Interceptor
放在第一个Parameters Interceptor
之前(通过复制它,或者通过移动它,我不确定哪个如果有任何副作用,它可以在两种情况下产生,你应该再次尝试在这里报告)。
然后定义以下堆栈,并使用它。
<interceptor-stack name="modelParamsPrepareParamsStack">
<interceptor-ref name="exception"/>
<interceptor-ref name="alias"/>
<interceptor-ref name="i18n"/>
<interceptor-ref name="checkbox"/>
<interceptor-ref name="multiselect"/>
<!-- NEW ModelDriven Position -->
<interceptor-ref name="modelDriven"/>
<interceptor-ref name="params">
<param name="excludeParams">^dojo\..*,^struts\..*,^session\..*,^request\..*,^application\..*,^servlet(Request|Response)\..*,^parameters\..*,^action:.*,^method:.*</param>
</interceptor-ref>
<interceptor-ref name="servletConfig"/>
<interceptor-ref name="prepare"/>
<interceptor-ref name="chain"/>
<!-- OLD ModelDriven Position -->
<!--interceptor-ref name="modelDriven"/-->
<interceptor-ref name="fileUpload"/>
<interceptor-ref name="staticParams"/>
<interceptor-ref name="actionMappingParams"/>
<interceptor-ref name="params">
<param name="excludeParams">^dojo\..*,^struts\..*,^session\..*,^request\..*,^application\..*,^servlet(Request|Response)\..*,^parameters\..*,^action:.*,^method:.*</param>
</interceptor-ref>
<interceptor-ref name="conversionError"/>
<interceptor-ref name="validation">
<param name="excludeMethods">input,back,cancel,browse</param>
</interceptor-ref>
<interceptor-ref name="workflow">
<param name="excludeMethods">input,back,cancel,browse</param>
</interceptor-ref>
</interceptor-stack>
希望有所帮助。
答案 1 :(得分:0)
在您提供的代码中,找不到运输类的声明。
所以我想可能是因为您的Transpoter类具有比2多的参数,而不仅仅是id和name。
实际上,此错误消息总是在我提到的情况下发生。
要解决此问题,您可以定义一个仅具有两个属性id和name的数据传输对象(DTO)。使用此DTO接受来自jsp的参数,然后将属性值传递给Transporter对象。
我在2019年看到了这个问题,并提供了解决方案,希望将来可以被其他人使用。