Srping + JPA + Hibernate 多数据源配置

本文介绍如何在Spring环境中配置多个数据源,并通过JPA和Hibernate进行数据库操作。重点包括Spring配置文件设置、数据源定义、实体管理工厂配置及事务管理。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

项目中有配置多数据源的需求,查了一些资料最终实现,主要参考资料:

http://www.cnblogs.com/linjiqin/archive/2011/02/12/1952904.html

https://my.oschina.net/frankly/blog/27702


我用的是spring3 + jpa2 + hibernate4,配置方式与资料中略有不同,但是都大同小异,下面说一下过程和几个我认为的关键点:


1.通常jpa项目会在classpath/META-INF目录下有个persistence.xml配置文件,我的项目没有这个文件,内容直接配到spring的配置文件里了,以下是spring配置文件内容:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
	xmlns:context="http://www.springframework.org/schema/context"
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop"
	xmlns:tx="http://www.springframework.org/schema/tx"
	xmlns:task="http://www.springframework.org/schema/task"
	xsi:schemaLocation="
    http://www.springframework.org/schema/beans 
    http://www.springframework.org/schema/beans/spring-beans.xsd
    http://www.springframework.org/schema/tx
    http://www.springframework.org/schema/tx/spring-tx.xsd
    http://www.springframework.org/schema/context
    http://www.springframework.org/schema/context/spring-context.xsd
    http://www.springframework.org/schema/aop
    http://www.springframework.org/schema/aop/spring-aop-2.0.xsd">
	
	<!-- 使用annotation @Repository,@Service自动注册bean, 并保证@Required、@Autowired的属性被注入的包范围 -->
    <context:component-scan base-package="com.cf" />
	<!-- 数据源1  -->
    <bean id="dataSource" class="com.zaxxer.hikari.HikariDataSource" destroy-method="close">
        <property name="driverClassName" value="com.mysql.jdbc.Driver"/>
        <!-- Connection Info   coding -->
        <property name="jdbcUrl" value="jdbc:mysql://192.168.199.162:3306/cf-db?autoreconect=true&useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull"/>
        <property name="username" value="root"/>
        <property name="password" value="root"/>
        <property name="maximumPoolSize" value="10"/>
        <property name="maxLifetime" value="28700000" />
        <property name="poolName" value="cf-pool0"/>
    </bean>


	<!-- Jpa Entity Manager 配置 -->
    <bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
        <property name="dataSource" ref="dataSource"/>
        <property name="jpaVendorAdapter" ref="hibernateJpaVendorAdapter"/>
        <property name="packagesToScan">  <!-- 扫描映射类的包 -->
            <list>
                <value>com.cf.bus.core.*.domain</value>
                <value>com.cf.bus.additional.*.domain</value>
            </list>
        </property>
        <property name="jpaProperties">  <!-- 如果有persistence.xml可以配置在那里  -->
            <props>
                <prop key="hibernate.show_sql">false</prop>
                <prop key="hibernate.format_sql">false</prop>
                <prop key="hibernate.dialect">org.hibernate.dialect.MySQL5InnoDBDialect</prop>
                <prop key="hibernate.ejb.naming_strategy">org.hibernate.cfg.ImprovedNamingStrategy</prop>
              	<prop key="hibernate.cache.region.factory_class">org.hibernate.cache.ehcache.EhCacheRegionFactory</prop>

            </props>
        </property>
        <property name ="persistenceUnitName" value="unit1"/> <!-- 这个在java代码中注入的时候用到 ,如果有 persistence.xml也可配在那里-->
    </bean>

	<bean id="hibernateJpaVendorAdapter" class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">
    </bean>
    
	<bean id="txManager" class="org.springframework.orm.jpa.JpaTransactionManager">
		<property name="entityManagerFactory" ref="entityManagerFactory" />
	</bean>
	
	<tx:advice id="txAdvice" transaction-manager="txManager">
		<tx:attributes>
			<tx:method name="add*" propagation="REQUIRED" />
			<tx:method name="delete*" propagation="REQUIRED" />
			<tx:method name="update*" propagation="REQUIRED" />
			<tx:method name="*" propagation="REQUIRED" />
		</tx:attributes>
	</tx:advice>

	<aop:config>
		<aop:pointcut id="allManagerMethod"
			expression="execution(* com..service.*.*(..))" />
		<aop:advisor advice-ref="txAdvice" pointcut-ref="allManagerMethod" />
	</aop:config>












	<!--  数据源2   -->
    <bean id="dataSource2" class="com.zaxxer.hikari.HikariDataSource" destroy-method="close">
        <property name="driverClassName" value="com.mysql.jdbc.Driver"/>
        <!-- Connection Info   coding -->
        <property name="jdbcUrl" value="jdbc:mysql://192.168.199.162:3306/cf-rs?autoreconect=true&useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull"/>
        <property name="username" value="root"/>
        <property name="password" value="root"/>
        <property name="maximumPoolSize" value="10"/>
        <property name="maxLifetime" value="28700000" />
        <property name="poolName" value="cf-pool2"/>
    </bean>

    <!-- Jpa Entity Manager 配置 -->
    <bean id="entityManagerFactory2" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
        <property name="dataSource" ref="dataSource2"/>
        <property name="jpaVendorAdapter" ref="hibernateJpaVendorAdapter2"/>
        <property name="packagesToScan">  <!--  这里要扫描另外的包,跟第一个数据源的要区别开  -->
            <list>
                <value>com.cf.bus.core.rs.template.domain</value>
            </list>
        </property>
        <property name="jpaProperties">   <!-- 如果有persistence.xml可以配置在那里  -->
            <props>
                <prop key="hibernate.show_sql">false</prop>
                <prop key="hibernate.format_sql">false</prop>
                <prop key="hibernate.dialect">org.hibernate.dialect.MySQL5InnoDBDialect</prop>
                <prop key="hibernate.ejb.naming_strategy">org.hibernate.cfg.ImprovedNamingStrategy</prop>
                <prop key="hibernate.cache.region.factory_class">org.hibernate.cache.ehcache.EhCacheRegionFactory</prop>

            </props>
        </property>
        <property name ="persistenceUnitName" value="unit2"/>   <!-- 这里是第二个数据源为entityManager注入时用的 ,如果有 persistence.xml也可配在那里  -->
    </bean>

    <bean id="hibernateJpaVendorAdapter2" class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">
    </bean>
	<!-- 由于第二个数据源不需要对数据库做改的操作,所以没有配置事务  -->



</beans>

两个数据源中有很多不同的地方,由于代码里不能直接标颜色,所以只能靠自己观察了,仔细看,spring配置完成,接下来是java代码。


2.在java中注入

a.数据源1的注入

package com.cf.bus.common.dao.jpa;

import org.apache.log4j.Logger;
import org.hibernate.SQLQuery;
import org.hibernate.transform.Transformers;

import javax.annotation.Resource;
import javax.persistence.*;
import javax.persistence.criteria.*;
import java.io.Serializable;
import java.lang.reflect.Field;
import java.util.*;

public class BaseDaoImpl<T> implements BaseDao<T> {

	private static final Logger logger = Logger.getLogger(BaseDaoImpl.class);

	@PersistenceContext(unitName = "unit1")
	protected EntityManager em;
	@Resource(name = "entityManagerFactory")
	protected EntityManagerFactory emf;
	
	.....   //业务代码
}

b.数据源2的注入

package com.cf.bus.core.rs.base.dao.jpa;


import org.apache.log4j.Logger;
import org.hibernate.SQLQuery;
import org.hibernate.transform.Transformers;

import javax.annotation.Resource;
import javax.persistence.*;
import javax.persistence.criteria.*;
import java.io.Serializable;
import java.lang.reflect.Field;
import java.util.*;

public class BaseDaoImpl<T> implements BaseDao<T> {

	private static final Logger logger = Logger.getLogger(BaseDaoImpl.class);

	@PersistenceContext(unitName = "unit2")
	protected EntityManager em;
	@Resource(name = "entityManagerFactory2")
	protected EntityManagerFactory emf;
	......   //业务代码
}


以上两个类所在包不同,注入时引用的数据源对象也不同,注意看。以后的dao都继承这两个不同的父类,就能访问不同的数据库了,亲测可用。

由于众所周知的原因,隐掉了一些包引用和修改了一些包名,请不要在意这些细节,并不影响本文讨论的重点。


如果有问题欢迎一起交流!!


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值