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:

RegEx for matching Date Pattern in Java
Java Program to Implement Slicker Algorithm that avoids Triangulation to Find Area of a Polygon
Java Program to Implement wheel Sieve to Generate Prime Numbers Between Given Range
Java Program to Implement Jarvis Algorithm
Using Spring ResponseEntity to Manipulate the HTTP Response
Guide to Spring @Autowired
A Guide to EnumMap
The Java 8 Stream API Tutorial
The Dining Philosophers Problem in Java
Java Program to Generate a Random UnDirected Graph for a Given Number of Edges
How to Manually Authenticate User with Spring Security
Java Stream Filter with Lambda Expression
Serialization và Deserialization trong java
Test a REST API with Java
Abstract class và Interface trong Java
Jackson – Decide What Fields Get Serialized/Deserialized
New Features in Java 15
Java Program to Implement Dijkstra’s Algorithm using Priority Queue
Java Program to Find Location of a Point Placed in Three Dimensions Using K-D Trees
Xử lý ngoại lệ đối với trường hợp ghi đè phương thức trong java
DistinctBy in the Java Stream API
A Guide to Java HashMap
Guava CharMatcher
Guide to CopyOnWriteArrayList
Guide to @ConfigurationProperties in Spring Boot
Java Program to Perform integer Partition for a Specific Case
Java Program to Find Shortest Path Between All Vertices Using Floyd-Warshall’s Algorithm
Giới thiệu Swagger – Công cụ document cho RESTfull APIs
Tính đa hình (Polymorphism) trong Java
Hướng dẫn Java Design Pattern – Dependency Injection
Tìm hiểu cơ chế Lazy Evaluation của Stream trong Java 8
Java Program to implement Bit Set