javascript
项目学生:Spring数据的持久性
這是Project Student的一部分。 其他職位包括帶有Jersey的Webservice Client,帶有Jersey的 Webservice Server和業(yè)務(wù)層 。
RESTful webapp onion的最后一層是持久層。
持久層有兩種哲學(xué)。 一個陣營將數(shù)據(jù)庫視為一個簡單的存儲,并希望保持這一層非常薄。 另一陣營知道,在數(shù)據(jù)庫中執(zhí)行任務(wù)通常比訪問數(shù)據(jù)庫中的數(shù)據(jù),用Java完成必要的工作以及第二次訪問結(jié)果要快得多。 該陣營通常需要一個相當(dāng)厚的持久層。
第二個陣營也有一群希望廣泛使用存儲過程的流放者。 這使數(shù)據(jù)庫功能更加強大,但在持久層中卻付出了一些額外的復(fù)雜性。 它的主要缺點是存儲過程是特定于數(shù)據(jù)庫的。
第二批流亡者僅使用存儲過程來提高安全性。 我已經(jīng)在前面進行了討論,例如,您應(yīng)該使用用戶名和密碼來調(diào)用存儲過程并獲得“接受”或“拒絕”響應(yīng),而不是檢索哈希密碼并在應(yīng)用程序中進行比較。 第一種方法允許您使用數(shù)據(jù)庫的GRANT和REVOKE特權(quán)將哈希密碼存儲在不可訪問的表中,即使攻擊者可以以應(yīng)用程序用戶身份執(zhí)行任意SQL注入。
(旁注:您通常可以用Java編寫存儲過程!Oracle支持它,PostgreSQL支持它(通過PL / Java擴展),H2支持它(通過類路徑)。我不知道MySQL是否支持它。這種方法有確定的成本,但這可能是解決許多問題的最佳解決方案。)
無論如何,該項目目前僅支持CRUD操作,它們是使用薄型持久層的經(jīng)典示例。 但是添加“ thick”方法很容易–只需使用它們創(chuàng)建一個CourseRepositoryImpl類。 (此類不應(yīng)實現(xiàn)CourseRepository接口!)
設(shè)計決策
Spring Data –我們正在使用Spring Data,因為它會自動生成持久層類。
局限性
分頁 –沒有嘗試支持分頁。 由于Spring Data已經(jīng)支持它,所以這不是一個大問題-我們只需要編寫膠水即可。
組態(tài)
基本配置僅需要@EnableJpaRepositories批注。
使用純內(nèi)存嵌入式數(shù)據(jù)庫的集成測試需要更多的工作。 即使使用H2網(wǎng)址(告訴它使數(shù)據(jù)庫服務(wù)器不工作),也不能直接使用Spring嵌入式數(shù)據(jù)庫。 數(shù)據(jù)庫已正確初始化,但在測試實際運行之前已關(guān)閉。 由于缺少數(shù)據(jù)庫架構(gòu),結(jié)果是失敗。
使用由文件支持的內(nèi)存數(shù)據(jù)庫是微不足道的,但可能導(dǎo)致并發(fā)運行問題,意外拉入舊的測試數(shù)據(jù)等。顯而易見的解決方案是使用隨機臨時支持文件,但這種方法引入了自己的問題。
下面的方法是將嵌入式數(shù)據(jù)庫緩存在配置類中,并僅在應(yīng)用程序關(guān)閉時銷毀它。 這引入了一些非顯而易見的行為,迫使我們也必須顯式創(chuàng)建一些其他bean。
(IIRC,如果您在配置類中創(chuàng)建嵌入式數(shù)據(jù)庫,并在配置文件中創(chuàng)建事務(wù)Bean,則spring在配置文件中創(chuàng)建幻像數(shù)據(jù)源,并且初始化失敗。當(dāng)我開始在同一文件中創(chuàng)建事務(wù)Bean時,此問題消失了放置為數(shù)據(jù)源。)
@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單元測試
由于所有代碼都是自動生成的,因此沒有單元測試。 這將隨著添加自定義方法而改變。
整合測試
現(xiàn)在,我們可以編寫業(yè)務(wù)層的集成測試:
@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
總結(jié)
以上是生活随笔為你收集整理的项目学生:Spring数据的持久性的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 安卓叶绿386金手指(安卓叶绿)
- 下一篇: 调查内存泄漏第2部分–分析问题