Hola:
Estoy desarrollando una aplicación que utiliza Struts 2 para la capa de vista/presentación, Spring en la lógica de negocio e Hibernate como motor de persistencia.
La arquitectura utilizada es muy robusta y estoy intentando evolucionarla, por lo que estoy cambiando Struts 1.3.x por Struts 2.x y además añado DWR para utilizar la tecnología AJAX.
Estoy realizando una gestión de usuarios y en el formulario de creación utilizo AJAX para seleccionar las localidades de una provincia.
El problema es que cuando pulso el botón para mostrar el formulario de creación, tarda cerca de 13 segundos. Todo esto es debido a los "lazy" que se colocan en los "hbm" de hibernate.
Me gustaría saber como puedo evitar que tarde tanto en hacer esto, es como si cargara de la BD las más de 9000 localidades que hay para que cuando se seleccione la provincia sea instantaneo.
A continuación coloco el código de los archivos involucrados.
datosUsuario.jsp (formulario de creación de usuario)
<%@ page contentType="text/html; charset=UTF-8"%>
<%@ taglib prefix="s" uri="/struts-tags"%>
<p>
${accion} Usuario
</p>
<s:form action="doCrearUsuario" method="post">
<s:textfield label="Nombre" name="usuario.nombre" ></s:textfield>
<s:textfield label="Apellido1" name="usuario.apellido1" ></s:textfield>
<s:textfield label="Apellido2" name="usuario.apellido2" ></s:textfield>
<s:textfield label="NIF" name="usuario.nif" ></s:textfield>
<s:textfield label="Dirección" name="usuario.direccion" ></s:textfield>
<s:textfield label="Email" name="usuario.email" ></s:textfield>
<s:textfield label="Teléfono" name="usuario.telefono" ></s:textfield>
<s:textfield label="Cód. Postal" name="usuario.codigoPostal" ></s:textfield>
<s:textfield label="Login" name="usuario.login" ></s:textfield>
<s:password label="Password" name="usuario.password" ></s:password>
<s:if test="${usuario.activo=='1' || accion=='Crear' }">
<s:set name="activacion" value="%{'true'}" />
</s:if>
<s:elseif test="${usuario.activo=='0'}">
<s:set name="activacion" value="%{'fasle'}" />
</s:elseif>
<s:checkbox label="Activo" name="usuario.activo"
value="${activacion}"></s:checkbox>
<s:select label="Perfil" listValue="nombre" listKey="idPerfil"
list="listaPerfiles" />
<s:select id="provincia" label="Provincia" listValue="nombre" listKey="idProvincia"
list="listaProvincias" name="${prov}"
onchange="Ajax.getLocByProv(objectEval($('provinci a').value), rellenaLocalidades);"/>
<s:select id="localidad" label="Localidad" listValue="nombre" listKey="idLocalidad"
list="listaLocalidades" name="${usuario.localidades.idLocalidad}" />
<s:submit value="Aceptar" />
<s:reset value="Limpiar" />
<s:submit action="cancelUserForm" value="Cancelar"
onclick="form.onsubmit=null" />
</s:form>
AjaxServiceImpl.java (clase con el método para seleccionar las localidades de una provincia)
public class AjaxServiceImpl implements AjaxService{
private static final Logger log = Logger.getLogger(AjaxServiceImpl.class);
private ProvinciasDAO provinciasDao;
private LocalidadesDAO localidadesDao;
public LocalidadesDAO getLocalidadesDao() {
return localidadesDao;
}
public void setLocalidadesDao(LocalidadesDAO localidadesDao) {
this.localidadesDao = localidadesDao;
}
public ProvinciasDAO getProvinciasDao() {
return provinciasDao;
}
public void setProvinciasDao(ProvinciasDAO provinciasDao) {
this.provinciasDao = provinciasDao;
}
public List getLocByProv(String idProvincia)
throws ServiceException {
log.debug(Constantes.COMIENZA_METODO);
List listLocalidades = new ArrayList();
ProvinciasService provService = (ProvinciasService) Factory
.getBean(CtesPortales.PROVINCIAS_SERVICE_BEAN_NAME );
Provincias prov=provService.findById(Integer.parseInt(idProvi ncia));
log.debug("provincia: "+prov.getIdProvincia()+" y su nombre: "+prov.getNombre());
LocalidadService localidadService = (LocalidadService) Factory
.getBean(CtesPortales.LOCALIDADES_SERVICE_BEAN_NAM E);
listLocalidades=localidadService.findByProvincia(
prov.getIdProvincia().toString());
log.debug("numero de localidades de la provincia: "+idProvincia+": "+
listLocalidades.size());
log.debug(Constantes.TERMINA_METODO);
return listLocalidades;
}
}
Localidades.hbm.xml (archivo de mapeo de hibernate de localidades, los lazy se los he puesto yo porque sino daba errores)
<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<!--
Mapping file autogenerated by MyEclipse Persistence Tools
-->
<hibernate-mapping>
<class name="es.myapp.portales.business.objects.Localidad es" table="localidades" schema="public">
<id name="idLocalidad" type="java.lang.Long">
<column name="id_localidad" />
<generator class="assigned" />
</id>
<many-to-one name="provincias" class="es.myapp.portales.business.objects.Provinci as"
fetch="select" lazy="false">
<column name="provincia" />
</many-to-one>
<property name="nombre" type="java.lang.String">
<column name="nombre" length="200" not-null="true" />
</property>
<set name="usuarioses" inverse="true" lazy="false" >
<key>
<column name="localidad" />
</key>
<one-to-many class="es.myapp.portales.business.objects.Usuarios " />
</set>
</class>
</hibernate-mapping>
Provincias.hbm.xml (archivo de mapeo de hibernate de provincias, los lazy se los he puesto yo porque sino daba errores)
<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<!--
Mapping file autogenerated by MyEclipse Persistence Tools
-->
<hibernate-mapping>
<class name="es.myapp.portales.business.objects.Provincia s" table="provincias" schema="public">
<id name="idProvincia" type="java.lang.Integer">
<column name="id_provincia" />
<generator class="assigned" />
</id>
<property name="nombre" type="java.lang.String">
<column name="nombre" length="150" not-null="true" />
</property>
<set name="localidadeses" inverse="true" lazy="false">
<key>
<column name="provincia" />
</key>
<one-to-many class="es.myapp.portales.business.objects.Localida des" />
</set>
</class>
</hibernate-mapping>
Funciones Javascript
function objectEval(text){
text = text.replace(/\n/g, ' ');
text = text.replace(/\r/g, ' ');
if (text.match(/^\s*\{.*\}\s*$/))
{
text = '[' + text + '][0]';
}
return eval(text);
}
function rellenaLocalidades(data){
DWRUtil.removeAllOptions("localidad", data);
DWRUtil.addOptions("localidad", data, "idLocalidad", "nombre");
}
CrearUsuarioFront.java (Action que muestra el formulario de creación de usuario)
public class CrearUsuarioFrontAction extends ActionSupport implements
ServletRequestAware {
/**
* Objeto log de trazas
*/
private static final Logger log = Logger
.getLogger(CrearUsuarioFrontAction.class);
private HttpServletRequest servletRequest;
private List listaPerfiles = new ArrayList();
private List listaProvincias = new ArrayList();
private List listaLocalidades = new ArrayList();
private String prov;
public String getProv() {
return prov;
}
public void setProv(String prov) {
this.prov = prov;
}
public List getListaLocalidades() {
return listaLocalidades;
}
public void setListaLocalidades(List listaLocalidades) {
this.listaLocalidades = listaLocalidades;
}
public List getListaProvincias() {
return listaProvincias;
}
public void setListaProvincias(List listaProvincias) {
this.listaProvincias = listaProvincias;
}
public List getListaPerfiles() {
return listaPerfiles;
}
public void setListaPerfiles(List listaPerfiles) {
this.listaPerfiles = listaPerfiles;
}
public HttpServletRequest getServletRequest() {
return servletRequest;
}
public void setServletRequest(HttpServletRequest servletRequest) {
this.servletRequest = servletRequest;
}
public String execute() throws ServiceException {
log.debug(Constantes.COMIENZA_METODO);
servletRequest.setAttribute(Constantes.PARAM_ACCIO N,
Constantes.ACCION_CREAR);
PerfilesService perfilService = (PerfilesService) Factory
.getBean(CtesPortales.PERFILES_SERVICE_BEAN_NAME);
listaPerfiles = perfilService.findAll();
log.debug("numero de perfiles: " + listaPerfiles.size());
ProvinciasService provinciaService = (ProvinciasService) Factory
.getBean(CtesPortales.PROVINCIAS_SERVICE_BEAN_NAME );
listaProvincias = provinciaService.findAll();
log.debug("numero de provincias: " + listaProvincias.size());
LocalidadService localidadService = (LocalidadService) Factory
.getBean(CtesPortales.LOCALIDADES_SERVICE_BEAN_NAM E);
listaLocalidades = localidadService.findByProvincia(
Config.get(Constantes.CFG_PROVINCIA_DEFECTO));
log.debug("numero de localidades: " + listaLocalidades.size());
prov=Config.get(Constantes.CFG_PROVINCIA_DEFECTO);
log.debug(Constantes.TERMINA_METODO);
return SUCCESS;
}
}
Muchas gracias de antemano