javascript
项目学生:Spring数据的持久性
這是Project Student的一部分。 其他職位包括帶有Jersey的Webservice Client,帶有Jersey的 Webservice Server和業務層 。
RESTful webapp onion的最后一層是持久層。
持久層有兩種哲學。 一個陣營將數據庫視為一個簡單的存儲,并希望保持這一層非常薄。 另一陣營知道,在數據庫中執行任務通常比訪問數據庫中的數據,用Java完成必要的工作以及第二次訪問結果要快得多。 該陣營通常需要一個相當厚的持久層。
第二個陣營也有一群希望廣泛使用存儲過程的流放者。 這使數據庫功能更加強大,但在持久層中卻付出了一些額外的復雜性。 它的主要缺點是存儲過程是特定于數據庫的。
第二批流亡者僅使用存儲過程來提高安全性。 我已經在前面進行了討論,例如,您應該使用用戶名和密碼來調用存儲過程并獲得“接受”或“拒絕”響應,而不是檢索哈希密碼并在應用程序中進行比較。 第一種方法允許您使用數據庫的GRANT和REVOKE特權將哈希密碼存儲在不可訪問的表中,即使攻擊者可以以應用程序用戶身份執行任意SQL注入。
(旁注:您通常可以用Java編寫存儲過程!Oracle支持它,PostgreSQL支持它(通過PL / Java擴展),H2支持它(通過類路徑)。我不知道MySQL是否支持它。這種方法有確定的成本,但這可能是解決許多問題的最佳解決方案。)
無論如何,該項目目前僅支持CRUD操作,它們是使用薄型持久層的經典示例。 但是添加“ thick”方法很容易–只需使用它們創建一個CourseRepositoryImpl類。 (此類不應實現CourseRepository接口!)
設計決策
Spring Data –我們正在使用Spring Data,因為它會自動生成持久層類。
局限性
分頁 –沒有嘗試支持分頁。 由于Spring Data已經支持它,所以這不是一個大問題-我們只需要編寫膠水即可。
組態
基本配置僅需要@EnableJpaRepositories批注。
使用純內存嵌入式數據庫的集成測試需要更多的工作。 即使使用H2網址(告訴它使數據庫服務器不工作),也不能直接使用Spring嵌入式數據庫。 數據庫已正確初始化,但在測試實際運行之前已關閉。 由于缺少數據庫架構,結果是失敗。
使用由文件支持的內存數據庫是微不足道的,但可能導致并發運行問題,意外拉入舊的測試數據等。顯而易見的解決方案是使用隨機臨時支持文件,但這種方法引入了自己的問題。
下面的方法是將嵌入式數據庫緩存在配置類中,并僅在應用程序關閉時銷毀它。 這引入了一些非顯而易見的行為,迫使我們也必須顯式創建一些其他bean。
(IIRC,如果您在配置類中創建嵌入式數據庫,并在配置文件中創建事務Bean,則spring在配置文件中創建幻像數據源,并且初始化失敗。當我開始在同一文件中創建事務Bean時,此問題消失了放置為數據源。)
@Configuration @EnableJpaRepositories(basePackages = { "com.invariantproperties.sandbox.student.repository" }) @EnableTransactionManagement @PropertySource("classpath:test-application.properties") @ImportResource("classpath:applicationContext-dao.xml") public class TestPersistenceJpaConfig implements DisposableBean {private static final Logger log = LoggerFactory.getLogger(TestPersistenceJpaConfig.class);private static final String PROPERTY_NAME_HIBERNATE_DIALECT = "hibernate.dialect";private static final String PROPERTY_NAME_HIBERNATE_FORMAT_SQL = "hibernate.format_sql";private static final String PROPERTY_NAME_HIBERNATE_NAMING_STRATEGY = "hibernate.ejb.naming_strategy";private static final String PROPERTY_NAME_HIBERNATE_SHOW_SQL = "hibernate.show_sql";private static final String PROPERTY_NAME_HIBERNATE_HBM2DDL_AUTO = "hibernate.hbm2ddl.auto";private static final String PROPERTY_NAME_ENTITYMANAGER_PACKAGES_TO_SCAN = "entitymanager.packages.to.scan";// private static final String PROPERTY_NAME_PERSISTENCE_UNIT_NAME =// "persistence.unit.name";@Resourceprivate Environment environment;private EmbeddedDatabase db = null;@Beanpublic DataSource dataSource() {final EmbeddedDatabaseBuilder builder = new EmbeddedDatabaseBuilder();db = builder.setType(EmbeddedDatabaseType.H2).build(); // .script("foo.sql")return db;}@Beanpublic LocalContainerEntityManagerFactoryBean entityManagerFactory() throws ClassNotFoundException {LocalContainerEntityManagerFactoryBean bean = new LocalContainerEntityManagerFactoryBean();bean.setDataSource(dataSource());bean.setPackagesToScan(environment.getRequiredProperty(PROPERTY_NAME_ENTITYMANAGER_PACKAGES_TO_SCAN));bean.setPersistenceProviderClass(HibernatePersistence.class);// bean.setPersistenceUnitName(environment// .getRequiredProperty(PROPERTY_NAME_PERSISTENCE_UNIT_NAME));HibernateJpaVendorAdapter va = new HibernateJpaVendorAdapter();bean.setJpaVendorAdapter(va);Properties jpaProperties = new Properties();jpaProperties.put(PROPERTY_NAME_HIBERNATE_DIALECT, environment.getRequiredProperty(PROPERTY_NAME_HIBERNATE_DIALECT));jpaProperties.put(PROPERTY_NAME_HIBERNATE_FORMAT_SQL,environment.getRequiredProperty(PROPERTY_NAME_HIBERNATE_FORMAT_SQL));jpaProperties.put(PROPERTY_NAME_HIBERNATE_NAMING_STRATEGY,environment.getRequiredProperty(PROPERTY_NAME_HIBERNATE_NAMING_STRATEGY));jpaProperties.put(PROPERTY_NAME_HIBERNATE_SHOW_SQL,environment.getRequiredProperty(PROPERTY_NAME_HIBERNATE_SHOW_SQL));jpaProperties.put(PROPERTY_NAME_HIBERNATE_HBM2DDL_AUTO,environment.getRequiredProperty(PROPERTY_NAME_HIBERNATE_HBM2DDL_AUTO));bean.setJpaProperties(jpaProperties);return bean;}@Beanpublic PlatformTransactionManager transactionManager() {JpaTransactionManager tm = new JpaTransactionManager();try {tm.setEntityManagerFactory(this.entityManagerFactory().getObject());} catch (ClassNotFoundException e) {// TODO: log.}return tm;}@Beanpublic PersistenceExceptionTranslationPostProcessor exceptionTranslation() {return new PersistenceExceptionTranslationPostProcessor();}@Overridepublic void destroy() {if (db != null) {db.shutdown();}} }applicationContext.xml文件為空。 該屬性文件是:
# hibernate configuration hibernate.dialect=org.hibernate.dialect.H2Dialect hibernate.ejb.naming_strategy=org.hibernate.cfg.ImprovedNamingStrategy hibernate.show_sql=false hibernate.format_sql=true hibernate.hbm2ddl.auto=create# jpa configuration entitymanager.packages.to.scan=com.invariantproperties.sandbox.student.domain persistence.unit.dataSource=java:comp/env/jdbc/ssDS persistence.unit.name=ssPU單元測試
由于所有代碼都是自動生成的,因此沒有單元測試。 這將隨著添加自定義方法而改變。
整合測試
現在,我們可以編寫業務層的集成測試:
@RunWith(SpringJUnit4ClassRunner.class) @ContextConfiguration(classes = { BusinessApplicationContext.class, TestBusinessApplicationContext.class,TestPersistenceJpaConfig.class }) @Transactional @TransactionConfiguration(defaultRollback = true) public class CourseServiceIntegrationTest {@Resourceprivate CourseService dao;@Testpublic void testCourseLifecycle() throws Exception {final String name = "Calculus 101";final Course expected = new Course();expected.setName(name);assertNull(expected.getId());// create courseCourse actual = dao.createCourse(name);expected.setId(actual.getId());expected.setUuid(actual.getUuid());expected.setCreationDate(actual.getCreationDate());assertThat(expected, equalTo(actual));assertNotNull(actual.getUuid());assertNotNull(actual.getCreationDate());// get course by idactual = dao.findCourseById(expected.getId());assertThat(expected, equalTo(actual));// get course by uuidactual = dao.findCourseByUuid(expected.getUuid());assertThat(expected, equalTo(actual));// update courseexpected.setName("Calculus 102");actual = dao.updateCourse(actual, expected.getName());assertThat(expected, equalTo(actual));// delete Coursedao.deleteCourse(expected.getUuid());try {dao.findCourseByUuid(expected.getUuid());fail("exception expected");} catch (ObjectNotFoundException e) {// expected}}/*** @test findCourseById() with unknown course.*/@Test(expected = ObjectNotFoundException.class)public void testfindCourseByIdWhenCourseIsNotKnown() {final Integer id = 1;dao.findCourseById(id);}/*** @test findCourseByUuid() with unknown Course.*/@Test(expected = ObjectNotFoundException.class)public void testfindCourseByUuidWhenCourseIsNotKnown() {final String uuid = "missing";dao.findCourseByUuid(uuid);}/*** Test updateCourse() with unknown course.* * @throws ObjectNotFoundException*/@Test(expected = ObjectNotFoundException.class)public void testUpdateCourseWhenCourseIsNotFound() {final Course course = new Course();course.setUuid("missing");dao.updateCourse(course, "Calculus 102");}/*** Test deleteCourse() with unknown course.* * @throws ObjectNotFoundException*/@Test(expected = ObjectNotFoundException.class)public void testDeleteCourseWhenCourseIsNotFound() {dao.deleteCourse("missing");} }源代碼
- 可從http://code.google.com/p/invariant-properties-blog/source/browse/student/student-business和http://code.google.com/p/invariant-properties-獲得源代碼。 blog / source / browse / student / student-persistence 。
翻譯自: https://www.javacodegeeks.com/2014/01/project-student-persistence-with-spring-data.html
總結
以上是生活随笔為你收集整理的项目学生:Spring数据的持久性的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 安卓叶绿386金手指(安卓叶绿)
- 下一篇: Spring Data MongoDB级