ActionMessages是Struts 1.1後所新增的類別,它變成了ActionErrors的父類別,同樣的,ActionMessage也是Struts 1.1新增的類別,ActionError則已經不建議使用。
ActionMessages搭配Struts的<html:messages>標籤,在管理訊息時就更為簡易,以 伺服端表單驗證 這篇為例,如果能使用ActionMessages搭配<html:messages>標籤,那麼訊息管理會很方便,例如 UserForm.java可以改為:
- UserForm.java
package onlyfun.caterpillar;
 
import javax.servlet.http.*; 
import org.apache.struts.action.*;
import org.apache.struts.Globals;
import org.apache.struts.util.MessageResources;
 
public class UserForm extends ActionForm {
    private String username;
    private String password;
    
    public void setUsername(String username) { 
        this.username = username; 
    } 
    
    public void setPassword(String password) { 
        this.password = password; 
    } 
    
    public String getUsername() { 
        return username; 
    }
    
    public String getPassword() {
        return password;
    } 
    
    public void reset(ActionMapping mapping, 
                      HttpServletRequest req) {
        username = null;
        password = null;
    }
    
    public ActionErrors validate(ActionMapping mapping, 
                                HttpServletRequest request) {
        ActionErrors errors = new ActionErrors();
        if(getUsername() == null || 
                getUsername().length() < 1) {
            errors.add(ActionMessages.GLOBAL_MESSAGE, 
                new ActionMessage("error.invalidUsername")); 
        }
        
        if(getPassword() == null || 
                getPassword().length() < 1) {
            errors.add(ActionMessages.GLOBAL_MESSAGE, 
                new ActionMessage("error.invalidPassword"));
        }
        
        return errors;
    }
}ActionMessage中的字串(例如 error.invalidUsername)對應於訊息檔案中的鍵值。LoginAction.java可以修改如下:
- LoginAction.java
package onlyfun.caterpillar;
 import javax.servlet.http.*; 
 import org.apache.struts.action.*; 
 import org.apache.commons.beanutils.PropertyUtils;
 
 public class LoginAction extends Action {
    public ActionForward execute(ActionMapping mapping,
                               ActionForm form,
                               HttpServletRequest request, 
                               HttpServletResponse response) 
                                  throws Exception { 
        
        String username = (String) 
          PropertyUtils.getSimpleProperty(form, "username"); 
        String password = (String) 
          PropertyUtils.getSimpleProperty(form, "password"); 
        
        request.setAttribute("username", username);
        
        if(username.equals("caterpillar") && 
                  password.equals("1234")) {
            return mapping.findForward("helloUser"); 
        }
        
        ActionMessages messages = new ActionMessages();
        messages.add(ActionMessages.GLOBAL_MESSAGE, 
          new ActionMessage("message.namePasswordMismatched")
          );
        addMessages(request, messages);
        
        return mapping.findForward("loginFail"); 
    }
}最後配合<html:messages>標籤,可以輸出ActionMessages的內容,來修改一下fail.jsp:
- fail.jsp
<%@ taglib uri="/tags/struts-html" prefix="html" %>
<%@ taglib uri="/tags/struts-bean" prefix="bean" %>
<html>
<head> 
<title>Sorry!</title> 
</head> 
 <body>
    <H1>
    <html:messages id="error"> 
        <bean:write name="error"/> 
    </html:messages>
    <html:messages id="message" message="true"> 
        <bean:write name="message"/> 
    </html:messages>
    
    </H1>
    <p>
    <a href='/strutsapp/html/form.htm'>Login</a>
 </body> 
</html><html:messages>的message屬性如果不設定為true,會輸出ActionErrors中所儲存的訊息,Error代表的是一個操作方面的錯誤,例如錯誤操作導致使用者名稱或密碼為空(當然也許也是故意的)。
<html:messages>的message屬性如果設定為true,會輸出ActionMessages中所儲存的訊息, Message表示一個提示訊息,也許使用者輸入了不正確的資訊,例如在輸入名稱與密碼時打錯了字,程式要提示使用者他們輸入了不正確的訊息。
在國際化訊息方面,<html:messages>標籤是根據session中的Locale物件來決定要顯示區域訊息,您可以在 Action中使用setLocale()方法來改變Locale物件,例如:
Locale locale = new Locale("zh", "CN");
setLocale(request, locale);
      
      
 setLocale(request, locale);
如上設定,<html:messages>會找messages_zh_CN.properties中的簡體中文訊息來顯示。現在的問題是,那ActionForm中的訊息呢?來看一下setLocale()中的原始碼您就知道怎麼作了:
protected void setLocale(HttpServletRequest request, 
Locale locale) {
HttpSession session = request.getSession();
if (locale == null) {
locale = Locale.getDefault();
}
session.setAttribute(Globals.LOCALE_KEY, locale);
}
      
      
 Locale locale) {
HttpSession session = request.getSession();
if (locale == null) {
locale = Locale.getDefault();
}
session.setAttribute(Globals.LOCALE_KEY, locale);
}
所以如果您要一開始進行 ActionForm 驗證時就使用區域化訊息,則可以在reset()或validate()方法中加入:
HttpSession session = request.getSession();
session.setAttribute(Globals.LOCALE_KEY, locale);
      
      
 session.setAttribute(Globals.LOCALE_KEY, locale);
或者您可以在必要的地方加入以上這段訊息,像是Servlet Filter等地方,這就看您的需求而定了。

