Chi tiết Spring Boot + Thymeleaf + MySQL + i18n
Tạo Database
script.sql
CREATE SCHEMA IF NOT EXISTS `todo_db` DEFAULT CHARACTER SET utf8mb4 ;
CREATE TABLE IF NOT EXISTS `todo_db`.`todo` (
`id` INT(11) NOT NULL AUTO_INCREMENT,
`title` VARCHAR(255) NULL DEFAULT NULL,
`detail` VARCHAR(255) NULL DEFAULT NULL,
PRIMARY KEY (`id`))
ENGINE = InnoDB
DEFAULT CHARACTER SET = utf8mb4;
Thêm 1 record vào DB
INSERT INTO `todo_db`.`todo` (`title`, `detail`) VALUES ('Làm bài tập', 'Hoàn thiện bài viết Spring Boot #13');
Cấu hình ứng dụng
Cấu hình là phần cực kì quan trọng rồi, chúng ta phải cung cấp cho Spring Boot các thông tin về Database và Thymeleaf. Ngoài ra, tùy chỉnh một số thông tin để giúp chúng ta lập trình đơn giản hơn. (Có thể đọc qua và tham khảo một số thông số có thể được sử dụng). Đây là application.properties
#Chạy ứng dụng trên port 8085
server.port=8085
# Bỏ tính năng cache của thymeleaf để lập trình cho nhanh
spring.thymeleaf.cache=false
# Các message tĩnh sẽ được lưu tại thư mục i18n
spring.messages.basename=i18n/messages
# Bỏ properties này đi khi deploy
# Nó có tác dụng cố định ngôn ngữ hiện tại chỉ là Tiếng Việt
spring.mvc.locale-resolver=fixed
# Mặc định ngôn ngữ là tiếng việt
spring.mvc.locale=vi_VN
# Đổi thành tiếng anh bằng cách bỏ comment ở dứoi
#spring.mvc.locale=en_US
spring.datasource.url=jdbc:mysql://localhost:3306/todo_db?useSSL=false
spring.datasource.username=root
spring.datasource.password=root
## Hibernate Properties
# The SQL dialect makes Hibernate generate better SQL for the chosen database
spring.jpa.properties.hibernate.dialect = org.hibernate.dialect.MySQL5InnoDBDialect
# Hibernate ddl auto (create, create-drop, validate, update)
spring.jpa.hibernate.ddl-auto = update
Tạo Model
Tạo model Todo
liên kết tới bảng todo
trong Database.
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import lombok.Data;
@Entity
@Data
public class Todo {
}
Ngoài ra, chúng ta tạo thêm một đối tượng là TodoValidator
, có trách nhiệm kiểm tra xem một object Todo
là hợp lệ hay không.
import org.thymeleaf.util.StringUtils;
/*
Đối tượng này dùng để kiểm tra xem một Object Todo có hợp lệ không
*/
public class TodoValidator {
/
* Kiểm tra một object Todo có hợp lệ không
* @param todo
* @return
*/
public boolean isValid(Todo todo) {
return Optional.ofNullable(todo)
.filter(t -> !StringUtils.isEmpty(t.getTitle())) // Kiểm tra title khác rỗng
.filter(t -> !StringUtils.isEmpty(t.getDetail())) // Kiểm tra detail khác rỗng
.isPresent(); // Trả về true nếu hợp lệ, ngược lại false
}
}
Vậy là xong phần chuẩn bị Model
.
TodoConfig
Trong ứng dụng của mình, tôi muốn tự tạo ra Bean TodoValidator
. Đây là lúc sử dụng @Configuration
và @Bean
đã học tại bài Spring Boot #6
config/TodoConfig.java
@Configuration
public class TodoConfig {
/
* Tạo ra Bean TodoValidator để sử dụng sau này
* @return
*/
@Bean
public TodoValidator validator() {
return new TodoValidator();
}
}
Tầng Repository
Tầng Repository, chịu trách nhiệm giao tiếp với Database. Chúng ta sử dụng Spring JPA. repository/TodoRepository.java
@Repository
public interface TodoRepository extends JpaRepository<Todo, Long> {
}
Tầng Service
Tầng Service, chị trách nhiệm thực hiện các xử lý logic, business, hỗ trợ cho tầng Controller. service/TodoService.java
@Service
public class TodoService {
@Autowired
private TodoRepository todoRepository;
@Autowired
private TodoValidator validator;
/
* Lấy ra danh sách List<Todo>
*
* @param limit - Giới hạn số lượng lấy ra
*
* @return Trả về danh sách List<Todo> dựa theo limit, nếu limit == null thì trả findAll()
*/
public List<Todo> findAll(Integer limit) {
return Optional.ofNullable(limit)
.map(value -> todoRepository.findAll(PageRequest.of(0, value)).getContent())
.orElseGet(() -> todoRepository.findAll());
}
/
* Thêm một Todo mới vào danh sách công việc cần làm
*
* @return Trả về đối tượng Todo sau khi lưu vào DB, trả về null nếu không thành công
*/
public Todo add(Todo todo) {
if (validator.isValid(todo)) {
return todoRepository.save(todo);
}
return null;
}
}
Tầng Controller
Tầng Controller, nơi đón nhận các request từ phía người dùng, và chuyển tiếp xử lý xuống tầng Service.
controller/TodoController.java
@Controller
public class TodoController {
@Autowired
private TodoService todoService;
/*
@RequestParam dùng để đánh dấu một biến là request param trong request gửi lên server.
Nó sẽ gán dữ liệu của param-name tương ứng vào biến
*/
@GetMapping("/listTodo")
public String index(Model model, @RequestParam(value = "limit", required = false) Integer limit) {
// Trả về đối tượng todoList.
model.addAttribute("todoList", todoService.findAll(limit));
// Trả về template "listTodo.html"
return "listTodo";
}
@GetMapping("/addTodo")
public String addTodo(Model model) {
model.addAttribute("todo", new Todo());
return "addTodo";
}
/*
@ModelAttribute đánh dấu đối tượng Todo được gửi lên bởi Form Request
*/
@PostMapping("/addTodo")
public String addTodo(@ModelAttribute Todo todo) {
return Optional.ofNullable(todoService.add(todo))
.map(t -> "success") // Trả về success nếu save thành công
.orElse("failed"); // Trả về failed nếu không thành công
}
}
Templates
Tầng Controller đã trả về templates, nhiệm vụ tiếp theo là sử dụng Template Engine để xử lý các templates này và trả về webpage cho người dùng.
i18n
Trong các template, tôi có sử dụng các message tĩnh, những message này hỗ trợ đa ngôn ngữ. Chúng ta định nghĩa các message này tại thư mục i18n
.
i18n/messages_vi.properties
loda.message.hello=Welcome to TodoApp
loda.message.success=Thêm Todo thành công!
loda.message.failed=Thêm Todo không thành công!
loda.value.addTodo=Thêm công việc
loda.value.viewListTodo=Xem danh sách công việc
loda.value.listTodo=Danh sách công việc
i18n/messages_en.properties
loda.message.hello=Welcome to TodoApp
loda.message.success=Add To-do Successfully!
loda.message.failed=Add To-do Failed!
loda.value.addTodo=Add To-do
loda.value.viewListTodo=View To-do list
loda.value.listTodo=To-do list