Java Tapestry:为什么 onActivate 方法被调用了两次?

Java Tapestry : why is onActivate method called twice?

我有一个挂毯页面,其中有两个参数不同的 onActivate() 方法。

// 在这种情况下调用了两个方法

void onActivate(String filterSetJson) {
    schema = getSchema();
    if (ValueHelper.isDefined(filterSetJson) && !filterSetJson.equals("all")) {
        setFilterSet( filterSetJson );
    }
    else {
        setFilterSet("{}"); // Empty filter set
    }
    this.loadedTagSummary = false;
    sortOrderForTags="count";
    sortOrderForTypes="count";
}


void onActivate(String filterSetJson, int pageCount) {
    //onActivate( filterSetJson );
    this.pageCount = pageCount;
}

有时使用两个参数调用此页面,有时使用一个参数。问题在于,当使用两个参数调用页面时,将调用两个 onActivate() 方法。只调用接受两个参数的方法。

按照下面 link 中的建议,我修改了代码,使其只有一个接受 EventContext 作为参数的 onActivate() 方法。 http://apache-tapestry-mailing-list-archives.1045711.n5.nabble.com/Multiple-onActivate-methods-td2434897.html#a2434901

所以修改后的代码如下

//下面的方法在本例中被调用了两次

Object onActivate(EventContext eContext){
    int paramCount = eContext.getCount();
    if( paramCount == 0 )   return true;
    String filterSetJson = eContext.get(String.class, 0);
    if( paramCount > 1 ){
        int pageCount = eContext.get(Integer.class, 1);
        this.pageCount = pageCount;
    }
    schema = getSchema();
    if (ValueHelper.isDefined(filterSetJson) && !filterSetJson.equals("all")) {
        setFilterSet( filterSetJson );
    }
    else {
        setFilterSet("{}"); // Empty filter set
    }
    this.loadedTagSummary = false;
    sortOrderForTags="count";
    sortOrderForTypes="count";
    return true;    
}

现在,问题是 onActivate() 仍然被调用了两次。一旦参数存在,另一次没有参数(我认为这次 EventContext 是空的)。因此,如果参数不存在,我必须添加对参数和 return 的检查。这避免了整个代码被执行两次,但仍然不是一个干净的修复。我不明白为什么该方法被调用两次。我考虑了一些邮件列表帖子中建议的潜在原因。

1: 我确定页面上没有图像被赋予相对路径。所有图像加载 'context'

2: 没有超级 class 实现 onActivate() 方法。它只出现在这个 class.

有趣的是,正如下面 link 中所建议的,如果我从两个 onActivate() 方法中 return 为真,那么只调用一个方法,问题就解决了! http://apache-tapestry-mailing-list-archives.1045711.n5.nabble.com/T5-onActivate-called-twice-td2428923.html

这是工作正常的代码。

//本例只调用了一个方法。这是正确的行为

Object onActivate(String filterSetJson) {
    schema = getSchema();
    if (ValueHelper.isDefined(filterSetJson) && !filterSetJson.equals("all")) {
        setFilterSet( filterSetJson );
    }
    else {
        setFilterSet("{}"); // Empty filter set
    }
    this.loadedTagSummary = false;
    sortOrderForTags="count";
    sortOrderForTypes="count";
    return true;
}


Object onActivate(String filterSetJson, int pageCount) {
    //onActivate( filterSetJson );
    this.pageCount = pageCount;
    return true;
}

我可以接受这个解决方案,但我不明白这里发生了什么。谁能解释为什么两个 onActivate() 方法在它们不 return true 时被调用,或者为什么 onActivate(EventContext eContext) 被调用两次?

发布此问题的另一个原因是为我自己和其他人记录这个问题,因为我在这个问题上花了很多时间。

Tapestry 版本为 5.3

请阅读页面 here,尤其是

A good "rule of thumb" is to use onActivate(...) and onPassivate() for no more than receiving and returning the activation context.

您应该小心使用 onActivate,因为它可能会在您意想不到的时候被调用(例如渲染 pagelink 时)。您的 onActivate 方法应该是 'dumb' 并且仅存储值,仅此而已。将您的逻辑放在另一个事件中(例如 setupRender)。

我似乎还记得 Tapestry 中的一些奇怪逻辑,它将调用所有 onActivate() 方法,其数量等于或小于 URL.[=23 上的页面激活上下文变量的数量=]

我会:

1) 有一个 onActivate(EventContext) 方法来存储值。将任何逻辑移动到 setupRender.

2) 使用@PageActivationContextsetupRender

@PageActivationContext(index=0)
private String filterSetJson;

@PageActivationContext(index=1)
private Integer pageCount;

void setupRender() {
    if (pageCount == null) {
        ...
    } else {
        ...
    }
}

注意:多个@PageActivationContext是最近new feature

这两个选项大致相同。