JPA - JPQL

Chương này cho bạn biết về JPQL và cách nó hoạt động với các đơn vị bền vững. Trong chương này, các ví dụ tuân theo cùng một cấu trúc phân cấp gói mà chúng ta đã sử dụng trong chương trước như sau:

Ngôn ngữ truy vấn bền vững Java

JPQL là Ngôn ngữ truy vấn bền vững của Java được định nghĩa trong đặc tả JPA. Nó được sử dụng để tạo các truy vấn chống lại các thực thể để lưu trữ trong cơ sở dữ liệu quan hệ. JPQL được phát triển dựa trên cú pháp SQL. Nhưng nó sẽ không ảnh hưởng trực tiếp đến cơ sở dữ liệu.

JPQL có thể truy xuất thông tin hoặc dữ liệu bằng mệnh đề SELECT, có thể cập nhật hàng loạt bằng mệnh đề UPDATE và mệnh đề DELETE. API EntityManager.createQuery () sẽ hỗ trợ ngôn ngữ truy vấn.

Cấu trúc truy vấn

Cú pháp JPQL rất giống với cú pháp của SQL. Có cú pháp giống SQL là một lợi thế vì SQL là ngôn ngữ truy vấn có cấu trúc đơn giản và nhiều nhà phát triển đang sử dụng nó trong các ứng dụng. SQL hoạt động trực tiếp với các bảng, bản ghi và trường cơ sở dữ liệu quan hệ, trong khi JPQL hoạt động với các lớp và phiên bản Java.

Ví dụ: một truy vấn JPQL có thể truy xuất một đối tượng thực thể hơn là tập kết quả trường từ cơ sở dữ liệu, như với SQL. Cấu trúc truy vấn JPQL như sau.

SELECT ... FROM ...
[WHERE ...]
[GROUP BY ... [HAVING ...]]
[ORDER BY ...]

Cấu trúc của các truy vấn JPQL DELETE và UPDATE đơn giản hơn như sau.

DELETE FROM ... [WHERE ...]
 
UPDATE ... SET ... [WHERE ...]

Các hàm vô hướng và tổng hợp

Hàm vô hướng trả về giá trị kết quả dựa trên giá trị đầu vào. Các hàm tổng hợp trả về các giá trị kết quả bằng cách tính toán các giá trị đầu vào.

Làm theo cùng một ví dụ về quản lý nhân viên được sử dụng trong các chương trước. Ở đây chúng ta sẽ đi qua các lớp dịch vụ sử dụng hàm vô hướng và tổng hợp của JPQL.

Giả sử bảng jpadb.employee chứa các bản ghi sau.

Eid Tên Tiền lương Độ
1201 Gopal 40000 Giám đốc kỹ thuật
1202 Manisha 40000 Trình đọc bằng chứng
1203 Masthanvali 40000 Người viết kỹ thuật
1204 Hài lòng 30000 Người viết kỹ thuật
1205 Krishna 30000 Người viết kỹ thuật
1206 Kiran 35000 Trình đọc bằng chứng

Tạo một lớp có tên ScalarandAggregateFunctions.java Dưới com.tutorialspoint.eclipselink.service gói như sau.

package com.tutorialspoint.eclipselink.service;

import java.util.List;
import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import javax.persistence.Persistence;
import javax.persistence.Query;

public class ScalarandAggregateFunctions {
   public static void main( String[ ] args ) {
   
      EntityManagerFactory emfactory = Persistence.createEntityManagerFactory( "Eclipselink_JPA" );
      EntityManager entitymanager = emfactory.createEntityManager();

      //Scalar function
      Query query = entitymanager.
      createQuery("Select UPPER(e.ename) from Employee e");
      List<String> list = query.getResultList();

      for(String e:list) {
         System.out.println("Employee NAME :"+e);
      }
      
      //Aggregate function
      Query query1 = entitymanager.createQuery("Select MAX(e.salary) from Employee e");
      Double result = (Double) query1.getSingleResult();
      System.out.println("Max Employee Salary :" + result);
   }
}

Sau khi biên dịch và thực thi chương trình trên, bạn sẽ nhận được kết quả đầu ra trong bảng điều khiển của Eclipse IDE như sau:

Employee NAME :GOPAL
Employee NAME :MANISHA
Employee NAME :MASTHANVALI
Employee NAME :SATISH
Employee NAME :KRISHNA
Employee NAME :KIRAN
ax Employee Salary :40000.0

Từ khóa Giữa, Và, Giống như

'Giữa', 'Và' và 'Thích' là các từ khóa chính của JPQL. Các từ khóa này được sử dụng sau mệnh đề Where trong một truy vấn.

Tạo một lớp có tên BetweenAndLikeFunctions.java Dưới com.tutorialspoint.eclipselink.service gói như sau:

package com.tutorialspoint.eclipselink.service;

import java.util.List;
import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import javax.persistence.Persistence;
import javax.persistence.Query;
import com.tutorialspoint.eclipselink.entity.Employee;

public class BetweenAndLikeFunctions {
   public static void main( String[ ] args ) {
   
      EntityManagerFactory emfactory = Persistence.createEntityManagerFactory( "Eclipselink_JPA" );
      EntityManager entitymanager = emfactory.createEntityManager();
      
      //Between
      Query query = entitymanager.createQuery( "Select e " + "from Employee e " + "where e.salary " + "Between 30000 and 40000" );
      
      List<Employee> list=(List<Employee>)query.getResultList( );

      for( Employee e:list ){
         System.out.print("Employee ID :" + e.getEid( ));
         System.out.println("\t Employee salary :" + e.getSalary( ));
      }

      //Like
      Query query1 = entitymanager.createQuery("Select e " + "from Employee e " + "where e.ename LIKE 'M%'");
      
      List<Employee> list1=(List<Employee>)query1.getResultList( );
      
      for( Employee e:list1 ) {
         System.out.print("Employee ID :"+e.getEid( ));
         System.out.println("\t Employee name :"+e.getEname( ));
      }
   }
}

Sau khi biên dịch và thực thi chương trình trên, bạn sẽ nhận được kết quả đầu ra trong bảng điều khiển của Eclipse IDE như sau:

Employee ID :1201	 Employee salary :40000.0
Employee ID :1202	 Employee salary :40000.0
Employee ID :1203	 Employee salary :40000.0
Employee ID :1204	 Employee salary :30000.0
Employee ID :1205	 Employee salary :30000.0
Employee ID :1206	 Employee salary :35000.0

Employee ID :1202	 Employee name :Manisha
Employee ID :1203	 Employee name :Masthanvali

Đặt hàng

Để đặt hàng các bản ghi trong JPQL, chúng tôi sử dụng mệnh đề ORDER BY. Việc sử dụng mệnh đề này giống như cách sử dụng trong SQL, nhưng nó liên quan đến các thực thể. Làm theo đơn đặt hàng bằng ví dụ.

Tạo một lớp Or Order.java trong com.tutorialspoint.eclipselink.service gói như sau:

package com.tutorialspoint.eclipselink.service;

import java.util.List;
import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import javax.persistence.Persistence;
import javax.persistence.Query;
import com.tutorialspoint.eclipselink.entity.Employee;

public class Ordering {

   public static void main( String[ ] args ) {
      EntityManagerFactory emfactory = Persistence.createEntityManagerFactory( "Eclipselink_JPA" );
      EntityManager entitymanager = emfactory.createEntityManager();
      
      //Between
      Query query = entitymanager.createQuery( "Select e " + "from Employee e " + "ORDER BY e.ename ASC" );

      List<Employee> list = (List<Employee>)query.getResultList( );

      for( Employee e:list ) {
         System.out.print("Employee ID :" + e.getEid( ));
         System.out.println("\t Employee Name :" + e.getEname( ));
      }
   }
}

Sau khi biên dịch và thực thi chương trình trên, bạn sẽ nhận được kết quả đầu ra trong bảng điều khiển của Eclipse IDE như sau:

Employee ID :1201	 Employee Name :Gopal
Employee ID :1206	 Employee Name :Kiran
Employee ID :1205	 Employee Name :Krishna
Employee ID :1202	 Employee Name :Manisha
Employee ID :1203	 Employee Name :Masthanvali
Employee ID :1204	 Employee Name :Satish

Truy vấn được đặt tên

Chú thích @NamedQuery được định nghĩa là một truy vấn có chuỗi truy vấn không thể thay đổi được xác định trước. Thay vì truy vấn động, việc sử dụng các truy vấn được đặt tên có thể cải thiện tổ chức mã bằng cách tách các chuỗi truy vấn JPQL khỏi POJO. Nó cũng chuyển các tham số truy vấn thay vì nhúng các ký tự động vào chuỗi truy vấn và dẫn đến các truy vấn hiệu quả hơn.

Trước hết, hãy thêm chú thích @NamedQuery vào lớp thực thể Nhân viên có tên Employee.java Dưới com.tutorialspoint.eclipselink.entity gói như sau:

package com.tutorialspoint.eclipselink.entity;

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

@Entity
@Table
@NamedQuery(query = "Select e from Employee e where e.eid = :id", name = "find employee by id")

public class Employee {
   @Id
   @GeneratedValue(strategy = GenerationType.AUTO) 	
   
   private int eid;
   private String ename;
   private double salary;
   private String deg;
   
   public Employee(int eid, String ename, double salary, String deg) {
      super( );
      this.eid = eid;
      this.ename = ename;
      this.salary = salary;
      this.deg = deg;
   }
   
   public Employee( ) {
      super();
   }

   public int getEid( ) {
      return eid;
   }
   
   public void setEid(int eid) {
      this.eid = eid;
   }

   public String getEname( ) {
      return ename;
   }
   
   public void setEname(String ename) {
      this.ename = ename;
   }

   public double getSalary( ) {
      return salary;
   }
   
   public void setSalary(double salary) {
      this.salary = salary;
   }

   public String getDeg( ) {
      return deg;
   }
   
   public void setDeg(String deg) {
      this.deg = deg;
   }
   
   @Override
   public String toString() {
      return "Employee [eid=" + eid + ", ename=" + ename + ", salary=" + salary + ", deg=" + deg + "]";
   }
}

Tạo một lớp có tên NamedQueries.java Dưới com.tutorialspoint.eclipselink.service gói như sau:

package com.tutorialspoint.eclipselink.service;

import java.util.List;
import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import javax.persistence.Persistence;
import javax.persistence.Query;
import com.tutorialspoint.eclipselink.entity.Employee;

public class NamedQueries {
   public static void main( String[ ] args ) {
   
      EntityManagerFactory emfactory = Persistence.createEntityManagerFactory( "Eclipselink_JPA" );
      EntityManager entitymanager = emfactory.createEntityManager();
      Query query = entitymanager.createNamedQuery("find employee by id");
      
      query.setParameter("id", 1204);
      List<Employee> list = query.getResultList( );
      
      for( Employee e:list ){
         System.out.print("Employee ID :" + e.getEid( ));
         System.out.println("\t Employee Name :" + e.getEname( ));
      }
   }
}

Sau khi biên dịch và thực thi chương trình trên, bạn sẽ nhận được kết quả đầu ra trong bảng điều khiển của Eclipse IDE như sau:

Employee ID :1204	 Employee Name :Satish

Sau khi thêm tất cả các lớp trên, hệ thống phân cấp gói được hiển thị như sau:

Đang tải háo hức và lười biếng

Khái niệm chính của JPA là tạo một bản sao của cơ sở dữ liệu trong bộ nhớ đệm. Trong khi giao dịch với cơ sở dữ liệu, đầu tiên nó sẽ ảnh hưởng đến dữ liệu trùng lặp và chỉ khi nó được cam kết sử dụng trình quản lý thực thể, các thay đổi mới được thực hiện vào cơ sở dữ liệu.

Có hai cách tìm nạp bản ghi từ cơ sở dữ liệu - tìm nạp háo hức và tìm nạp lười biếng.

Tìm nạp háo hức

Tìm nạp toàn bộ bản ghi trong khi tìm bản ghi bằng Khóa chính.

Tìm nạp lười biếng

Nó kiểm tra tính khả dụng của thông báo bằng khóa chính nếu nó tồn tại. Sau đó, nếu bạn gọi bất kỳ phương thức getter nào của thực thể đó thì nó sẽ tìm nạp toàn bộ.

Nhưng có thể tìm nạp lười biếng khi bạn cố gắng tìm nạp bản ghi lần đầu tiên. Bằng cách đó, một bản sao của toàn bộ bản ghi đã được lưu trữ trong bộ nhớ đệm. Hiệu suất khôn ngoan, tìm nạp lười biếng là thích hợp hơn.