• Spring与JPA


    Java持久化API(Java Persistence API),即JPA

    Spring中使用JPA的第一步是要在Spring应用上下文中将实体管理器工厂(entity manager factory)按照Bean的形式进行配置。

    1.配置实体管理器工厂 

    基于JPA的应用程序使用EntityManagerFactory的实现类来获取EntityManager实例。JPA定义了两种类型的实体管理器: 

    (1)应用程序管理类型(Application-managed):当应用程序向实体管理器工厂直接请求实体管理器时,工厂会创建一个实体管理器。在这种模式下,程序要负责打开和关闭实体管理器并在事务中对其进行控制。 

    (2)容器管理类型(Container-managed):实体管理器由JavaEE创建和管理。应用程序根本不与实体管理器工厂打交道。相反,实体管理器直接通过注入或JNDI来获取。容器负责配置实体管理器工厂。这种类型的实体管理器最适合用于Java EE容器。 

    这两种实体管理器工厂分别由对应的Spring工厂Bean创建:

    (1)LocalEntityManagerFactoryBean生成应用程序管理类型的EntityManagerFactory

    (2)LocalContainerEntityManagerFactoryBean生成容器管理类型的EntityManagerFactory

    2、这里介绍:使用容器管理类型的JPA

    <bean id="entityManagerFactory"  class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
        <property name="dataSource" ref="dataSource"/>
        <property name="jpaVendorAdapter" ref="hibernateJpaVendorAdapter"/>
    </bean>

    当在容器中运行时,可以使用容器(在我们的场景下是Spring)提供的信息来生成EntityManagerFactory,

    可以将数据源信息配置在Spring应用上下文中,

    jpaVendorAdapter属性用于指明所使用的是哪一个厂商的JPA实现。

    <!-- 设定Hibernate对jpa的代理 -->
    <bean id="hibernateJpaVendorAdapter" class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">
        <property name="database" value="ORACLE"/>
        <property name="generateDdl" value="false"/>
    </bean>

    database:说明使用的哪种数据库

    项目中的applicationContext.xml,先记下

    <?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:context="http://www.springframework.org/schema/context"
        xmlns:aop="http://www.springframework.org/schema/aop"
        xmlns:tx="http://www.springframework.org/schema/tx"
        xmlns:jdbc="http://www.springframework.org/schema/jdbc"
        xmlns:jpa="http://www.springframework.org/schema/data/jpa"
        xsi:schemaLocation="http://www.springframework.org/schema/beans 
                            http://www.springframework.org/schema/beans/spring-beans-3.2.xsd
                            http://www.springframework.org/schema/context
                            http://www.springframework.org/schema/context/spring-context-3.2.xsd
                            http://www.springframework.org/schema/aop
                            http://www.springframework.org/schema/aop/spring-aop-3.2.xsd
                            http://www.springframework.org/schema/tx
                            http://www.springframework.org/schema/tx/spring-tx-3.2.xsd
                            http://www.springframework.org/schema/jdbc
                            http://www.springframework.org/schema/jdbc/spring-jdbc-3.2.xsd
                            http://www.springframework.org/schema/data/jpa
                            http://www.springframework.org/schema/data/jpa/spring-jpa-1.3.xsd"
        default-lazy-init="true">
        
        <!-- 加载配置文件:applicationContext.properties [BEGIN] -->
        <context:property-placeholder ignore-unresolvable="true" ignore-resource-not-found="true" 
            location="classpath*:/applicationContext.properties"/>
        <!-- 加载配置文件:applicationContext.properties [END] -->
        
        <!-- 开启注解扫描功能,同时过滤@Controller注解 [BEGIN] -->
        <context:component-scan base-package="com.jinhetech.yogurt">
            <context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
            <context:exclude-filter type="annotation" expression="org.springframework.web.bind.annotation.ControllerAdvice"/>
        </context:component-scan>
        <!-- 开启注解扫描功能,同时过滤@Controller注解 [BEGIN] -->
        
        <!-- 配置DataSource [BEGIN] -->
        <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
            <property name="driverClass" value="${jdbc.driver}"/>
            <property name="jdbcUrl" value="${jdbc.url}"/>
            <property name="user" value="${jdbc.username}"/>
            <property name="password" value="${jdbc.password}"/>
            <property name="maxPoolSize" value="${jdbc.pool.maxPoolSize}"/>
            <property name="minPoolSize" value="${jdbc.pool.minPoolSize}"/>
            <property name="initialPoolSize" value="${jdbc.pool.initialPoolSize}"/>
            <!-- 设置最大空闲时间,若在设置的时间内未使用,则连接被丢弃,若为0,则永不丢弃 -->
            <property name="maxIdleTime" value="${jdbc.pool.maxIdleTime}"/>
        </bean>
        <!-- 配置DataSource [END] -->
        
        <!-- 配置SessionFactory [Begin] -->
        <bean id="sessionFactory" class="org.springframework.orm.hibernate4.LocalSessionFactoryBean">
            <property name="dataSource" ref="dataSource"/>
            <property name="packagesToScan" value="com.jinhetech.bbs"/>
            <property name="namingStrategy">
                <bean class="org.hibernate.cfg.ImprovedNamingStrategy"/>
            </property>
            <property name="hibernateProperties">
                <props>
                    <prop key="hibernate.dialect">org.hibernate.dialect.OracleDialect</prop>
                    <prop key="hibernate.show_sql">false</prop>
                    <prop key="hibernate.format_sql">false</prop>
                </props>
            </property>
        </bean>
        <!-- 配置SessionFactory [End] -->
        
        <!-- 配置JdbcTemplate (若要使用JDBC形式的DAO,则需要注入该属性) [BEGIN] -->
        <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
            <property name="dataSource" ref="dataSource"/>
        </bean>
        <!-- 配置JdbcTemplate [END] -->
        
        <!-- JdbcTransactionManager [Begin] -->
        <bean id="jdbcTransactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
            <property name="dataSource" ref="dataSource"/>
            
        </bean>
        <!-- JdbcTransactionManager [End] -->
        
        <!-- 配置 Jpa Entity Manager [BEGIN] -->
        <bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
            <property name="dataSource" ref="dataSource"/>
            <property name="jpaVendorAdapter" ref="hibernateJpaVendorAdapter"/>
            <property name="packagesToScan" value="com.jinhetech.yogurt"/>
            <property name="jpaProperties">
                <props>
                    <prop key="hibernate.cache.region.factory_class">
                        org.hibernate.cache.ehcache.EhCacheRegionFactory
                    </prop>
                    <!-- 对实体属性名字的解析,可将如user_id和userId进行对应 -->
                    <prop key="hibernate.ejb.naming_strategy">
                        org.hibernate.cfg.ImprovedNamingStrategy
                    </prop>
                    <prop key="hibernate.show_sql">true</prop>
                    <prop key="hibernate.format_sql">true</prop>
                </props>
            </property>
        </bean>
        <!-- 设定Hibernate对jpa的代理 -->
        <bean id="hibernateJpaVendorAdapter" class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">
            <property name="database" value="ORACLE"/>
            <property name="generateDdl" value="false"/>
        </bean>
        
        <!-- Spring Data Jpa配置, 扫描base-package下所有继承于Repository<T,ID>的接口 -->
        <jpa:repositories base-package="com.jinhetech.yogurt" transaction-manager-ref="transactionManager"
            entity-manager-factory-ref="entityManagerFactory"/>
        
        <!-- 事务管理器配置, Jpa单数据源事务 -->
        <bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager">
            <property name="entityManagerFactory" ref="entityManagerFactory"/>
            <property name="jpaDialect" ref="hibernateJpaDialect"/>
        </bean>
        <bean id="hibernateJpaDialect" class="org.springframework.orm.jpa.vendor.HibernateJpaDialect"/>
        
        <!-- 使用annotation定义事务 [BEGIN] -->
        <!-- 设置proxy-target-class为true,则表明使用CGLIB动态代理机制 -->
        <tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true"/>
        <!-- 使用annotation定义事务 [END] -->
        
        <!-- hibernate validator -->
        <bean id="validator" class="org.springframework.validation.beanvalidation.LocalValidatorFactoryBean"/>
    </beans>

    事务管理器

    (1)JDBC事务

    <bean id="jdbcTransactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource"/>
        
    </bean>

    (2)Hibernate事务

    <bean id="sessionFactory" class="org.springframework.orm.hibernate4.LocalSessionFactoryBean">
        <property name="dataSource" ref="dataSource"/>
        <property name="packagesToScan" value="com.jinhetech.bbs"/>
        <property name="namingStrategy">
            <bean class="org.hibernate.cfg.ImprovedNamingStrategy"/>
        </property>
        <property name="hibernateProperties">
            <props>
                <prop key="hibernate.dialect">org.hibernate.dialect.OracleDialect</prop>
                <prop key="hibernate.show_sql">false</prop>
                <prop key="hibernate.format_sql">false</prop>
            </props>
        </property>
    </bean>

    (3)JPA事务

    <bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager">
        <property name="entityManagerFactory" ref="entityManagerFactory"/>
        <property name="jpaDialect" ref="hibernateJpaDialect"/>
    </bean>
    <bean id="hibernateJpaDialect" class="org.springframework.orm.jpa.vendor.HibernateJpaDialect"/>

     JpaTransactionManager还支持将事务应用于简单的JDBC操作之中,这些JDBC操作所使用的Datasource与entityManagerFactory所使用的DataSource必须是相同的,为了做到这一点,JpaTransactionManager必须装配一个JPADialect的实现

  • 相关阅读:
    本机mysql 5.7服务启动后停止,某些服务在未有其他应用程序使用时停止
    java数据结构 • 面向对象 • 异常 • 随机数·时间
    获取对象信息
    集合的运算律
    常用的逻辑符号
    未解决问题
    继承和多态
    访问权限
    类和实例
    第三方模块安装
  • 原文地址:https://www.cnblogs.com/Donnnnnn/p/6208204.html
Copyright © 2020-2023  润新知