Spring Boot 配置 : JpaBaseConfiguration

本文围绕Spring Boot自动配置展开,介绍了提供配置的抽象基类,针对不同实现有具体实现类。不同提供商实现的子类主要包装信息,配置任务主要落在特定组件上,还说明了其确保的定义内容,最后给出源代码版本和参考文章。

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

概述

JpaBaseConfigurationSpring Boot提供的对JPA进行配置的抽象基类。针对不同的JPA实现,会有不同的具体实现类。比如Spring Boot内置支持基于HibernateJPA,所以它提供了相应的实现HibernateJpaConfiguration。而如果开发人员想使用其他JPA实现,比如EclipseLink或者OpenJPA,就要继承JpaBaseConfiguration提供相应的具体实现。

虽然针对不同的JPA提供商实现需要有不同的JpaBaseConfiguration实现子类,但这些子类主要是将JPA提供商实现信息包装成统一的格式供JpaBaseConfiguration使用,而Spring Data JPA主要工作组件的配置任务,还是主要落在JpaBaseConfiguration身上。

JpaBaseConfiguration主要是确保以下bean的定义 :

  • PlatformTransactionManager transactionManager

    仅在该类型的bean未被定义的情况下才定义

  • LocalContainerEntityManagerFactoryBean entityManagerFactory

    仅在该类型的bean或者bean EntityManagerFactory未被定义的情况下才定义

  • JpaVendorAdapter jpaVendorAdapter

    仅在该类型的bean未被定义的情况下才定义;
    bean LocalContainerEntityManagerFactoryBean entityManagerFactory定义的基础。

  • EntityManagerFactoryBuilder entityManagerFactoryBuilder

    仅在该类型的bean未被定义的情况下才定义;
    bean LocalContainerEntityManagerFactoryBean entityManagerFactory定义的基础。

可以这么理解,一个JpaBaseConfiguration(当然具体类是它的某个实现子类)应用之后,我们就可以注入bean EntityManagerFactory了。

源代码

源代码版本 : spring-boot-autoconfigure-2.1.3.RELEASE

package org.springframework.boot.autoconfigure.orm.jpa;

// 省略 imports


@Configuration
// 确保前缀为 spring.jpa 的属性被加载到 bean JpaProperties
@EnableConfigurationProperties(JpaProperties.class)
// 导入 DataSourceInitializedPublisher.Registrar, 用于向容器注册 bean DataSourceInitializedPublisher,
// DataSourceInitializedPublisher 是一个 BeanPostProcessor
@Import(DataSourceInitializedPublisher.Registrar.class)
public abstract class JpaBaseConfiguration implements BeanFactoryAware {

	private final DataSource dataSource;

	private final JpaProperties properties;

	private final JtaTransactionManager jtaTransactionManager;

	private final TransactionManagerCustomizers transactionManagerCustomizers;

	private ConfigurableListableBeanFactory beanFactory;

    // dataSource 在其他地方定义 : 
    // jtaTransactionManager 在其他地方定义 : 
    // transactionManagerCustomizers 在其他地方定义 : 
    // JpaProperties properties 来源参考本类注解
	protected JpaBaseConfiguration(DataSource dataSource, JpaProperties properties,
			ObjectProvider<JtaTransactionManager> jtaTransactionManager,
			ObjectProvider<TransactionManagerCustomizers> transactionManagerCustomizers) {
		this.dataSource = dataSource;
		this.properties = properties;
		this.jtaTransactionManager = jtaTransactionManager.getIfAvailable();
		this.transactionManagerCustomizers = transactionManagerCustomizers
				.getIfAvailable();
	}

    // 定义 bean 事务处理器 PlatformTransactionManager transactionManager,
    // 缺省情况下的实现类使用 JpaTransactionManager
	@Bean
    // 仅在该 bean 未被定义的情况下才定义
	@ConditionalOnMissingBean
	public PlatformTransactionManager transactionManager() {
		JpaTransactionManager transactionManager = new JpaTransactionManager();
		if (this.transactionManagerCustomizers != null) {
			this.transactionManagerCustomizers.customize(transactionManager);
		}
		return transactionManager;
	}

    // 定义 bean JpaVendorAdapter jpaVendorAdapter, JPA供应商实现适配器
	@Bean
    // 仅在该 bean 未被定义的情况下才定义
	@ConditionalOnMissingBean   
	public JpaVendorAdapter jpaVendorAdapter() {
		AbstractJpaVendorAdapter adapter = createJpaVendorAdapter();
		adapter.setShowSql(this.properties.isShowSql());
		adapter.setDatabase(this.properties.determineDatabase(this.dataSource));
		adapter.setDatabasePlatform(this.properties.getDatabasePlatform());
		adapter.setGenerateDdl(this.properties.isGenerateDdl());
		return adapter;
	}

    // 定义 bean EntityManagerFactoryBuilder entityManagerFactoryBuilder
    // 该 bean 会使用其他地方定义的 JpaVendorAdapter, PersistenceUnitManager
    // 构建一个 EntityManagerFactoryBuilder, 这是一个 EntityManagerFactory 构建器
    // 此 bean 可以理解成底层组件提供者 JpaVendorAdapter, PersistenceUnitManager 跟
    // 上层 LocalContainerEntityManagerFactoryBean 使用者之间的一个纽带
	@Bean
    // 仅在该 bean 未被定义的情况下才定义
	@ConditionalOnMissingBean
	public EntityManagerFactoryBuilder entityManagerFactoryBuilder(
			JpaVendorAdapter jpaVendorAdapter,
			ObjectProvider<PersistenceUnitManager> persistenceUnitManager,
			ObjectProvider<EntityManagerFactoryBuilderCustomizer> customizers) {
		EntityManagerFactoryBuilder builder = new EntityManagerFactoryBuilder(
				jpaVendorAdapter, this.properties.getProperties(),
				persistenceUnitManager.getIfAvailable());
		customizers.orderedStream()
				.forEach((customizer) -> customizer.customize(builder));
		return builder;
	}

    // 定义 bean LocalContainerEntityManagerFactoryBean entityManagerFactory
    // 该 bean 定义方法使用上面定义的 EntityManagerFactoryBuilder bean 执行其构建过程,
    // 产出物为一个 LocalContainerEntityManagerFactoryBean , 这其实是一个
    // EntityManagerFactory, EntityManager 的工厂组件 bean
	@Bean
	@Primary
    // 仅在同类型 bean 和 类型为 EntityManagerFactory 的 bean 未被定义的情况下才定义
	@ConditionalOnMissingBean({ LocalContainerEntityManagerFactoryBean.class,
			EntityManagerFactory.class })
	public LocalContainerEntityManagerFactoryBean entityManagerFactory(
			EntityManagerFactoryBuilder factoryBuilder) {
		Map<String, Object> vendorProperties = getVendorProperties();
		customizeVendorProperties(vendorProperties);
		return factoryBuilder.dataSource(this.dataSource).packages(getPackagesToScan())
				.properties(vendorProperties).mappingResources(getMappingResources())
				.jta(isJta()).build();
	}

	protected abstract AbstractJpaVendorAdapter createJpaVendorAdapter();

	protected abstract Map<String, Object> getVendorProperties();

	/**
	 * Customize vendor properties before they are used. Allows for post processing (for
	 * example to configure JTA specific settings).
	 * @param vendorProperties the vendor properties to customize
	 */
	protected void customizeVendorProperties(Map<String, Object> vendorProperties) {
	}

	protected String[] getPackagesToScan() {
		List<String> packages = EntityScanPackages.get(this.beanFactory)
				.getPackageNames();
		if (packages.isEmpty() && AutoConfigurationPackages.has(this.beanFactory)) {
			packages = AutoConfigurationPackages.get(this.beanFactory);
		}
		return StringUtils.toStringArray(packages);
	}

	private String[] getMappingResources() {
		List<String> mappingResources = this.properties.getMappingResources();
		return (!ObjectUtils.isEmpty(mappingResources)
				? StringUtils.toStringArray(mappingResources) : null);
	}

	/**
	 * Return the JTA transaction manager.
	 * @return the transaction manager or null
	 */
	protected JtaTransactionManager getJtaTransactionManager() {
		return this.jtaTransactionManager;
	}

	/**
	 * Returns if a JTA PlatformTransactionManager is being used.
	 * @return if a JTA transaction manager is being used
	 */
	protected final boolean isJta() {
		return (this.jtaTransactionManager != null);
	}

	/**
	 * Return the JpaProperties.
	 * @return the properties
	 */
	protected final JpaProperties getProperties() {
		return this.properties;
	}

	/**
	 * Return the DataSource.
	 * @return the data source
	 */
	protected final DataSource getDataSource() {
		return this.dataSource;
	}

	@Override
	public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
		this.beanFactory = (ConfigurableListableBeanFactory) beanFactory;
	}


    // 内部配置类, 对 Open Session In View 模式的支持
	@Configuration
	@ConditionalOnWebApplication(type = Type.SERVLET)
	@ConditionalOnClass(WebMvcConfigurer.class)
	@ConditionalOnMissingBean({ OpenEntityManagerInViewInterceptor.class,
			OpenEntityManagerInViewFilter.class })
	@ConditionalOnMissingFilterBean(OpenEntityManagerInViewFilter.class)
	@ConditionalOnProperty(prefix = "spring.jpa", name = "open-in-view", 
		havingValue = "true", matchIfMissing = true)
	protected static class JpaWebConfiguration {

		// Defined as a nested config to ensure WebMvcConfigurerAdapter is not read when
		// not on the classpath
		@Configuration
		protected static class JpaWebMvcConfiguration implements WebMvcConfigurer {

			private static final Log logger = LogFactory
					.getLog(JpaWebMvcConfiguration.class);

			private final JpaProperties jpaProperties;

			protected JpaWebMvcConfiguration(JpaProperties jpaProperties) {
				this.jpaProperties = jpaProperties;
			}

			@Bean
			public OpenEntityManagerInViewInterceptor openEntityManagerInViewInterceptor() {
				if (this.jpaProperties.getOpenInView() == null) {
					logger.warn("spring.jpa.open-in-view is enabled by default. "
							+ "Therefore, database queries may be performed during view "
							+ "rendering. Explicitly configure "
							+ "spring.jpa.open-in-view to disable this warning");
				}
				return new OpenEntityManagerInViewInterceptor();
			}

			@Override
			public void addInterceptors(InterceptorRegistry registry) {
				registry.addWebRequestInterceptor(openEntityManagerInViewInterceptor());
			}

		}

	}

}

参考文章

### 解决 Spring Boot 配置 JPA 出现 `jpaVendorAdapter` 错误的方法 当遇到 `JpaBaseConfiguration jpaVendorAdapter error processing condition` 类型的错误时,通常是因为应用程序未能正确初始化 JPA 或者缺少必要的依赖项。以下是详细的解决方案: #### 1. 检查 Maven/Gradle 中的依赖关系 确保项目中的构建文件包含了所有必需的 JPA 和数据库驱动程序依赖项。 对于 Maven 构建工具,在 pom.xml 文件中应包含以下依赖项[^1]: ```xml <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-jpa</artifactId> </dependency> <!-- MySQL Driver --> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <scope>runtime</scope> </dependency> ``` 对于 Gradle 用户,则应在 build.gradle 添加相应的依赖声明: ```groovy dependencies { implementation 'org.springframework.boot:spring-boot-starter-data-jpa' runtimeOnly 'mysql:mysql-connector-java' // 如果使用MySQL作为数据库 } ``` #### 2. 数据源配置属性设置 确认 application.properties 或 application.yml 文件中有正确的数据源连接字符串和其他必要参数。例如: application.properties: ```properties spring.datasource.url=jdbc:mysql://localhost:3306/mydb?useSSL=false&serverTimezone=UTC spring.datasource.username=root spring.datasource.password=password spring.jpa.hibernate.ddl-auto=update ``` application.yml: ```yaml spring: datasource: url: jdbc:mysql://localhost:3306/mydb?useSSL=false&serverTimezone=UTC username: root password: password jpa: hibernate: ddl-auto: update ``` #### 3. 启用实体扫描路径 如果应用内存在多个包结构下的实体类,需指定@EntityScan 注解来告知框架哪些位置下有持久化对象定义。 ```java @SpringBootApplication @EntityScan(basePackages = {"com.example.model", "com.anotherpackage"}) public class MainApplication { public static void main(String[] args) { SpringApplication.run(MainApplication.class, args); } } ``` 通过以上措施可以有效解决大部分由 `jpaVendorAdapter` 引发的问题。然而具体原因还需依据日志信息进一步排查可能存在的其他潜在因素。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值