为了账号安全,请及时绑定邮箱和手机立即绑定

action和actionListener之间的差异

action和actionListener之间的差异

慕村225694 2019-05-27 13:25:28
action和actionListener之间的差异是什么区别action和actionListener,什么时候应该使用action与actionListener?
查看完整描述

4 回答

?
慕桂英4014372

TA贡献1871条经验 获得超13个赞

的ActionListener

使用actionListener,如果你想有一个钩子之前得到执行的实际业务操作,如记录它,和/或设置附加属性(通过<f:setPropertyActionListener>),和/或访问其调用的动作(这是可以由组件ActionEvent参数)。因此,在实际业务操作被调用之前,纯粹是为了准备目的。

actionListener方法默认具有以下签名:

import javax.faces.event.ActionEvent;// ...public void actionListener(ActionEvent event) {
    // ...}

它应该声明如下,没有任何方法括号:

<h:commandXxx ... actionListener="#{bean.actionListener}" />

请注意,您不能通过EL 2.2 传递其他参数。但是,您可以ActionEvent通过传递和指定自定义参数来完全覆盖参数。以下示例有效:

<h:commandXxx ... actionListener="#{bean.methodWithoutArguments()}" /><h:commandXxx ... actionListener="#{bean.methodWithOneArgument(arg1)}" />
<h:commandXxx ... actionListener="#{bean.methodWithTwoArguments(arg1, arg2)}" />
public void methodWithoutArguments() {}public void methodWithOneArgument(Object arg1) {}public void methodWithTwoArguments(Object arg1, Object arg2) {}

请注意无参数方法表达式中括号的重要性。如果它们不存在,JSF仍然期望一个带ActionEvent参数的方法。

如果您使用的是EL 2.2+,则可以通过声明多个动作侦听器方法<f:actionListener binding>

<h:commandXxx ... actionListener="#{bean.actionListener1}">
    <f:actionListener binding="#{bean.actionListener2()}" />
    <f:actionListener binding="#{bean.actionListener3()}" /></h:commandXxx>
public void actionListener1(ActionEvent event) {}public void actionListener2() {}public void actionListener3() {}

请注意binding属性中括号的重要性。如果它们不存在,EL会混淆地抛出一个javax.el.PropertyNotFoundException: Property 'actionListener1' not found on type com.example.Bean,因为该binding属性默认被解释为值表达式,而不是方法表达式。添加EL 2.2+样式括号透明地将值表达式转换为方法表达式。另请参阅ao 为什么我能够将<f:actionListener>绑定到任意方法(如果JSF不支持它)?


行动

使用action,如果你想执行业务操作,并在必要时处理导航。该action方法可以(因此,不必)返回String将用作导航案例结果(目标视图)的方法。返回值为nullvoid将使其返回到同一页面并使当前视图范围保持活动状态。空字符串或相同视图ID的返回值也将返回到同一页面,但会重新创建视图范围,从而销毁任何当前活动的视图范围bean,并在适用时重新创建它们。

action方法可以是任何有效的MethodExpression,也可以是使用EL 2.2参数的方法,如下所示:

<h:commandXxx value="submit" action="#{bean.edit(item)}" />

用这种方法:

public void edit(Item item) {
    // ...}

请注意,当您的action方法仅返回一个字符串时,您也可以只在该action属性中指定该字符串。因此,这完全是笨拙的:

<h:commandLink value="Go to next page" action="#{bean.goToNextpage}" />

使用这种无意义的方法返回硬编码字符串:

public String goToNextpage() {
    return "nextpage";}

相反,只需将该硬编码字符串直接放在属性中:

<h:commandLink value="Go to next page" action="nextpage" />

请注意,这反过来表明设计不好:通过POST导航。这不是用户也不是SEO友好。这一切都在我何时应该使用h:outputLink而不是h:commandLink?应该被解决为

<h:link value="Go to next page" outcome="nextpage" />

另请参见如何在JSF中导航?如何使URL反映当前页面(而不是之前的页面)


f:ajax监听器

从JSF 2.x开始,第三种方式就是<f:ajax listener>

<h:commandXxx ...>
    <f:ajax listener="#{bean.ajaxListener}" /></h:commandXxx>

ajaxListener方法默认具有以下签名:

import javax.faces.event.AjaxBehaviorEvent;// ...public void ajaxListener(AjaxBehaviorEvent event) {
    // ...}

在Mojarra中,AjaxBehaviorEvent参数是可选的,下面的效果很好。

public void ajaxListener() {
    // ...}

但是在MyFaces中,它会抛出一个MethodNotFoundException。当你想省略参数时,下面的两个JSF实现都适用。

<h:commandXxx ...>
    <f:ajax execute="@form" listener="#{bean.ajaxListener()}" render="@form" /></h:commandXxx>

Ajax侦听器在命令组件上并不真正有用。它们对输入和选择组件<h:inputXxx>/ 更有用<h:selectXxx>。在命令组件中,只需坚持action和/或actionListener清晰,以及更好的自我记录代码。而且,例如actionListenerf:ajax listener不支持返回导航结果。

<h:commandXxx ... action="#{bean.action}">
    <f:ajax execute="@form" render="@form" /></h:commandXxx>

有关executerender属性的说明,请前往了解PrimeFaces进程/更新和JSF f:ajax执行/渲染属性


调用顺序

所述actionListeners的总是调用之前action以相同的顺序,因为它们是在视图被声明和连接到该组件。将f:ajax listener始终调用之前的任何动作侦听器。那么,以下示例:

<h:commandButton value="submit" actionListener="#{bean.actionListener}" action="#{bean.action}">
    <f:actionListener type="com.example.ActionListenerType" />
    <f:actionListener binding="#{bean.actionListenerBinding()}" />
    <f:setPropertyActionListener target="#{bean.property}" value="some" />
    <f:ajax listener="#{bean.ajaxListener}" /></h:commandButton>

将按以下顺序调用方法:

  1. Bean#ajaxListener()

  2. Bean#actionListener()

  3. ActionListenerType#processAction()

  4. Bean#actionListenerBinding()

  5. Bean#setProperty()

  6. Bean#action()


异常处理

actionListener支持一个特殊的例外:AbortProcessingException。如果从actionListener方法抛出此异常,则JSF将跳过任何剩余的动作侦听器和操作方法,并继续直接呈现响应。您不会看到错误/异常页面,但JSF会记录它。每当从一个异常抛出任何其他异常时,也会隐式地执行此操作actionListener。因此,如果您打算通过错误页面阻止页面作为业务异常的结果,那么您肯定应该在该action方法中执行该作业。

如果使用a的唯一理由actionListener是让void方法返回到同一页面,那么这是一个糟糕的方法。这些action方法可以完美地返回void,相反,某些IDE让您通过EL验证可以相信。请注意,PrimeFaces展示的例子actionListener在所有地方都充满了这种情况。这确实是错的。不要以此为借口自己也这样做。

但是,在ajax请求中,需要一个特殊的异常处理程序。这与您是否使用listener属性<f:ajax>无关。有关解释和示例,请转到JSF ajax请求中的异常处理


查看完整回答
反对 回复 2019-05-27
?
蛊毒传说

TA贡献1895条经验 获得超3个赞

TL; DR

ActionListener在他们登记前的顺序S(也可以是多个)执行action

答案很长

企业action通常调用EJB服务,并且如果必要的话也设置最终结果和/或导航到不同的视图,如果这不是您正在做的事情actionListener更合适,即当用户与组件交互时,例如h:commandButton或者h:link他们可以通过在actionListenerUI组件的属性中传递托管bean方法的名称或实现ActionListener接口并将实现类名称传递actionListener给UI组件的属性来处理。


查看完整回答
反对 回复 2019-05-27
  • 4 回答
  • 0 关注
  • 1230 浏览

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信