【问题标题】:JPA: create EntityManagerFactory from propertiesJPA:从属性创建 EntityManagerFactory
【发布时间】:2013-08-01 07:58:25
【问题描述】:

我在 JAR 项目中使用 JPA 并使用 persistence.xml 来设置我的 EntityManager。

但由于persistence.xml 在构建后位于JAR 中,因此用户在之后更改设置非常复杂。所以我正在寻找一种解决方案,我可以通过在运行时加载的属性文件来配置我的连接。

我在网上看到了这个解决方案:

Map properties = new HashMap();

// Configure the internal EclipseLink connection pool
properties.put(JDBC_DRIVER, "oracle.jdbc.OracleDriver");
properties.put(JDBC_URL, "jdbc:oracle:thin:@localhost:1521:ORCL");
properties.put(JDBC_USER, "user-name");
properties.put(JDBC_PASSWORD, "password");

Persistence.createEntityManagerFactory("unit-name", properties);

这是我一直在寻找的解决方案,但我在这里遗漏了一件事:在我的 persistence.xml 中,我还在映射文件上声明了一个架构名称:

persistence.xml:

<persistence version="2.0" ...>
  <persistence-unit name="jpa" transaction-type="RESOURCE_LOCAL">
    <provider>org.eclipse.persistence.jpa.PersistenceProvider</provider>
    <class>...</class>
    <exclude-unlisted-classes>true</exclude-unlisted-classes>
    <properties>
      <property name="javax.persistence.jdbc.url" value="..."/>
      <property name="javax.persistence.jdbc.password" value="..."/>
      <property name="javax.persistence.jdbc.driver" value="oracle.jdbc.OracleDriver"/>
      <property name="javax.persistence.jdbc.user" value="..."/>
    </properties>
    <mapping-file>META-INF/orm.xml</mapping-file>
  </persistence-unit>
</persistence>

orm.xml:

<entity-mappings ...>
 <persistence-unit-metadata>
  <persistence-unit-defaults>
   <schema>SCHEMA_NAME</schema>
  </persistence-unit-defaults>
 </persistence-unit-metadata>
</entity-mappings>

所以我的问题基本上是:是否有一个属性可以用来在运行时设置架构,就像我使用其他属性一样?

或者有更好的解决方案吗?

提前致谢!

【问题讨论】:

    标签: java jpa eclipselink


    【解决方案1】:

    切换到 java 配置。然后您可以通过自动装配 Environment 轻松注入属性值

    这个例子非常基础。但总的来说,如果您知道如何进行 xml 配置,则可以将其直接映射到 Java 配置中

    contextConfig.java

    /**
     * Spring Context configuration.
     */
    @ComponentScan(basePackages = { "com.example" })
    @PropertySource({ "classpath:common.properties" })
    @Configuration
    @Import(JpaConfig.class)
    public class ContextConfig extends WebMvcConfigurerAdapter {
        /**
         * This bean is needed because Spring when you use xml config to load property files the bean is automatically
         * created... when you use @PropertySource then not so much
         * @return new bean
         */
        @Bean
        public PropertySourcesPlaceholderConfigurer propertySourcesPlaceholderConfigurer() {
            return new PropertySourcesPlaceholderConfigurer();
        }
    }
    

    jpaConfig.java

    @Configuration
    @EnableJpaRepositories("com.example.repository")
    public class JpaConfig {
    
        @Autowired
        private Environment env;
    
        /**
         * Create the fooDataSource Bean.
         * @return fooDataSource Bean
         */
        @Bean
        public BasicDataSource fooDataSource() {
    
            BasicDataSource basicDataSource = new BasicDataSource();
            basicDataSource.setDriverClassName(env.getProperty("cfg_foo.driver.name"));
            basicDataSource.setUrl(env.getProperty("cfg_foo.jdbc.url"));
            basicDataSource.setUsername(env.getProperty("cfg_foo.username"));
            basicDataSource.setPassword(env.getProperty("cfg_foo.password"));
            basicDataSource.setPoolPreparedStatements(Boolean.valueOf(env.getProperty("cfg_foo.poolPreparedStatements")));
            basicDataSource.setInitialSize(Integer.valueOf(env.getProperty("cfg_foo.poolInitialSize")));
            basicDataSource.setMaxActive(Integer.valueOf(env.getProperty("cfg_foo.poolMaxActive")));
            basicDataSource.setMaxIdle(Integer.valueOf(env.getProperty("cfg_foo.poolMaxIdle")));
            basicDataSource.setValidationQuery("SELECT '1'");
    
            return basicDataSource;
        }
    
        /**
         * Create the hibernateJpaVendorAdapter Bean.
         * @return hibernateJpaVendorAdapter Bean
         */
        @Bean
        public HibernateJpaVendorAdapter hibernateJpaVendorAdapter() {
    
            HibernateJpaVendorAdapter adapter = new HibernateJpaVendorAdapter();
            adapter.setDatabasePlatform("org.hibernate.dialect.MySQLDialect");
            adapter.setShowSql(Boolean.valueOf(env.getProperty("show.sql")));
            adapter.setGenerateDdl(Boolean.valueOf(env.getProperty("format.sql")));
    
            return adapter;
        }
    
        /**
         * Create the entityManagerFactory Bean.
         * @return entityManagerFactory Bean
         */
        @Bean
        public LocalContainerEntityManagerFactoryBean entityManagerFactory() {
    
            LocalContainerEntityManagerFactoryBean entityManagerFactory = new LocalContainerEntityManagerFactoryBean();
            entityManagerFactory.setPersistenceUnitName("foo");
            entityManagerFactory.setDataSource(fooDataSource());
            entityManagerFactory.setJpaVendorAdapter(hibernateJpaVendorAdapter());
            entityManagerFactory.setPackagesToScan("com.example.repository");
    
            return entityManagerFactory;
        }
    
    }
    

    【讨论】:

    • 我在答案中添加了一个非常基本的示例。希望它有所帮助:)
    • 感谢这个例子,现在你说的更清楚了。但是我们在这个项目中没有使用 Spring,我看不到你在哪里配置模式? (我错过了什么吗?)
    • 哦...抱歉。如果没有弹簧,我不知道如何做到这一点
    【解决方案2】:

    我不知道它是否是一个更好的解决方案,但您可以使用所需的架构注释您的 JPA 实体

    @Entity
    @Table(name = "Foo", schema = "Bar")
    

    【讨论】:

    • 不会硬编码模式,让用户根本无法更改它吗?
    • 确实如此。避免这种硬编码。
    【解决方案3】:

    META-INF/orm.xml 是默认名称,如果文件存在,则无论是否已在持久化单元中指定,都将使用它。如果persistence.xml 中的映射文件有另一个名称,则不使用默认名称。

    要使用多个不兼容的数据库提供程序,例如 SQL Server 和 Oracle,可以在 persistence.xml 中拥有多个持久性单元,并在运行时选择合适的单元。如果映射文件使用非默认名称命名,则每个单元可以有自己的映射文件或根本没有。

    【讨论】:

      【解决方案4】:

      使用此类 PersistenceUnitProperties 作为名称。

      查看链接 https://gerardnico.com/jpa/property

      属性示例

      javax.persistence.jdbc.driver=org.sqlite.JDBC
      eclipselink.ddl-generation=create-tables
      #eclipselink.ddl-generation=none
      javax.persistence.jdbc.url=jdbc:sqlite:domo.db
      

      持久性示例(persistence.xml)

      <persistence
          version="2.0" 
          xmlns="http://java.sun.com/xml/ns/persistence" 
          xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
          xsi:schemaLocation="http://java.sun.com/xml/ns/persistence http://java.sun.com/xml/ns/persistence/persistence_2_0.xsd">
          <persistence-unit name="PersistenceNameExample" transaction-type="RESOURCE_LOCAL">
              <provider>org.eclipse.persistence.jpa.PersistenceProvider</provider>
              <properties></properties>
          </persistence-unit>
      </persistence>
      

      实例

      Properties props = new Properties();
      //props.setProperty("hibernate.hbm2ddl.auto", "none");
      props.load(new FileInputStream("prop_example.properties"));
      Persistence.createEntityManagerFactory("PersistenceNameExample",props);
      

      提供者的 Eclipse 示例依赖

          <dependency>
              <groupId>org.eclipse.persistence</groupId>
              <artifactId>org.eclipse.persistence.jpa</artifactId>
              <version>2.7.4</version>
          </dependency>
      

      【讨论】:

        猜你喜欢
        • 2015-07-24
        • 1970-01-01
        • 2020-06-20
        • 1970-01-01
        • 1970-01-01
        • 2013-07-28
        • 2015-07-19
        • 2019-08-16
        • 2018-01-13
        相关资源
        最近更新 更多