Skip to content

17. Exemplo 15 – Integração Struts 2 / Spring

No exemplo anterior, não tínhamos quaisquer dados no âmbito da aplicação partilhados entre todas as solicitações de todos os utilizadores. Vamos mostrar um exemplo disso aqui. Para o implementar, utilizaremos o framework Spring [http://www.springsource.org/]. O Spring é uma ferramenta extremamente valiosa. Estamos a mostrar apenas uma pequena fração dele neste exemplo. Um exemplo posterior irá utilizá-lo de forma mais abrangente.

O objetivo desta aplicação é apresentar dados no âmbito da aplicação.

17.1. O projeto NetBeans

O projeto NetBeans para a aplicação é o seguinte:

  • em [1]:
  • O ficheiro [web.xml], que configura a aplicação web, irá sofrer alterações em relação às versões anteriores
  • [applicationContext.xml] é o ficheiro de configuração do Spring
  • em [2]: a vista [Context.jsp], que irá apresentar dados no âmbito da aplicação
  • em [3]:
  • o ficheiro de mensagens [messages.properties]
  • o ficheiro de configuração principal do Struts [struts.xml]. Sofrerá alterações em comparação com aplicações anteriores.
  • em [4]:
  • a ação Struts [Action1.java]
  • a classe [Config.java], que armazenará dados no âmbito da aplicação
  • o ficheiro de configuração secundário do Struts [example.xml]
  • em [5]: a biblioteca Struts 2
  • em [6]:
  • os arquivos necessários para o Spring [spring-core, spring-context, spring-beans, spring-web, commons-logging]
  • o arquivo do plugin Spring para o Struts 2. Permite a integração do Spring com o Struts 2 [struts2-spring-plugin]

Em comparação com versões anteriores:

  • é necessário adicionar arquivos ao projeto
  • modificar os ficheiros [web.xml] e [struts.xml]

17.2. Configuração

17.2.1. O ficheiro [web.xml]

O ficheiro [web.xml] é alterado da seguinte forma:


<?xml version="1.0" encoding="UTF-8"?>
<web-app id="WebApp_9" version="2.4" xmlns="http://java.sun.com/xml/ns/j2ee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd">
  <display-name>Exemple 14</display-name>
  <filter>
    <filter-name>struts2</filter-name>
    <filter-class>org.apache.struts2.dispatcher.FilterDispatcher</filter-class>
  </filter>
  <filter-mapping>
    <filter-name>struts2</filter-name>
    <url-pattern>/*</url-pattern>
  </filter-mapping>
  <listener>
    <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
  </listener>
</web-app>

A alteração ocorre nas linhas 12–14. É adicionado um ouvinte à aplicação web. Quando a aplicação web for iniciada, a classe que implementa este ouvinte será instanciada. Esta é uma classe Spring. Esta classe utilizará o ficheiro [WEB-INF/applicationContext.xml]. O ficheiro é o seguinte:


<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:tx="http://www.springframework.org/schema/tx"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.0.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.0.xsd">
 
  <!-- application scope bean configuration -->
  <bean id="config" class="example.Config" >
    <property name="nbMaxUsers" value="10"/>
  </bean>
 
</beans>
  • O ficheiro de configuração do Spring tem a tag raiz <beans> (linhas 2 e 11). Os beans podem ser considerados como objetos. O Spring irá instanciar todos os objetos (beans) encontrados neste ficheiro de configuração. Faz isso apenas uma vez, quando o ouvinte do Spring é iniciado no arranque da aplicação web.
  • Linhas 7–9: definem um bean chamado config (id). O bean config está associado à classe [example.Config]. O Spring irá instanciar esta classe.
  • Linha 8: define uma propriedade da classe [example.Config]. O Spring chamará o método [example.Config].setNbMaxUsers(10). Portanto, o método setNbMaxUsers deve existir na classe. É o seguinte:

package example;
 
public class Config {
 
  private int nbMaxUsers;
 
  public int getNbMaxUsers() {
    return nbMaxUsers;
  }
 
  public void setNbMaxUsers(int nbMaxUsers) {
    this.nbMaxUsers = nbMaxUsers;
  }
 
}

Esta classe define apenas um campo, nbMaxUsers, com os seus métodos get e set. Se acompanhou até aqui, quando a aplicação web é iniciada, é instanciada uma instância da classe [Config] com o valor 10 para o seu campo nbMaxUsers. Definimos apenas um único campo, mas isso é suficiente para a nossa demonstração. Queremos mostrar que este campo, que poderia representar um número máximo de utilizadores, é um dado no âmbito da Aplicação acessível a todas as ações. É isso que iremos demonstrar com a ação [Action1].

17.2.2. O ficheiro [struts.xml]

O ficheiro de configuração principal do Struts é o seguinte:


<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE struts PUBLIC
    "-//Apache Software Foundation//DTD Struts Configuration 2.0//EN"
    "http://struts.apache.org/dtds/struts-2.0.dtd">
 
<struts>
  <!-- internationalization -->
  <constant name="struts.custom.i18n.resources" value="messages" />
  <!-- spring integration -->
  <constant name="struts.objectFactory.spring.autoWire" value="name" />
 
  <include file="example/example.xml"/>
 
  <package name="default" namespace="/" extends="struts-default">
    <default-action-ref name="index" />
    <action name="index">
      <result type="redirectAction">
        <param name="actionName">Action1</param>
        <param name="namespace">/example</param>
      </result>
    </action>
  </package>
 
</struts>

Apenas a linha 10 difere das versões anteriores deste ficheiro. Ela define uma constante Struts. Os objetos instanciados pelo Spring podem ser injetados noutros objetos. Esta é a base do Spring. Aqui, uma referência ao objeto [Config] criado pelo Spring pode ser injetada em objetos Struts. É aqui que entra o plugin Spring para Struts 2, garantindo a integração do Spring com o Struts.

A linha 10 indica que a injeção de objetos Spring num objeto Struts ocorrerá automaticamente (autowire) por nome (valor). Voltemos ao ficheiro de configuração [applicationContext.xml]:


<!-- configuration des beans de portée application -->
  <bean id="config" class="example.Config" >
    <property name="nbMaxUsers" value="10"/>
  </bean>

O que o Struts 2 designa como nome do bean é, na verdade, o ID aqui. Assim, neste caso, o bean [Config] instanciado é denominado config.

A ação [Action1.java] é a seguinte:


package example;
 
import com.opensymphony.xwork2.ActionSupport;
...
public class Action1 extends ActionSupport implements SessionAware, RequestAware, ParameterAware {
 
  // constructor without parameters
  public Action1() {
  }
  
  // Session, Request, Parametres
  private Map<String, Object> session;
  private Map<String, Object> request;
  private Map<String, String[]> parameters;
  private Config config;
 
  @Override
  public String execute() {
  ....
    // request
    request.put("nbMaxUsers", config.getNbMaxUsers());
 
    // display page JSP
    return SUCCESS;
  }
...
  public Config getConfig() {
    return config;
  }
 
  public void setConfig(Config config) {
    this.config = config;
  }
}

A ação [Action1] é idêntica à da aplicação anterior, com as seguintes pequenas diferenças:

  • linha 15: foi adicionado um campo de configuração. Como tem o nome de um bean instanciado pelo Spring, este campo receberá automaticamente uma referência a esse objeto. Assim, a ação tem acesso à configuração da aplicação ou, de forma mais geral, aos dados no âmbito da aplicação.
  • linha 31: O Spring irá instanciar o campo config através do seu método set. Por isso, este deve estar presente.
  • Linha 21: A ação acede a informações no âmbito da aplicação. Incluímos o número máximo de utilizadores, nbMaxUsers, na solicitação para que a vista [Context.jsp] possa exibi-lo.

17.2.3. O ficheiro [example.xml]

O ficheiro de configuração secundário do Struts é idêntico ao da versão anterior:


<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE struts PUBLIC
        "-//Apache Software Foundation//DTD Struts Configuration 2.0//EN"
        "http://struts.apache.org/dtds/struts-2.0.dtd">
 
<struts>
  <package name="example" namespace="/example" extends="struts-default">
    <action name="Action1" class="example.Action1">
      <result name="success">/example/Context.jsp</result>
    </action>
  </package>
</struts>

17.3. A vista [Context.jsp]

A vista [Context.jsp] é a seguinte:


<%@ page contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
<%@ taglib prefix="s" uri="/struts-tags" %>
<html>
  <head>
    <title><s:text name="Context.titre"/></title>
    <s:head/>
  </head>
 
  <body background="<s:url value="/ressources/standard.jpg"/>">
    <h2><s:text name="Context.message"/></h2>
    <h3><s:text name="Context.parameters"/></h3>
    nom : <s:property value="#parameters['nom']"/><br/>
    age : <s:property value="#parameters['age']"/>
    <h3><s:text name="Context.session"/></h3>
    compteur : <s:property value="#session['compteur']"/>
    <h3><s:text name="Context.request"/></h3>
    nbMaxUsers : <s:property value="#request['nbMaxUsers']"/>
  </body>
</html>

17.4. Testes

Um exemplo de execução é o seguinte:

Image