Skip to content

Latest commit

 

History

History
227 lines (176 loc) · 11.5 KB

README.md

File metadata and controls

227 lines (176 loc) · 11.5 KB

dgtic-jpa

Proyecto de la formación de JPA con tecnología estándar de la DGTIC

Puntos importantes para mejorar el rendimiento de una aplicación JPA

Getters en @ManagedBeans

Los getters en @ManagedBeans son llamados en repetidas ocasiones por el motor de JSF, por lo que poner cualquier tipo de consultas en estos es una mala práctica. Estas deberían realizarse en un @PostConstruct o en los métodos que ejecutan acciones a partir de botones o links (los atributos action) que solo son llamados una vez, cuando se realiza la acción.

Lógica de negocio en @ManagedBeans

Recordemos que el Contexto de Persistencia tiene una vida que dura lo mismo que la transacción, que a su vez empieza y termina con cada método de negocio de un EJB. Esto implica que si realizamos lógica de negocio dentro de un @ManagedBean, las consultas de base de datos que se realizan a un EJB, ya mapeadas a objetos java, se tienen que volver a consultar y mapear en una nueva llamada a un método distinto (o incluso el mismo) de un EJB. Y es al realizar las modificaciones fuera del EJB cuando se hace necesario crear un saveOrUpdate() (con un em.merge() dentro), para guardar los cambios. Si la lógica se realiza dentro del EJB estos cambios ya se hacen persistentes de forma automática.

Los EJB sirven para realizar la lógica de negocio, y los @ManagedBeans deberían usarse solamente para controlar el front y enlazarlo con la lógica de negocio de los EJB.

En esta misma aplicación podemos ver un ejemplo de lo mencionado en este punto. En el front tenemos dos botones, que sirven para lanzar do smétodos dentro del @ManagedBean Index.java:

@ManagedBean
public class Index {

    @EJB(name = PostsManager.JNDI_NAME)
    PostManagerLocalFacade postsManager;
    
    // . . .
    
    public void testPersistenceContextFromEJB() {
        postsManager.testPCOK(posts.get(0).getId());
    }

    public void testPersistenceContextFromManagedBean() {
        Long postId = posts.get(0).getId();
        Post post;
        for (int i = 0; i < 100; i++) {
            post = postsManager.find(postId);
            post.setContent("New content " + RANDOM.nextInt(100));
        }
    }
@Stateless
public class PostsManager implements PostManagerLocalFacade {
    
    // . . .
    
    public void testPCOK(Long postId) {
        Post post;
        for (int i = 0; i < 100; i++) {
            post = em.find(Post.class, postId);
            post.setContent("New content " + RANDOM.nextInt(100));
        }
    }

La lógica consiste en un bucle que consulta el EntityManager 100 veces, en uno se hace desde el @ManagedBean y en el otro desde el EJB. Puede obversvarse en las consultas generadas que el primero lanza 100 consultas contra la base de datos mientras que el segundo solo 1. Esto da una idea de los problemas de rendimiento que puede generar realizar la lógica de negocio fuera de los EJB.

Podréis observar también que en testPersistenceContextFromManagedBean no se genera ninguna consulta UIPDATE, esto es debido a que los cambios se han hecho siempre fuera del contexto de persistencia, y es entonces cuando se hace necesario crear el clsico saveOrUpdate() para hacer el merge() de las entidades que hemos modificado (porque las hemos modificado fuera del contexto de persistencia). Sin embargo en testPersistenceContextFromEJB no hace falta ningún saveOrUpdate(), ni merge() ni nada. Todas las modificaciones se realizan dentro del contexto de persistencia y por lo tanto todo queda guargado cuando finaliza el método.

EAGER

Nunca deberemos meter un FetchType.EAGER en una relación @OneToMany o @ManyToMany. Esto suele hacerse para evitar un LazyIitializationException, que normalmente se da por dos motivos:

  1. Por un lado se está realizando lógica de negocio en los @ManagedBeans, y por lo tanto las propiedades LAZY ya no pueden ser recuperadas.
  2. No se han recuperado las propiedades LAZY mediante un join fetch, para poder utilizarlas fuera del contexto de persistencia. Fuera de este se pueden necesitar para el front, pero no para realizar lógica de negocio, que debería realizarse dentro del EJB.

Pero si lo solucionamos con un EAGER perdemos la capacidad de poder hacerlo LAZY en consultas explícitas, sin embargo esto no sucede al revés.

Y recordemos que esto tampoco significa que LAZY sea siempre la opción correcta en un @OneToOne o en un @ManyToOne. Todo dependerá de la complejidad de las relaciones.

Revisar las consultas SQL que se generan

La diferencia entre poner un fetch join y olvidárselo puede ser pasar de 4000 consultas a unas pocas. Revisar siempre las consultas que se generan no siempre es viable por restricciones de tiempo en el desarrollo, pero en la medida de lo posible deberan revisarse, al menos en los puntos que puedan resultar más conflictivos, y siempre debera ser el primer paso cuando se detectan problemas de rendimiento.

Para poder mostrar las consultas en el JBoss de la dgtic deberemos hacer lo siguiente (el archivo a modificar es jboss-eap-5.2/jboss-as/server/default/conf/jboss-log4j.xml):

  1. Añadir un nuevo Appender
    <!-- A time/date based rolling appender -->
   <appender name="JPA" class="org.jboss.logging.appender.DailyRollingFileAppender">
      <errorHandler class="org.jboss.logging.util.OnlyOnceErrorHandler"/>
      <param name="File" value="${jboss.server.log.dir}/jpa.log"/>
      <param name="Append" value="true"/>
      <!-- In AS 5.0.x the server log threshold was set by a system
           property. In 5.1 and later we are instead using the system
           property to set the priority on the root logger (see <root/> below)
      <param name="Threshold" value="${jboss.server.log.threshold}"/>
      -->
      
      <!-- Rollover at midnight each day -->
      <param name="DatePattern" value="'.'yyyy-MM-dd"/>

      <!-- Rollover at the top of each hour
      <param name="DatePattern" value="'.'yyyy-MM-dd-HH"/>
      -->

      <layout class="org.apache.log4j.PatternLayout">
         <!-- The default pattern: Date Priority [Category] (Thread) Message\n -->
         <param name="ConversionPattern" value="%d %m%n"/>

         <!-- The full pattern: Date MS Priority [Category] (Thread:NDC) Message\n
         <param name="ConversionPattern" value="%d %-5r %-5p [%c] (%t:%x) %m%n"/>
          -->
      </layout>
   </appender>
  1. Añadir las categorías que queramos:

Para que muestre solo las consultas:

   <category name="org.hibernate.SQL">
      <priority value="DEBUG"/>
      <appender-ref ref="JPA"/>
   </category>

Para que muestre también los parámetros:

   <category name="org.hibernate.type">
      <priority value="TRACE"/>
      <appender-ref ref="JPA"/>
   </category>

Vlad Mihalcea

Esta es la persona que os comenté que es Java Champion y experto en JPA/Hibernate.

Y este es su blog: https://vladmihalcea.com/.

Tiene artículos muy interesantes y didácticos.

Cómo hacer funcionar las relaciones @OneToOne(fetch = FetchType.LAZY, optional = true)

Cuando una relación @OneToONe se marca como optiona=true Hibernate no sabe si existe la relación sin hacer una consulta. Y necesita saberlo para saber si tiene que lanzar un NullPointerException (si no existe) o un LazyInitializationException.

Sin embargo, a través de una sería de tareas ant (en versiones antiguas de Hibernate) y plugins de Maven (en versiones no tan antiguas) hibernate puede introducir, en tiempo de compilación, modificaciones en el bytecode de las entidades para solventar este problema. Este proceso se llama Bytecode Enhancement.

En el caso concreto de la DGTIC, siendo un JBoss 5.2, con una versión de Hibernate 3.3, es necesaria la tarea Ant. Pero esta puede ser llamada a traves de un plugin de maven que lo único que hace es lanzar tareas Ant.

Para ello hemos añadido el siguiente fragmento en el módulo jpa-ejb:

    <plugin>
        <artifactId>maven-antrun-plugin</artifactId>
        <executions>
            <execution>
                <id>Instrument domain classes</id>
                <configuration>
                    <tasks>
                        <taskdef name="instrument"
                                 classname="org.hibernate.tool.instrument.javassist.InstrumentTask">
                            <classpath>
                                <path refid="maven.dependency.classpath"/>
                                <path refid="maven.plugin.classpath"/>
                            </classpath>
                        </taskdef>
                        <instrument verbose="true">
                            <fileset dir="${project.build.outputDirectory}">
                                <include name="**/entity/*.class"/>
                            </fileset>
                        </instrument>
                    </tasks>
                </configuration>
                <phase>process-classes</phase>
                <goals>
                    <goal>run</goal>
                </goals>
            </execution>
        </executions>
        <dependencies>
            <dependency>
                <groupId>org.hibernate</groupId>
                <artifactId>hibernate-core</artifactId>
                <version>3.3.2.GA</version>
            </dependency>
            <dependency>
                <groupId>javassist</groupId>
                <artifactId>javassist</artifactId>
                <version>3.12.0.GA</version>
            </dependency>
        </dependencies>
    </plugin>

Este proceso en ocasiones requiere modificar clases que usan las entidades fuera del módulo ejb, por lo que es posible que sea necesario añadir la siguiente dependencia en otros módulos:

    <dependency>
        <groupId>org.hibernate</groupId>
        <artifactId>hibernate-core</artifactId>
        <version>3.3.2.GA</version>
        <scope>provided</scope>
    </dependency>

Pero esto no es suficiente, hay que añadir una anotación en la clase padre de la relación @OneToOne:

    @LazyToOne(value = LazyToOneOption.NO_PROXY)

Esta anotación se encuentra en el artefacto de hibernate hibernate-annotations, cuya version en JBoss 5.2 es 3.4.0.GA. Para usar dicha anotación deberemos añadir la siguiente dependencia en el pom.xml del módulo ejb:

        <dependency>
            <groupId>org.hibernate</groupId>
            <artifactId>hibernate-annotations</artifactId>
            <version>3.4.0.GA</version>
            <scope>provided</scope>
        </dependency>

Y además deberemos asegurarnos de que la relación hija no tenga @MapsId (es decir, que tenga su propio id o que este no sea Composite Id)

Con esto debería funcionar con Java 6. El problema es que el estándar de la DGTIC establece que se use java 7, y cuando el JBoss se arranca con Java 7 el bytecode generado no está preparado para superar el Verificador de bytecode de java 7 y este lanza una excepción.

Afortunadamente existe un workarround, pero implica añadir una propiedad del sistema para el arranque del servidor:

    -XX:-UseSplitVerifier

Esto hará que el verificador de java funcione en modo java 6 cuando sea necesario.

En resumen, para que funcione un @OneToOne(fetch = FetchType.LAZY, optional = true) se han de realizar los siguientes pasos:

  1. Introducir el plugin de maven para realizar el Bytecode Enhancer en el modulo ejb
  2. Introducir las dependencias necesarias en otros módulos
  3. Añadir @LazyToOne(value = LazyToOneOption.NO_PROXY) en la relación del lado del padre
  4. Añadir a JAVA_OPTS la propiedad -XX:-UseSplitVerifier en el arranque del JBoss.