Hướng dẫn Java Design Pattern – Transfer Object

1. Transfer Object Pattern là gì?

Transfer Object/ Data Transfer Object Pattern là một dạng Architectural Design Pattern, được sử dụng khi chúng ta muốn truyền dữ liệu qua lại giữa các tầng trong ứng dụng, giữa Client – Server. Data Transfer Object (DTO) còn được gọi là Value Object (VO).

Transfer Object đơn giản là một POJO (Plain Old Java Object), chỉ chứa các getter/ setter method và có thể có implement serialize để truyền tải dữ liệu thông qua network.

DTO hoàn toàn không chứa behavior/ logic, chỉ được sử dụng để truyền dữ liệu và map dữ liệu từ các Domain Model trước khi truyền tới Client. Trong các ứng dụng đơn giản, các Domain Model thường có thể được sử dụng lại trực tiếp dưới dạng DTO và được truyền trực tiếp đến lớp hiển thị, do đó chỉ có một Data Model thống nhất. Đối với các ứng dụng phức tạp hơn, chúng ta không muốn hiển thị toàn bộ Domain Model cho Client, do đó, việc ánh xạ từ các Domain Model sang DTO là cần thiết.

2. Cài đặt Transfer Object Pattern như thế nào?

Các thành phần tham gia Transfer Object Pattern:

  • Business Object : là một Business Service, tạo Transfer Object và trả nó về Client khi cần thiết. Nó cũng có thể nhận dữ liệu từ Client trong một Transfer Object và gửi đến Server để cập nhật vào database.
  • Transfer Object : là một POJO, chỉ chứa các getter/ setter method.
  • Client : người sử dụng ứng dụng.

2.1. Ví dụ sử dụng Transfer Object Pattern

Lớp xử lý nghiệp vụ ở phía Server thường truy vấn dữ liệu từ database và gán các giá trị vào Transfer Object để gửi lại Client. Phía Client có thể tạo một Transfer Object và gán giá trị vào để gửi lại Server thực hiện update vào database.

Trong ví dụ bên dưới, chúng ta sẽ cùng tìm hiểu cách áp dụng DTO với DAO ở bài viết trước.

UserModel.java

package com.maixuanviet.patterns.other.dto;
 
import lombok.Data;
 
/**
 * Domain Model / Entity
 */
@Data
public class UserModel {
 
    private Integer id;
    private String userName;
    private String fullName;
    private String password;
    private String email;
    private String bankAccount;
}

Dao.java

package com.maixuanviet.patterns.other.dto;
 
import java.util.List;
import java.util.Optional;
 
/**
 * Data Access Object
 */
public interface Dao<T> {
 
    List<T> getAll();
 
    Optional<T> get(Integer id);
 
    void save(T t);
 
    void update(T t);
 
    void delete(T t);
}

UserDao.java

package com.maixuanviet.patterns.other.dto;
 
import java.util.ArrayList;
import java.util.List;
import java.util.Optional;
 
public class UserDao implements Dao<UserModel> {
 
    // Temporary database
    private List<UserModel> users = new ArrayList<>();
 
    public UserDao() {
        UserModel user = new UserModel();
        user.setId(1);
        user.setUserName("maixuanviet");
        user.setEmail("maixuanvietvn@gmail.com");
        user.setFullName("GP Coder");
        user.setPassword("1234567");
        user.setBankAccount("9999-9999-9999");
        users.add(user);
    }
 
    @Override
    public List<UserModel> getAll() {
        return users;
    }
 
    @Override
    public Optional<UserModel> get(Integer id) {
        return users.stream().filter(u -> u.getId() == id).findFirst();
    }
 
    @Override
    public void save(UserModel user) {
        users.add(user);
    }
 
    @Override
    public void update(UserModel user) {
        int index = -1;
        for (UserModel u : users) {
            index++;
            if (user.getId().equals(u.getId())) {
                users.set(index, user);
                break;
            }
        }
    }
 
    @Override
    public void delete(UserModel user) {
        get(user.getId()).ifPresent(existUser -> users.remove(existUser));
    }
}

UserDTO.java

package com.maixuanviet.patterns.other.dto;
 
import lombok.Data;
 
/**
 * Data Transfer Object
 */
@Data
public class UserDTO {
 
    private Integer id;
    private String userName;
    private String fullName;
    private String email;
}

UserService.java

package com.maixuanviet.patterns.other.dto;
 
/**
 * Business Object / Logic
 */
public class UserService {
 
    private UserDao dao = new UserDao();
 
    public UserDTO getUser(Integer id) {
        UserModel model = dao.get(id).get();
        return convertToDTO(model);
    }
 
    public void saveUser(UserDTO dto) {
        UserModel model = convertToModel(dto);
        dao.save(model);
    }
 
    public void updateUser(UserDTO dto) {
        UserModel model = convertToModel(dto);
        dao.update(model);
    }
 
    private UserModel convertToModel(UserDTO dto) {
        UserModel model = new UserModel();
        model.setId(dto.getId());
        model.setFullName(dto.getFullName());
        model.setUserName(dto.getUserName());
        model.setEmail(dto.getEmail());
        return model;
    }
 
    private UserDTO convertToDTO(UserModel model) {
        UserDTO dto = new UserDTO();
        dto.setId(model.getId());
        dto.setFullName(model.getFullName());
        dto.setUserName(model.getUserName());
        dto.setEmail(model.getEmail());
        return dto;
    }
}

DataAccessObjectPatternExample.java

package com.maixuanviet.patterns.other.dto;
 
/**
 * Client - Data Access Object Pattern Example
 */
public class DataAccessObjectPatternExample {
 
    public static void main(String[] args) {
        UserService service = new UserService();
        UserDTO dto = service.getUser(1);
        System.out.println("User: " + dto);
 
        dto.setFullName("maixuanviet.com");
        service.updateUser(dto);
        System.out.println("User Updated: " + dto);
    }
}

Output của chương trình:

User: UserDTO(id=1, userName=maixuanviet, fullName=Mai Xuan Viet, email=maixuanviet.com@gmail.com)
User Updated: UserDTO(id=1, userName=maixuanviet, fullName=maixuanviet.com, email=maixuanviet.com@gmail.com)

3. Lợi ích của Transfer Object Pattern là gì?

  • Tách biệt logic một cách rõ ràng : Transfer Object chỉ chứa data, còn logic được implement trong phần khác.
  • Cãi thiện hiệu suất ứng dụng : chi phí của mỗi request/ response là lớn, chúng ta nên cố gắng gửi nhiều nhất có thể. Để làm điều này, chúng ta có thể tạo một Transfer Object để gửi data từ Client lên Server hay từ Server đến Client một lần duy nhất, thay vì phải gửi từng phần riêng lẻ.
  • Giảm kết dính giữa các tầng trong ứng dụng: Client chỉ thao tác với Transfer Object, nên nó không bị ảnh hưởng khi Domain Model thay đổi.
  • Bao đóng các đối số : một phương thức có nhiều đối số, chúng ta có thể bao đóng chúng trong một Transfer Object. Giúp chúng ta dễ dàng mở rộng, thêm/ bớt đối số.
  • Nhận nhiều dữ liệu trả về : trong Java, một phương thức chỉ có thể trả về một giá trị, để có thể nhận được nhiều giá trị, chúng ta có thể bao đóng chúng trong một Transfer Object.
  • Tăng bảo mật ứng dụng : tùy vào người dùng khác nhau có thể xem được một số dữ liệu nhất định. Chúng ta có thể tạo nhiều Transfer Object khác nhau cho từng loại người dùng thay vì trả về một Domain Object một cách trực tiếp. Trường hợp rõ ràng nhất là User Model, domain object này chứa thông tin cả email, password, số tài khoản ngân hàng. Chúng ta có thể tạo một Transfer Object đơn giản chỉ chứa thông tin họ tên, ngày sinh. Không cần thiết phải trả tất cả dữ liệu Domain Model về Client.
  • Transfer Object thường được sử dụng với Data Access Object .

Related posts:

Java Program to Check Whether an Input Binary Tree is the Sub Tree of the Binary Tree
ETags for REST with Spring
Compare Two JSON Objects with Jackson
Toán tử instanceof trong java
Tránh lỗi NullPointerException trong Java như thế nào?
Java Program to Implement Dijkstra’s Algorithm using Set
Java Program to Implement K Way Merge Algorithm
Java Program to Find ith Largest Number from a Given List Using Order-Statistic Algorithm
Java Program to add two large numbers using Linked List
Merging Streams in Java
Spring Boot with Multiple SQL Import Files
Hướng dẫn Java Design Pattern – Null Object
Create a Custom Exception in Java
List Interface trong Java
Guide to WeakHashMap in Java
Java Program to Generate All Subsets of a Given Set in the Lexico Graphic Order
Java Program to implement Priority Queue
Các chương trình minh họa sử dụng Cấu trúc điều khiển trong Java
Java Program to Implement AA Tree
Finding Max/Min of a List or Collection
Call Methods at Runtime Using Java Reflection
Java Program to Encode a Message Using Playfair Cipher
Java Program to Check whether Graph is a Bipartite using 2 Color Algorithm
Spring Boot - Interceptor
A Guide to Java 9 Modularity
Java Program to Perform integer Partition for a Specific Case
Hướng dẫn Java Design Pattern – Interpreter
Spring Boot - Tomcat Deployment
How to Get All Dates Between Two Dates?
Java Program to Implement Coppersmith Freivald’s Algorithm
Copy a List to Another List in Java
Java Program to Check if an UnDirected Graph is a Tree or Not Using DFS