Singleton là một Design Pattern thuộc nhóm Creational Pattern. Nó đảm bảo chỉ duy nhất một thể hiện của một lớp được tạo ra trong suốt chương trình.
Trong nhiều trường hợp, cần đảm bảo chỉ có một thể hiện của một lớp. Ví dụ trong hệ thống quản lý người dùng, chỉ nên có duy nhất một đối tượng UserManager để quản lý người dùng.
Nếu tạo nhiều đối tượng UserManager có thể dẫn đến:
- Dữ liệu bị trùng lặp
- Xung đột tài nguyên
- Khó kiểm soát
graph TD
A[User Management System] --> B1[UserManager 1]
A --> B2[UserManager 2]
A --> B3[UserManager 3]
B1 --> C[User 1]
B2 --> D[User 2]
B3 --> E[User 3]
classDef grey fill:#dddddd,stroke:#333333,stroke-width:2px;
classDef pink fill:#ffd6d6,stroke:#333,stroke-width:2px;
class A grey
class B1,B2,B3 grey
class C,D,E pink
Singleton giải quyết bằng cách đảm bảo chỉ tạo duy nhất một thể hiện trong toàn bộ chương trình.
graph TD
A[User Management System] --> B[UserManager]
B --> C[User 1]
B --> D[User 2]
B --> E[User 3]
classDef grey fill:#dddddd,stroke:#333333,stroke-width:2px;
classDef pink fill:#ffd6d6,stroke:#333,stroke-width:2px;
class A grey
class B grey
class C,D,E pink
Giải thích:
- Lớp UserManager được triển khai Singleton
- Chỉ có DUY NHẤT một đối tượng UserManager trong hệ thống
- Quản lý tất cả người dùng một cách tập trung
- Tránh được các vấn đề như dữ liệu trùng lặp, xung đột tài nguyên, khó kiểm soát
Với cách triển khai này, chỉ có một đối tượng UserManager duy nhất được tạo ra, và đối tượng này có thể được truy cập từ bất kỳ nơi nào trong chương trình.
Singleton Pattern có cấu trúc đơn giản, bao gồm các thành phần sau:
classDiagram
direction TB
class Singleton {
-static instance:Singleton
-Singleton()
+static getInstance() : Singleton
}
class Client {
+main()
}
Client --> Singleton
Singleton ..> Singleton : return instance
- Lớp Singleton: Lớp này chứa các phương thức và biến cần thiết để triển khai Singleton Pattern.
- Phương thức khởi tạo private: Phương thức này chỉ có thể được gọi từ bên trong lớp.
- Biến static private: Biến này giữ đối tượng của lớp.
- Phương thức static public để trả về đối tượng của lớp: Phương thức này trả về đối tượng của lớp.
Có nhiều cách để triển khai Singleton Pattern trong Java. Một cách phổ biến là sử dụng một biến static private để lưu trữ instance của class.
public class Singleton {
private static Singleton instance;
private Singleton() {
// Constructor is private to prevent direct instantiation
}
public static Singleton getInstance() {
if (instance == null) {
instance = new Singleton();
}
return instance;
}
}
Cách triển khai này đảm bảo rằng chỉ có một instance của Singleton được tạo ra. Khi một đối tượng Singleton được yêu cầu, phương thức getInstance()
sẽ kiểm tra xem instance đã tồn tại hay chưa. Nếu chưa, phương thức sẽ tạo ra một instance mới. Nếu đã tồn tại, phương thức sẽ trả về instance hiện tại.
Một cách triển khai khác của Singleton Pattern là sử dụng một biến static final private.
public final class Singleton {
private static final Singleton instance = new Singleton();
private Singleton() {
// Constructor is private to prevent direct instantiation
}
public static Singleton getInstance() {
return instance;
}
}
Cách triển khai này tương tự như cách triển khai đầu tiên, nhưng nó sử dụng một biến static final private thay vì một biến static private. Cách triển khai này có một số ưu điểm như sau:
- Sử dụng biến static final private sẽ ngăn chặn việc thay đổi giá trị của biến instance.
- Cấu trúc code sẽ gọn gàng hơn.
Dưới đây là một ví dụ minh họa cách sử dụng Singleton Pattern để tạo một đối tượng DatabaseConnection.
public class DatabaseConnection {
private static final DatabaseConnection instance = new DatabaseConnection();
private DatabaseConnection() {
// Connect to database
}
public static DatabaseConnection getInstance() {
return instance;
}
public void query(String sql) {
// Execute query
}
}
Trong ví dụ này, DatabaseConnection là một class singleton. Nó có một phương thức getInstance() để truy cập instance duy nhất của class.
Singleton Pattern có thể được so sánh với một số Design Pattern tương tự, chẳng hạn như:
- Factory Pattern: Factory Pattern cung cấp một cách để tạo các đối tượng của lớp một cách linh hoạt. Tuy nhiên, Factory Pattern không đảm bảo rằng chỉ có một đối tượng của lớp được tạo ra.
- Prototype Pattern: Prototype Pattern cung cấp một cách để tạo các bản sao của đối tượng. Prototype Pattern cũng có thể được sử dụng để tạo một đối tượng duy nhất của lớp. Tuy nhiên, Prototype Pattern có thể phức tạp hơn Singleton Pattern.
Khi áp dụng Singleton Pattern, cần lưu ý một số điểm sau:
- Singleton Pattern có thể làm giảm tính linh hoạt của ứng dụng. Ví dụ, nếu bạn cần tạo ra nhiều instance của một class, bạn sẽ cần phải thay đổi code để xóa phương thức getInstance().
- Singleton Pattern có thể gây ra vấn đề khi test. Ví dụ, nếu bạn đang test một class sử dụng Singleton Pattern, bạn sẽ cần tạo ra một instance giả của class đó.
Singleton Pattern là một Design Pattern hữu ích trong những trường hợp cần đảm bảo rằng chỉ có một thể hiện duy nhất của một lớp được tạo ra. Tuy nhiên, cần lưu ý những điểm hạn chế của Singleton Pattern khi áp dụng.
Dưới đây là một số hướng dẫn sử dụng Singleton Pattern:
- Nên sử dụng Singleton Pattern khi cần đảm bảo rằng chỉ có một thể hiện duy nhất của một lớp được tạo ra.
- Tránh sử dụng Singleton Pattern khi không cần thiết.
- Hạn chế sử dụng Singleton trong các hệ thống lớn hoặc phức tạp.