跳到主要內容

Spring boot v1.5 (六) spring data jpa 基本操作

最近天氣好熱,做甚麼事都覺得很懶,想要寫個spring data jpa也是懶懶的,不過這部分卻也是滿重要的一部分,前一篇介紹JDBCTemplate,已經覺得跟以前寫SQL方式有所差異了,JPA帶來的是物件導向的設計面思考,說到JPA不得不提提ORM,Object-relational mapping主要想法為簡化及物件導向的設計,讓RDB更貼近Object,在設計上可以更加便利,甚至透過一些設計可以讓Table具有物件導向的特性如繼承等等,以往要使用ORM的框架,都會先以Hibernate進行,不過近來慢慢地轉向JPA,主要還是在減少程式碼、增加彈性等等,大體的功能沒有差異很大,所以從Hibernate轉到JPA問題不大,JPA要介紹的東西還滿多的,所以我這裡會再分成三個章節來介紹。

  • SPRING DATA JPA基本操作
  • JPQL & Named SQL & Native SQL
  • Cache & DB Design Pattern
SPRING DATA JPA更加簡化的程式撰寫,只需要一個 Interface內寫一些查詢 method就可以操作JPA,因為利用 method 組合查詢條件,確實很方便也很容易理解,若是都沒有辦法符合需求當然也可以自己實作一個來用當然沒有問題。

學習目的:SPRING DATA JPA基本操作。
學習時數:3.5hr
教學影片:


pom.xml 說明
spring-boot-starter-web:配置 Web Project所需的函式庫。
spring-boot-starter-test:配置 unit or mock test 所需的函式庫。
spring-boot-starter-actuator:配置監控spring boot所需的函式庫,後續spring cloud會使用到,所以一開就導入。
spring-boot-starter-jdbc:配置使用jdbc所需的函式庫。
postgresql:配置postgresql連接Driver所需的函式庫。
jasypt-spring-boot-starter:加解密所需的函式庫。
spring-boot-starter-data-jpa:配置Spring data jpa所需的函式庫。
lombok:本次的彩蛋,降低getter與setter的套件。
有使用到上一篇的加解密
程式碼說明
  • Example6.java
package louisz.springboot.example6;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.autoconfigure.domain.EntityScan;
import org.springframework.context.annotation.PropertySource;
import org.springframework.data.jpa.repository.config.EnableJpaRepositories;
import org.springframework.transaction.annotation.EnableTransactionManagement;

/**
 * Louisz Spring boot introduce ex.6
 *
 */

@SpringBootApplication
@PropertySource(name = "EncryptedProperties", value = "classpath:example5.properties")//針對需加解密的設定檔檔名及位置
@EntityScan(basePackages = {"louisz.springboot.example6"})//載入指定package的entity
@EnableJpaRepositories(basePackages = {"louisz.springboot.example6"})//載入指定package jpaRepositories
@EnableTransactionManagement//啟動交易模式
public class Example6 {
 /**
  * 程式執行起點
  * 
  * @param args
  */
 public static void main(String[] args) {
  SpringApplication.run(Example6.class, args);
 }

}
  • Member.java
package louisz.springboot.example6;

import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.Id;
import javax.persistence.Table;

import lombok.Getter;
import lombok.Setter;

@Entity//Entity宣告
@Table(name = "member")//指定對應Table,請記得一定要有ID欄位(名稱可以改變)作為PK喔
public class Member {
 @Id //PK
    @GeneratedValue //之後針對SEQUENCE會需要
    @Setter @Getter //lombok 減少setter及getter method撰寫
    private Long id;
    @Column(nullable = false)//不可為空值
    @Setter @Getter
    private String name;
    @Column(nullable = false)
    @Setter @Getter
    private Integer age;
}

  • MemberRepository.java
package louisz.springboot.example6;

import java.util.List;

import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.data.jpa.repository.Query;
import org.springframework.data.repository.query.Param;

public interface MemberRepository extends JpaRepository{
 /**
  * 使用ID查詢資料,並回傳Member Entity
  * @param id
  * @return
  */
 Member findById(Long id);
 /**
  * 使用ID查詢資料,並回傳Member Entity
  * @param name
  * @return
  */
 Member findByName(String name);
 /**
  * 透過JPQL,使用NAME查詢,並回傳Member Entity
  * @param name
  * @return
  */
    @Query("from Member u where u.name=:name")
    Member findMember(@Param("name") String name);    
    /**
     * 使用NAME及AGE查詢,並回傳Member Entity
     * @param name
     * @param age
     * @return
     */
    Member findByNameAndAge(String name, Integer age);
    /**
     * 使用NAME進行Count,並回傳筆數
     * @param name
     * @return
     */
    Long countByName(String name);
    //不分姓名大小寫的查詢
    /**
     * 使用NAME進行查詢且不分大小寫,並回傳List
     * @param name
     * @return
     */
    List findByNameIgnoreCase(String name); 
}
  • TestMemberRepository.java
package louisz.springboot.example6;

import org.junit.Assert;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.boot.test.context.SpringBootTest.WebEnvironment;
import org.springframework.test.context.junit4.SpringRunner;

//spring boot 1.4以後的方式
@RunWith(SpringRunner.class)
@SpringBootTest(webEnvironment = WebEnvironment.RANDOM_PORT)
public class TestMemberRepository {

 @Autowired
 private MemberRepository memberRepository;

 @Before
 public void setUp() {
  // 先清除所有資料
  memberRepository.deleteAll();
 }

 @Test
 public void test() throws Exception {
  /*
   * 進行單元測試,先規劃相關情境
   */
  // 先確認是否清除完成
  Assert.assertEquals(0, memberRepository.findAll().size());
  // 新增一筆資料
  Member m = new Member();
  m.setId(1l);
  m.setAge(new Integer(12));
  m.setName("louisz");
  memberRepository.save(m);
  // 確認新增資料筆數
  Assert.assertEquals(1, memberRepository.findAll().size());
  // 檢查新增的資料是否正確
  Assert.assertEquals(12, memberRepository.findByName("louisz").getAge().longValue());
  // 修改姓名
  m.setName("jessie");
  memberRepository.save(m);
  // 檢查新增的資料是否正確
  Assert.assertEquals("jessie", memberRepository.findById(1l).getName());
  // 刪除新增的資料
  memberRepository.delete(memberRepository.findById(1l));
  // 確認刪除資料筆數
  Assert.assertEquals(0, memberRepository.findAll().size());
  // 再度儲存
  memberRepository.save(m);
  // 查詢姓名為jessie的筆數
  Assert.assertEquals(1, memberRepository.countByName("jessie").longValue());
  // 不分大小寫查詢姓名
  Assert.assertEquals(0, memberRepository.countByName("JESSIE").longValue());
  Assert.assertEquals(1, memberRepository.findByNameIgnoreCase("JESSIE").size());
 }

}

單元測試看結果,執行時請加上-Djasypt.encryptor.password=supersecretz
Github code(louisz.springboot.example6)
參考連結:
https://docs.spring.io/spring-data/jpa/docs/current/reference/html/#repositories.query-methods
http://blog.didispace.com/springbootdata2/
https://projectlombok.org/

留言

這個網誌中的熱門文章

使用Apache Http Server進行Proxy和LoadB alance

環境概述 OS:Linux Apache Http Server:2.2.21 安裝可以使用rpm或是抓source下來compile,建議前者 rpm –ivh apache-http-xxx.rpm 這裡請注意一下安裝的版本 http.conf 參數設定 Proxy < VirtualHost *:80>      ProxyPass targetPath fromPath | fromUrl < VirtualHost/>  範例如下 < VirtualHost *:80>      ProxyPass /app http://DomainName或IP:8888/app      ProxyPassReverse /app http://DomainName或IP:8888/app < VirtualHost/> 上述的參數設定為,將某主機的http通訊協定下的Web application(app),對應到本台Web Server的app下。這樣的Proxy架構的設定對企業內的Web應用程式相當有用處,若是遇到Web Application Server掛點只需要修改Proxy對應,不需要動到dns等等,是個滿便宜的Proxy架構設定喔。﹝不過要注意Web Application要注意redirectc和forward等的撰寫,盡量不要有絕對IP的出現﹞

IBM MQ Server To Server的簡易設定

IBM MQ 的STS建置方式,IBM Red Book已經說明的滿完整的,這裡稍微整理一下,步驟如下 在 MQ-A Server MQ QMGR(MQA) 上需要建立的物件定義如下: • 遠端佇列定義 PAYROLL.QUERY • 傳輸佇列定義 MQB(預設 = 遠端佇列管理程式名稱) • 程序定義 MQA.TO.MQB.PROCESS(對於 WebSphere MQ for AIX、HP-UX、Solaris 與 Windows,及 MQSeries for Compaq Tru64 UNIX 與 OS/2 Warp,並非必要) • 傳送端通道定義 MQA.TO.MQB • 接收端通道定義 MQB.TO.MQA • 本端佇列定義 MQ2 以下是必須建立在 MQ-B Server MQ QMGR(MQB) 中的物件定義: • 遠端佇列定義 PAYROLL.RESPONE • 傳輸佇列定義 MQA(預設值=遠端佇列管理程式名稱) • 程序定義 MQB.TO.MQA.PROCESS(對於 WebSphere MQ for AIX、HP-UX、Solaris 與 Windows,及 MQSeries for Compaq Tru64 UNIX 與 OS/2 Warp,並非必要) • 傳送端通道定義 MQB.TO.MQA • 接收端通道定義 MQA.TO.MQB • 本端佇列定義 MQ4 1.先啟動MQ-A Server上的QMGR(MQA),並使用Runmqsc進入。 2.在佇列管理程式 MQA 上執行下列指令。 遠端佇列定義 DEFINE QREMOTE(PAYROLL.QUERY) DESCR('MQA 的遠端佇列') REPLACE + PUT(ENABLED) XMITQ(MQB) RNAME(MQ4) RQMNAME(MQB) 註: 遠端佇列定義並非實體的佇列,但卻是引導訊息至傳輸佇列 (MQB) 的一種方式,以便能將訊息送至佇列管理程式 MQB。 傳輸佇列定義 DEFINE QLOCAL(MQB) DESCR('對 MQB 的傳輸佇列') REPLACE + USAGE(XMITQ) PUT(ENABLED) GET(ENABLED) TRIGGER TRIGTYPE(FIRST) + INITQ(SYSTEM.CHANN

IReport字型下拉選單中文亂碼

這個問題其實也不是很大啦,不過當你有很多的中文字型檔的時候可能就不知道要選哪一個,啟動IReport後,開啟報表後會發現左邊下拉選單中,最下面的字型清單中有出現方框,顯示不出該字型的名稱,這幾個字型應該是判斷新細明體,標楷體及細明體,如下圖 下載IReport的Source Code來檢查一下,it.businesslogic.ireport.gui.MainFrame發現這個JComboBox有特別設定Arial字型,當然只要是中文的都顯示不出來ㄚ,所以點掉這一行後重新編譯,嘿嘿就可以了。 jComboBoxFont.setFont(new java.awt.Font("Arial", 0, 11)); 我目前使用的版本為 IReport-3.0.0-src