javascript
orm jdbc_Spring Data JDBC通用DAO实现–迄今为止最轻量的ORM
orm jdbc
我很高興宣布Spring Data JDBC存儲庫項目的第一個版本。 這個開源庫的目的是為基于Spring框架中 JdbcTemplate關(guān)系數(shù)據(jù)庫提供通用,輕量且易于使用的DAO實現(xiàn),與項目的Spring Data 框架兼容。
設(shè)計目標(biāo)
- 輕巧,快速且開銷低。 只有少數(shù)幾個類, 沒有XML,注釋,反射
- 這不是成熟的ORM 。 沒有關(guān)系處理,延遲加載,臟檢查,緩存
- 在幾秒鐘內(nèi)實現(xiàn)CRUD
- 對于JPA過大的小型應(yīng)用程序
- 在需要簡單性或考慮將來遷移到JPA時使用
- 對數(shù)據(jù)庫方言差異的最小化支持(例如,結(jié)果的透明分頁)
特征
每個DAO為以下內(nèi)容提供內(nèi)置支持:
- 通過RowMapper抽象到域?qū)ο?從域?qū)ο笥成?
- 生成的和用戶定義的主鍵
- 提取生成的密鑰
- 復(fù)合(多列)主鍵
- 不變的領(lǐng)域?qū)ο?
- 分頁(請求結(jié)果子集)
- 按幾列排序(與數(shù)據(jù)庫無關(guān))
- 對多對一關(guān)系的可選支持
- 支持的數(shù)據(jù)庫(不斷測試):
- MySQL
- 通過SqlGenerator類可以輕松擴展到其他數(shù)據(jù)庫方言。
- 通過ID輕松檢索記錄
API
與Spring Data PagingAndSortingRepository抽象兼容, 所有這些方法都為您實現(xiàn) :
public interface PagingAndSortingRepository<T, ID extends Serializable> extends CrudRepository<T, ID> {T save(T entity);Iterable<T> save(Iterable<? extends T> entities);T findOne(ID id);boolean exists(ID id);Iterable<T> findAll();long count();void delete(ID id);void delete(T entity);void delete(Iterable<? extends T> entities);void deleteAll();Iterable<T> findAll(Sort sort);Page<T> findAll(Pageable pageable); }還完全支持Pageable和Sort參數(shù),這意味著您可以通過任意屬性免費獲得分頁和排序 。 例如,假設(shè)您有userRepository擴展了PagingAndSortingRepository<User, String>接口(由庫為您實現(xiàn)),并且在應(yīng)用某種排序后,您請求了USERS表的第5頁,每頁10個:
Page<User> page = userRepository.findAll(new PageRequest(5, 10,new Sort(new Order(DESC, "reputation"),new Order(ASC, "user_name"))) );Spring Data JDBC存儲庫庫將把此調(diào)用轉(zhuǎn)換為(PostgreSQL語法):
SELECT * FROM USERS ORDER BY reputation DESC, user_name ASC LIMIT 50 OFFSET 10…甚至(Derby語法):
SELECT * FROM (SELECT ROW_NUMBER() OVER () AS ROW_NUM, t.*FROM (SELECT *FROM USERSORDER BY reputation DESC, user_name ASC) AS t) AS a WHERE ROW_NUM BETWEEN 51 AND 60無論使用哪個數(shù)據(jù)庫,都將獲得Page<User>對象作為返回對象(您仍然必須自己提供RowMapper<User>才能將其從ResultSet轉(zhuǎn)換為域?qū)ο蟆H绻€不知道Spring Data項目,那么Page<T>是一個很棒的抽象,不僅封裝了List<User> ,而且還提供了元數(shù)據(jù),例如記錄總數(shù),我們當(dāng)前所在的頁面等。
使用理由
- 由于將來您的代碼將僅依賴于Spring Data Commons傘項目中的PagingAndSortingRepository和CrudRepository定義的方法, PagingAndSortingRepository您可以自由地從JdbcRepository實現(xiàn)(從該項目)切換到: JpaRepository , MongoRepository , GemfireRepository或GraphRepository 。 它們都實現(xiàn)相同的通用API。 當(dāng)然,不要指望從JDBC切換到JPA或MongoDB就像切換導(dǎo)入的JAR依賴項一樣簡單-但是至少您可以通過使用相同的DAO API最小化影響。
- 您需要一個快速,簡單的JDBC包裝器庫。 JPA甚至MyBatis都不過分
- 如果需要,您想完全控制生成SQL
- 您想使用對象,但是不需要延遲加載,關(guān)系處理,多級緩存,臟檢查……您需要CRUD等等
- 您想干嗎
- 您已經(jīng)在使用Spring或什至JdbcTemplate ,但仍然覺得手工工作過多
- 您的數(shù)據(jù)庫表很少
入門
有關(guān)更多示例和工作代碼,請不要忘記檢查項目測試 。
先決條件
Maven坐標(biāo):
<dependency><groupId>com.blogspot.nurkiewicz</groupId><artifactId>jdbcrepository</artifactId><version>0.1</version> </dependency>不幸的是,該項目尚未在Maven中央存儲庫中 。 目前,您可以通過克隆將庫安裝在本地存儲庫中:
$ git clone git://github.com/nurkiewicz/spring-data-jdbc-repository.git $ git checkout 0.1 $ mvn javadoc:jar source:jar install為了啟動您的項目,必須存在DataSource bean并啟用事務(wù)管理。 這是一個最小MySQL配置:
@EnableTransactionManagement @Configuration public class MinimalConfig {@Beanpublic PlatformTransactionManager transactionManager() {return new DataSourceTransactionManager(dataSource());}@Beanpublic DataSource dataSource() {MysqlConnectionPoolDataSource ds = new MysqlConnectionPoolDataSource();ds.setUser("user");ds.setPassword("secret");ds.setDatabaseName("db_name");return ds;}}具有自動生成的密鑰的實體
假設(shè)您有一個具有自動生成的密鑰(MySQL語法)的以下數(shù)據(jù)庫表:
CREATE TABLE COMMENTS (id INT AUTO_INCREMENT,user_name varchar(256),contents varchar(1000),created_time TIMESTAMP NOT NULL,PRIMARY KEY (id) );首先,您需要創(chuàng)建到該表的域?qū)ο骍ser映射(就像在任何其他ORM中一樣):
public class Comment implements Persistable<Integer> {private Integer id;private String userName;private String contents;private Date createdTime;@Overridepublic Integer getId() {return id;}@Overridepublic boolean isNew() {return id == null;}//getters/setters/constructors/... }除了標(biāo)準(zhǔn)的Java樣板之外,您還應(yīng)該注意實現(xiàn)Persistable<Integer> ,其中Integer是主鍵的類型。 Persistable<T>是一個來自Spring Data項目的接口,這是我們對您的域?qū)ο蟮奈ㄒ灰蟆?
最后,我們準(zhǔn)備創(chuàng)建CommentRepository DAO:
@Repository public class CommentRepository extends JdbcRepository<Comment, Integer> {public CommentRepository() {super(ROW_MAPPER, ROW_UNMAPPER, "COMMENTS");}public static final RowMapper<Comment> ROW_MAPPER = //see belowprivate static final RowUnmapper<Comment> ROW_UNMAPPER = //see below@Overrideprotected Comment postCreate(Comment entity, Number generatedId) {entity.setId(generatedId.intValue());return entity;} }首先,我們使用@Repository批注標(biāo)記DAO bean。 它啟用持久性異常轉(zhuǎn)換。 通過CLASSPATH掃描也可以找到這種帶注釋的bean。
如您所見,我們擴展了JdbcRepository<Comment, Integer> (該庫的中心類),它提供了所有PagingAndSortingRepository方法的實現(xiàn)。 它的構(gòu)造函數(shù)具有三個必需的依賴項: RowMapper , RowUnmapper和表名。 您也可以提供ID列名稱,否則使用默認的"id" 。
如果您曾經(jīng)使用過Spring的JdbcTemplate ,則應(yīng)該熟悉RowMapper界面。 我們需要以某種方式將ResultSet列提取到一個對象中。 畢竟,我們不想使用原始的JDBC結(jié)果。 這很簡單:
public static final RowMapper<Comment> ROW_MAPPER = new RowMapper<Comment>() {@Overridepublic Comment mapRow(ResultSet rs, int rowNum) throws SQLException {return new Comment(rs.getInt("id"),rs.getString("user_name"),rs.getString("contents"),rs.getTimestamp("created_time"));} };RowUnmapper來自此庫,它本質(zhì)上與RowMapper相反:接收一個對象并將其轉(zhuǎn)換為Map 。 庫稍后使用此映射來構(gòu)造SQL CREATE / UPDATE查詢:
private static final RowUnmapper<Comment> ROW_UNMAPPER = new RowUnmapper<Comment>() {@Overridepublic Map<String, Object> mapColumns(Comment comment) {Map<String, Object> mapping = new LinkedHashMap<String, Object>();mapping.put("id", comment.getId());mapping.put("user_name", comment.getUserName());mapping.put("contents", comment.getContents());mapping.put("created_time", new java.sql.Timestamp(comment.getCreatedTime().getTime()));return mapping;} };如果您從不更新數(shù)據(jù)庫表(僅讀取插入在其他位置的一些參考數(shù)據(jù)),則可以跳過RowUnmapper參數(shù)或使用MissingRowUnmapper 。
最后一個難題是postCreate()回調(diào)方法,該方法在插入對象后調(diào)用。 您可以使用它來檢索生成的主鍵并更新域?qū)ο?#xff08;如果域?qū)ο笫遣豢勺兊?#xff0c;則返回新的主鍵)。 如果不需要它,就不要重寫postCreate() 。 根據(jù)此示例,檢查JdbcRepositoryGeneratedKeyTest以獲取有效的代碼。
到目前為止,您可能會覺得與JPA或Hibernate相比,有很多手動工作。 但是,眾所周知,各種JPA實現(xiàn)和其他ORM框架都會引入大量開銷并顯示一些學(xué)習(xí)曲線。 這個微小的庫有意讓用戶承擔(dān)一些責(zé)任,以避免復(fù)雜的映射,反射,注釋……所有并非總是需要的隱式性。 該項目無意替代成熟穩(wěn)定的ORM框架。 相反,它試圖填補原始JDBC和ORM之間的利基,其中簡單性和低開銷是關(guān)鍵特征。
具有手動分配的密鑰的實體
在此示例中,我們將看到如何處理具有用戶定義的主鍵的實體。 讓我們從數(shù)據(jù)庫模型開始:
CREATE TABLE USERS (user_name varchar(255),date_of_birth TIMESTAMP NOT NULL,enabled BIT(1) NOT NULL,PRIMARY KEY (user_name) );…和User域模型:
public class User implements Persistable<String> {private transient boolean persisted;private String userName;private Date dateOfBirth;private boolean enabled;@Overridepublic String getId() {return userName;}@Overridepublic boolean isNew() {return !persisted;}public User withPersisted(boolean persisted) {this.persisted = persisted;return this;}//getters/setters/constructors/...}注意,添加了特殊的persisted瞬態(tài)標(biāo)志。 來自Spring Data項目的CrudRepository.save()合同要求一個實體知道其是否已保存( isNew() )方法–沒有單獨的create()和update()方法。 對于自動生成的鍵,實現(xiàn)isNew()很簡單(請參見上面的Comment ),但是在這種情況下,我們需要一個額外的瞬態(tài)字段。 如果您討厭這種解決方法,并且只插入數(shù)據(jù)而從不更新,則始終可以從isNew()返回true 。
最后是我們的DAO, UserRepository bean:
@Repository public class UserRepository extends JdbcRepository<User, String> {public UserRepository() {super(ROW_MAPPER, ROW_UNMAPPER, "USERS", "user_name");}public static final RowMapper<User> ROW_MAPPER = //...public static final RowUnmapper<User> ROW_UNMAPPER = //...@Overrideprotected User postUpdate(User entity) {return entity.withPersisted(true);}@Overrideprotected User postCreate(User entity, Number generatedId) {return entity.withPersisted(true);} }"USERS"和"user_name"參數(shù)指定表名和主鍵列名。 我將保留mapper和unmapper的詳細信息(請參閱源代碼 )。 但是請注意postUpdate()和postCreate()方法。 它們確保一旦對象被持久保存,就設(shè)置了persisted標(biāo)志,以便隨后對save()調(diào)用將更新現(xiàn)有實體,而不是嘗試重新插入它。
根據(jù)此示例,檢查JdbcRepositoryManualKeyTest以獲取有效的代碼。
復(fù)合主鍵
我們還支持復(fù)合主鍵(由幾列組成的主鍵)。 以該表為例:
CREATE TABLE BOARDING_PASS (flight_no VARCHAR(8) NOT NULL,seq_no INT NOT NULL,passenger VARCHAR(1000),seat CHAR(3),PRIMARY KEY (flight_no, seq_no) );我希望您注意到Peristable<T>的主鍵類型:
public class BoardingPass implements Persistable<Object[]> {private transient boolean persisted;private String flightNo;private int seqNo;private String passenger;private String seat;@Overridepublic Object[] getId() {return pk(flightNo, seqNo);}@Overridepublic boolean isNew() {return !persisted;}//getters/setters/constructors/...}不幸的是,我們不支持將所有ID值封裝在一個對象中的小數(shù)值類(就像JPA使用@IdClass ),因此您必須使用Object[]數(shù)組。 定義DAO類類似于我們已經(jīng)看到的內(nèi)容:
public class BoardingPassRepository extends JdbcRepository<BoardingPass, Object[]> {public BoardingPassRepository() {this("BOARDING_PASS");}public BoardingPassRepository(String tableName) {super(MAPPER, UNMAPPER, new TableDescription(tableName, null, "flight_no", "seq_no"));}public static final RowMapper<BoardingPass> ROW_MAPPER = //...public static final RowUnmapper<BoardingPass> UNMAPPER = //...}需要注意的兩件事:我們擴展了JdbcRepository<BoardingPass, Object[]>并且按預(yù)期提供了兩個ID列名稱: "flight_no", "seq_no" 。 我們通過提供由Object[]包裹的flight_no和seq_no (必須seq_no順序)值來查詢此類DAO:
BoardingPass pass = repository.findOne(new Object[] {"FOO-1022", 42});毫無疑問,這在實踐中很麻煩,因此我們提供了微小的輔助方法,您可以靜態(tài)導(dǎo)入:
import static com.blogspot.nurkiewicz.jdbcrepository.JdbcRepository.pk; //...BoardingPass foundFlight = repository.findOne(pk("FOO-1022", 42));根據(jù)此示例,檢查JdbcRepositoryCompoundPkTest以獲取工作代碼。
交易次數(shù)
該庫與事務(wù)管理完全正交。 每個存儲庫的每種方法都需要運行事務(wù),并且要由您來設(shè)置它。 通常,您將@Transactional放在服務(wù)層上(稱為DAO bean)。 我不建議將@Transactional放在每個DAO bean上 。
快取
Spring Data JDBC存儲庫庫不提供任何緩存抽象或支持。 但是, 在Spring中使用緩存抽象將@Cacheable層添加@Cacheable DAO或服務(wù)之上非常簡單。 另請參見: Spring中的@Cacheable開銷 。
會費
..總是歡迎。 不要猶豫, 提交錯誤報告并提出請求 。 現(xiàn)在最大的缺失功能是對MSSQL和Oracle數(shù)據(jù)庫的支持。 如果有人可以看一下,那將是非常棒的。
測試中
該庫已使用Travis( )。 測試套件包含265個測試 (53個不同的測試,每個測試針對5個不同的數(shù)據(jù)庫運行:MySQL,PostgreSQL,H2,HSQLDB和Derby。
在填寫錯誤報告或提交新功能時,請嘗試包括支持測試用例。 每個拉取請求都會在單獨的分支上自動進行測試。
建造
分叉后, 正式的存儲庫構(gòu)建就像運行一樣簡單:
$ mvn install 在JUnit測試執(zhí)行期間,您會注意到很多異常。 這是正常的。 一些測試是針對僅在Travis CI服務(wù)器上可用MySQL和PostgreSQL運行的。 當(dāng)這些數(shù)據(jù)庫服務(wù)器不可用時,只需跳過整個測試:
結(jié)果:
異常堆棧跟蹤來自根AbstractIntegrationTest 。
設(shè)計
庫僅包含少數(shù)幾個類,如下圖所示:
JdbcRepository是實現(xiàn)所有PagingAndSortingRepository方法的最重要的類。 每個用戶存儲庫都必須擴展此類。 同樣,每個此類存儲庫都必須至少實現(xiàn)RowMapper和RowUnmapper (僅當(dāng)您要修改表數(shù)據(jù)時)。
SQL生成委托給SqlGenerator 。 PostgreSqlGenerator. 和DerbySqlGenerator用于與標(biāo)準(zhǔn)生成器不DerbySqlGenerator的數(shù)據(jù)庫。
執(zhí)照
該項目是在Apache許可的 2.0版(與Spring框架相同)下發(fā)布的。
參考: NoBlogDefFound博客中的JCG合作伙伴 Tomasz Nurkiewicz 為程序員提供的概率分布 。
翻譯自: https://www.javacodegeeks.com/2013/01/spring-data-jdbc-generic-dao-implementation-most-lightweight-orm-ever.html
orm jdbc
總結(jié)
以上是生活随笔為你收集整理的orm jdbc_Spring Data JDBC通用DAO实现–迄今为止最轻量的ORM的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: linux的内核源码在哪(linux的内
- 下一篇: 企业内网搭建要多少钱(企业内网ddos)