SpringBoot----SpringBoot 整合Spring Data JPA(一)

一.Spring Data JPA介紹

二.SpringBoot 整合Spring Data JPA

1.修改pom文件,添加座標

		<!-- mysql數據庫驅動 -->
		<dependency>
			<groupId>mysql</groupId>
			<artifactId>mysql-connector-java</artifactId>
		</dependency>
		<!-- druid數據庫連接池 -->
		<dependency>
			<groupId>com.alibaba</groupId>
			<artifactId>druid</artifactId>
			<version>1.1.20</version>
		</dependency>

		<!-- spring data jpa的啓動器 -->
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-data-jpa</artifactId>
		</dependency>

2.在項目中添加application.properties文件

spring.datasource.driverClassName=com.mysql.cj.jdbc.Driver
spring.datasource.url=jdbc:mysql://localhost:3306/ssm?serverTimezone=UTC&useUnicode=true&characterEncoding=utf-8
spring.datasource.username=root
spring.datasource.password=root

spring.datasource.type=com.alibaba.druid.pool.DruidDataSource

spring.jpa.hibernate.ddl-auto=update
spring.jpa.show-sql=true

3.添加實體類

package com.kennosaur.pojo;

import java.io.Serializable;

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

@Entity
@Table(name = "t_users")
public class Users implements Serializable{
	@Id
	@GeneratedValue(strategy = GenerationType.IDENTITY)
	@Column(name = "id")
	private Integer id;
	@Column(name = "name")
	private String name;
	@Column(name = "age")
	private Integer age;
	@Column(name = "address")
	private String address;
	public Integer getId() {
		return id;
	}
	public void setId(Integer id) {
		this.id = id;
	}
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public Integer getAge() {
		return age;
	}
	public void setAge(Integer age) {
		this.age = age;
	}
	public String getAddress() {
		return address;
	}
	public void setAddress(String address) {
		this.address = address;
	}
	@Override
	public String toString() {
		return "Users [id=" + id + ", name=" + name + ", age=" + age + ", address=" + address + "]";
	}
	
	
	
}

5.編寫DAO接口

package com.kennosaur.dao;

import org.springframework.data.jpa.repository.JpaRepository;

import com.kennosaur.pojo.Users;

/**
 * 參數一T:當前需要映射的實體
 * 參數二ID:當前映射的實體中的OID的類型
 * @author Administrator
 *
 */
public interface UsersRepository extends JpaRepository<Users, Integer> {

}

6.在pom文件中添加測試啓動器的座標

		<!-- 添加junit環境的jar包 -->
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-test</artifactId>
		</dependency>

7.創建啓動類

package com.kennosaur;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;

@SpringBootApplication
public class App {
	public static void main(String[] args) {
		SpringApplication.run(App.class, args);
	}
}

8.編寫測試代碼

package com.kennosaur.test;

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.test.context.junit4.SpringJUnit4ClassRunner;

import com.kennosaur.App;
import com.kennosaur.dao.UsersRepository;
import com.kennosaur.pojo.Users;

/**
 * 測試類
 * @author Administrator
 *
 */

@RunWith(SpringJUnit4ClassRunner.class)
@SpringBootTest(classes = App.class)
public class UsersRepositoryTest {
	@Autowired
	private UsersRepository usersRepository;
	
	@Test
	public void testSave() {
		Users users = new Users();
		users.setAddress("北京市海淀");
		users.setAge(20);
		users.setName("張三");
		this.usersRepository.save(users);
	}
}

三.Spring Data JPA提供的核心接口

1.Repository接口

2.CrudRepository接口

3.PagingAndSortingRepository接口

4.JpaRepository接口

5.JPASpecificationExecutor接口

四.Repository接口的使用

1.提供了方法名稱命名查詢方式

1.1編寫接口

//方法的名稱必須要遵循駝峯式命名規則。findBy(關鍵字)+屬性名稱(首字母要大寫)+查詢條件(首字母大寫)

package com.kennosaur.dao;

import java.util.List;

import org.springframework.data.repository.Repository;

import com.kennosaur.pojo.Users;

/**
 * Repository接口的方法名稱命名查詢
 *
 *
 */
public interface UsersRepositoryByName extends Repository<Users, Integer> {

	//方法的名稱必須要遵循駝峯式命名規則。findBy(關鍵字)+屬性名稱(首字母要大寫)+查詢條件(首字母大寫)
	List<Users> findByName(String name);
	
	List<Users> findByNameAndAge(String name,Integer age);
	
	List<Users> findByNameLike(String name);
}

1.2測試代碼

package com.kennosaur.test;

import java.util.List;

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.test.context.junit4.SpringJUnit4ClassRunner;

import com.kennosaur.App;
import com.kennosaur.dao.UsersRepository;
import com.kennosaur.dao.UsersRepositoryByName;
import com.kennosaur.pojo.Users;

/**
 * 測試類
 * @author Administrator
 *
 */

@RunWith(SpringJUnit4ClassRunner.class)
@SpringBootTest(classes = App.class)
public class UsersRepositoryTest {
	
	@Autowired
	private UsersRepositoryByName usersRepositoryByName;
	
	
	/**
	 * Repository--方法名稱命名測試
	 */
	@Test
	public void testFindByName(){
		List<Users> list = this.usersRepositoryByName.findByName("張三");
		for (Users users : list) {
			System.out.println(users);
		}
	}
	
	/**
	 * Repository--方法名稱命名測試
	 */
	@Test
	public void testFindByNameAndAge(){
		List<Users> list = this.usersRepositoryByName.findByNameAndAge("張三", 20);
		for (Users users : list) {
			System.out.println(users);
		}
	}
	
	/**
	 * Repository--方法名稱命名測試
	 */
	@Test
	public void testFindByNameLike(){
		List<Users> list = this.usersRepositoryByName.findByNameLike("張%");
		for (Users users : list) {
			System.out.println(users);
		}
	}
}

2.提供了基於@Query註解查詢與更新

2.1編寫接口

@Query("update Users set name=:name where id=:id") --------注意使用?佔位符會報錯
@Modifying //需要執行一個更新操作

package com.kennosaur.dao;

import java.util.List;

import org.springframework.data.jpa.repository.Modifying;
import org.springframework.data.jpa.repository.Query;
import org.springframework.data.repository.Repository;

import com.kennosaur.pojo.Users;

public interface UsersRepositoryQueryAnnotation extends Repository<Users, Integer> {
	@Query("from Users where name=:name")
	List<Users> queryByNameUseHQL(String name);
	
	@Query(value="select * from t_users where name = ?",nativeQuery=true)
	List<Users> queryByNameUseSQL(String name);
	
	@Query("update Users set name=:name where id=:id")
	@Modifying //需要執行一個更新操作
	void updateUsersNameById(String name,Integer id);
}

2.2測試代碼

    @Transactional //@Transactional與@Test 一起使用時 事務是自動回滾的。
    @Rollback(false) //取消自動回滾

package com.kennosaur.test;

import java.util.List;

import javax.transaction.Transactional;

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.test.annotation.Rollback;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;

import com.kennosaur.App;
import com.kennosaur.dao.UsersRepository;
import com.kennosaur.dao.UsersRepositoryByName;
import com.kennosaur.dao.UsersRepositoryQueryAnnotation;
import com.kennosaur.pojo.Users;

/**
 * 測試類
 * @author Administrator
 *
 */

@RunWith(SpringJUnit4ClassRunner.class)
@SpringBootTest(classes = App.class)
public class UsersRepositoryTest {
	@Autowired
	private UsersRepositoryQueryAnnotation usersRepositoryQueryAnnotation;
	
	/**
	 * Repository--@Query測試
	 */
	@Test
	public void testQueryByNameUseHQL() {
		List<Users> list = this.usersRepositoryQueryAnnotation.queryByNameUseHQL("張三");
		for (Users users : list) {
			System.out.println(users);
		}
	}

	/**
	 * Repository--@Query測試
	 */
	@Test
	public void testQueryByNameUseSQL() {
		List<Users> list = this.usersRepositoryQueryAnnotation.queryByNameUseSQL("張三");
		for (Users users : list) {
			System.out.println(users);
		}
	}
	
	/**
	 * Repository--@Query測試
	 */
	@Test
	@Transactional //@Transactional與@Test 一起使用時 事務是自動回滾的。
	@Rollback(false) //取消自動回滾
	public void testUpdateUsersNameById() {
		this.usersRepositoryQueryAnnotation.updateUsersNameById("張三三", 1);
	}
	

	
	
}

五.CrudRepository接口-----主要作用是完成一些增刪改查的操作,CrudRepository接口繼承了Repository接口

1.編寫接口

package com.kennosaur.dao;

import org.springframework.data.repository.CrudRepository;

import com.kennosaur.pojo.Users;

/**
 * CrudRepository接口
 *
 *
 */
public interface UsersRepositoryCrudRepository extends CrudRepository<Users, Integer> {

}

2.測試代碼

注意:Optional<Users> users = this.usersRepositoryCrudRepository.findById(4);

	/**
	 * CrudRepository測試
	 */
	@Test
	public void testCrudRepositorySave() {
		Users user = new Users();
		user.setAddress("天津");
		user.setAge(32);
		user.setName("張三丰");
		this.usersRepositoryCrudRepository.save(user);
	}
	
	/**
	 * CrudRepository測試
	 */
	@Test
	public void testCrudRepositoryUpdate() {
		Users user = new Users();
		user.setId(4);
		user.setAddress("南京");
		user.setAge(40);
		user.setName("張三丰");
		this.usersRepositoryCrudRepository.save(user);
	}
	
	/**
	 * CrudRepository測試
	 */
	@Test
	public void testCrudRepositoryFindOne() {
		Optional<Users> users = this.usersRepositoryCrudRepository.findById(4);
		System.out.println(users);
	}
	
	/**
	 * CrudRepository測試
	 */
	@Test
	public void testCrudRepositoryFindAll() {
		List<Users> list  =  (List<Users>)this.usersRepositoryCrudRepository.findAll();
		for (Users users : list) {
			System.out.println(users);
		}
	}
	
	/**
	 * CrudRepository測試
	 */
	@Test
	public void testCrudRepositoryDeleteById() {
		this.usersRepositoryCrudRepository.deleteById(4);
		
	}

六.PagingAndSortingRepository接口-------提供了分頁與排序操作,該接口繼承了CrudRepository接口

1.編寫接口

package com.kennosaur.dao;

import org.springframework.data.repository.PagingAndSortingRepository;

import com.kennosaur.pojo.Users;
/**
 * 
 *PagingAndSortingRepository接口
 *
 */
public interface UsersRepositoryPagingAndSorting extends PagingAndSortingRepository<Users,Integer> {

}

2.測試代碼

注意:Sort sort = new Sort(order);報錯

Pageable pageable = new PageRequest(1, 2);報錯

Pageable pageable = new PageRequest(1, 2, sort);報錯

	/**
	 * PagingAndSortingRepository   排序測試
	 */
	@Test
	public void testPagingAndSortingRepositorySort() {
		//Order 定義排序規則
		Order order = new Order(Direction.DESC,"id");
		//Sort對象封裝了排序規則
//		Sort sort = new Sort(order);
		List<Users> list = (List<Users>)this.usersRepositoryPagingAndSorting.findAll(Sort.by(order));
		for (Users users : list) {
			System.out.println(users);
		}
	}
	
	/**
	 * PagingAndSortingRepository   分頁測試
	 */
	@Test
	public void testPagingAndSortingRepositoryPaging() {
		//Pageable:封裝了分頁的參數,當前頁,每頁顯示的條數。注意:他的當前頁是從0開始。
		//PageRequest(page,size) page:當前頁。size:每頁顯示的條數
//		Pageable pageable = new PageRequest(1, 2);
		Pageable pageable = PageRequest.of(1, 2);
	    Page<Users> page = this.usersRepositoryPagingAndSorting.findAll(pageable);
		System.out.println("總條數:"+page.getTotalElements());
		System.out.println("總頁數"+page.getTotalPages());
		List<Users> list = page.getContent();
		for (Users users : list) {
			System.out.println(users);
		}
	}
	
	/**
	 * PagingAndSortingRepository   排序+分頁
	 */
	@Test
	public void testPagingAndSortingRepositorySortAndPaging() {
		
//		Sort sort = new Sort(new Order(Direction.DESC, "id"));		
//		Pageable pageable = new PageRequest(1, 2, sort);
		
		Sort sort = Sort.by(new Order(Direction.DESC, "id"));		
		Pageable pageable = PageRequest.of(1, 2, sort);
				
		Page<Users> page = this.usersRepositoryPagingAndSorting.findAll(pageable);
		System.out.println("總條數:"+page.getTotalElements());
		System.out.println("總頁數"+page.getTotalPages());
		List<Users> list = page.getContent();
		for (Users users : list) {
			System.out.println(users);
		}
	}

 

七.JpaRepository接口----------開發中用的比較多.

繼承了PagingAndSortingRepository接口,對繼承的父接口中的方法的返回值進行適配(不需要做強制類型轉換)

1.編寫接口

package com.kennosaur.dao;

import org.springframework.data.jpa.repository.JpaRepository;

import com.kennosaur.pojo.Users;

/**
 * 參數一T:當前需要映射的實體
 * 參數二ID:當前映射的實體中的OID的類型
 * @author Administrator
 *
 */
public interface UsersRepository extends JpaRepository<Users, Integer> {

}

2.測試代碼

	/**
	 * JapRepository   排序測試
	 */
	@Test
	public void testJpaRepositorySort() {
		//Order 定義排序規則
		Order order = new Order(Direction.DESC,"id");
		//Sort對象封裝了排序規則
//		Sort sort = new Sort(order);
		List<Users> list = this.usersRepository.findAll(Sort.by(order));
		for (Users users : list) {
			System.out.println(users);
		}
	}

 

八.JPASpecificationExecutor接口----是單獨存在,完全獨立的(與上文中的四個接口不存在繼承關係)

該接口主要提供了多條件查詢的支持,並且可以在查詢中添加分頁和排序.通常與JpaRepository共同作用

1.編寫接口

package com.kennosaur.dao;

import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.data.jpa.repository.JpaSpecificationExecutor;

import com.kennosaur.pojo.Users;
/**
 * 
 *JpaSpecificationExecutor
 *
 */
public interface UsersRepositorySpecification extends JpaRepository<Users, Integer>, JpaSpecificationExecutor<Users> {

}

2.測試代碼

/**
	 * JpaSpecificationExecutor   單條件測試
	 */
	@Test
	public void testJpaSpecificationExecutor1() {
		
		/**
		 * Specification<Users>:用於封裝查詢條件
		 */
		Specification<Users> spec = new Specification<Users>() {
			
			//Predicate:封裝了 單個的查詢條件
			/**
			 * Root<Users> root:查詢對象的屬性的封裝。
			 * CriteriaQuery<?> query:封裝了我們要執行的查詢中的各個部分的信息,select  from order by
			 * CriteriaBuilder cb:查詢條件的構造器。定義不同的查詢條件
			 */
			@Override
			public Predicate toPredicate(Root<Users> root, CriteriaQuery<?> query, CriteriaBuilder cb) {
				// where name = '張三三'
				/**
				 * 參數一:查詢的條件屬性
				 * 參數二:條件的值
				 */
				Predicate pre = cb.equal(root.get("name"), "張三三");
				return pre;
			}
		};
		List<Users> list = this.usersRepositorySpecification.findAll(spec);
		for (Users users : list) {
			System.out.println(users);
		}
	}
	
	/**
	 * JpaSpecificationExecutor   多條件測試
	 */
	@Test
	public void testJpaSpecificationExecutor2() {
		
		/**
		 * Specification<Users>:用於封裝查詢條件
		 */
		Specification<Users> spec = new Specification<Users>() {
			
			//Predicate:封裝了 單個的查詢條件
			/**
			 * Root<Users> root:查詢對象的屬性的封裝。
			 * CriteriaQuery<?> query:封裝了我們要執行的查詢中的各個部分的信息,select  from order by
			 * CriteriaBuilder cb:查詢條件的構造器。定義不同的查詢條件
			 */
			@Override
			public Predicate toPredicate(Root<Users> root, CriteriaQuery<?> query, CriteriaBuilder cb) {
				// where name = '張三三' and age = 20				
			    List<Predicate> list = new ArrayList<>();
			    list.add(cb.equal(root.get("name"),"張三三"));
			    list.add(cb.equal(root.get("age"),20)); Predicate[] arr = new
			    Predicate[list.size()];				
				return cb.and(list.toArray(arr));
			}
		};
		List<Users> list = this.usersRepositorySpecification.findAll(spec);
		for (Users users : list) {
			System.out.println(users);
		}
	}

3.多條件查詢測試第二種寫法

/**
	 * JpaSpecificationExecutor   多條件測試第二種寫法
	 */
	@Test
	public void testJpaSpecificationExecutor3() {
		
		/**
		 * Specification<Users>:用於封裝查詢條件
		 */
		Specification<Users> spec = new Specification<Users>() {
			
			//Predicate:封裝了 單個的查詢條件
			/**
			 * Root<Users> root:查詢對象的屬性的封裝。
			 * CriteriaQuery<?> query:封裝了我們要執行的查詢中的各個部分的信息,select  from order by
			 * CriteriaBuilder cb:查詢條件的構造器。定義不同的查詢條件
			 */
			@Override
			public Predicate toPredicate(Root<Users> root, CriteriaQuery<?> query, CriteriaBuilder cb) {
				// where name = '張三三' and age = 20
				/*List<Predicate> list = new ArrayList<>();
				list.add(cb.equal(root.get("name"),"張三三"));
				list.add(cb.equal(root.get("age"),20));
				Predicate[] arr = new Predicate[list.size()];*/
				//(name = '張三' and age = 20) or id = 2
				return cb.or(cb.and(cb.equal(root.get("name"),"張三三"),cb.equal(root.get("age"),20)),cb.equal(root.get("id"), 2));
			}
		};
		
		Sort sort = Sort.by(new Order(Direction.DESC,"id"));
		List<Users> list = this.usersRepositorySpecification.findAll(spec,sort);
		for (Users users : list) {
			System.out.println(users);
		}
	}

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章