SpringBoot結(jié)合HTMX實現(xiàn)高效Web開發(fā)實戰(zhàn)
在當今的 Web 開發(fā)領(lǐng)域,前后端分離已成為主流趨勢。
傳統(tǒng)的全??蚣芡枰獜?fù)雜的模板引擎來處理視圖邏輯,而前端框架如 React、Vue 等雖然強大,但也帶來了學習曲線陡峭、構(gòu)建復(fù)雜等問題。
本文將介紹一種輕量級的解決方案 —— 結(jié)合 Spring Boot 與 HTMX,實現(xiàn)高效、簡潔的前后端分離開發(fā)。
為什么選擇 SpringBoot 與 HTMX?
Spring Boot 是 Java 生態(tài)中最流行的應(yīng)用開發(fā)框架之一,它提供了自動配置、嵌入式服務(wù)器等特性,讓開發(fā)者可以快速搭建企業(yè)級應(yīng)用。
而 HTMX 是一個輕量級的 JavaScript 庫,它允許你使用 HTML 屬性直接與服務(wù)器進行 AJAX 通信,無需編寫大量的 JavaScript 代碼。
這種組合既保留了傳統(tǒng) HTML 的簡單性,又具備現(xiàn)代 Web 應(yīng)用的交互性。
項目架構(gòu)概述
我們將構(gòu)建一個簡單的任務(wù)管理應(yīng)用,采用前后端完全分離的架構(gòu):
- 后端:Spring Boot REST API
- 前端:純 HTML + HTMX + doT.js + Tailwind CSS
這種架構(gòu)使得前后端可以獨立開發(fā)、測試和部署,同時保持高效的通信和良好的用戶體驗。
后端實現(xiàn)
首先,讓我們創(chuàng)建 SpringBoot 后端
package com.example.taskmanager;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.*;
import java.util.ArrayList;
import java.util.List;
import java.util.UUID;
@SpringBootApplication
@RestController
@RequestMapping("/api/tasks")
public class TaskManagerApplication {
// 內(nèi)存中的任務(wù)存儲
private List<Task> tasks = new ArrayList<>();
public static void main(String[] args) {
SpringApplication.run(TaskManagerApplication.class, args);
}
// 任務(wù)模型
record Task(String id, String title, boolean completed) {}
// 獲取所有任務(wù)
@GetMapping
public List<Task> getAllTasks() {
return tasks;
}
// 創(chuàng)建新任務(wù)
@PostMapping
public Task createTask(@RequestBody Task task) {
Task newTask = new Task(UUID.randomUUID().toString(), task.title(), false);
tasks.add(newTask);
return newTask;
}
// 更新任務(wù)狀態(tài)
@PutMapping("/{id}/toggle")
public Task toggleTask(@PathVariable String id) {
Task task = tasks.stream()
.filter(t -> t.id().equals(id))
.findFirst()
.orElseThrow(() -> new IllegalArgumentException("Task not found"));
Task updatedTask = new Task(task.id(), task.title(), !task.completed());
tasks.remove(task);
tasks.add(updatedTask);
return updatedTask;
}
// 刪除任務(wù)
@DeleteMapping("/{id}")
public void deleteTask(@PathVariable String id) {
tasks.removeIf(t -> t.id().equals(id));
}
}
這個后端實現(xiàn)了基本的 CRUD 操作:獲取任務(wù)列表、創(chuàng)建新任務(wù)、切換任務(wù)狀態(tài)和刪除任務(wù)。
為了方便演示和快速能夠運行DEOM,所有數(shù)據(jù)都存儲在內(nèi)存中的 List 中。
前端實現(xiàn)
接下來是前端部分,我們將使用 HTMX 來處理與后端的交互
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Task Manager</title>
<script src="https://cdn.tailwindcss.com"></script>
<link rel="external nofollow" rel="stylesheet">
<script src="https://unpkg.com/htmx.org@1.9.6/dist/htmx.min.js"></script>
<script src="https://unpkg.com/htmx.org/dist/ext/json-enc.js"></script>
<script src="https://cdn.jsdelivr.net/gh/olado/doT@master/doT.min.js"></script>
<!-- Tailwind 配置 -->
<script>
tailwind.config = {
theme: {
extend: {
colors: {
primary: '#3B82F6',
secondary: '#10B981',
danger: '#EF4444',
dark: '#1F2937',
},
fontFamily: {
sans: ['Inter', 'system-ui', 'sans-serif'],
},
}
}
}
</script>
<style type="text/tailwindcss">
@layer utilities {
.content-auto {
content-visibility: auto;
}
.task-item {
@apply flex items-center justify-between p-4 mb-3 rounded-lg transition-all duration-300;
}
.task-item-complete {
@apply bg-gray-100 text-gray-500 line-through;
}
.btn {
@apply px-4 py-2 rounded-lg font-medium transition-all duration-200;
}
.btn-primary {
@apply bg-primary text-white hover:bg-primary/90;
}
.btn-danger {
@apply bg-danger text-white hover:bg-danger/90;
}
.btn-secondary {
@apply bg-secondary text-white hover:bg-secondary/90;
}
.animate-fade-in {
@apply opacity-0 transform translate-y-2;
animation: fadeIn 0.3s ease-out forwards;
}
.animate-fade-out {
@apply opacity-100 transform translate-y-0;
animation: fadeOut 0.3s ease-in forwards;
}
@keyframes fadeIn {
to { opacity: 1; transform: translateY(0); }
}
@keyframes fadeOut {
to { opacity: 0; transform: translateY(-10px); }
}
}
</style>
</head>
<body class="bg-gray-50 font-sans text-dark">
<div class="max-w-3xl mx-auto px-4 py-8">
<!-- 頁面標題 -->
<header class="text-center mb-8">
<h1 class="text-[clamp(2rem,5vw,3rem)] font-bold text-primary mb-2">Task Manager</h1>
<p class="text-gray-600">Simple task management with Spring Boot, HTMX and doT.js</p>
</header>
<!-- 任務(wù)管理卡片 -->
<div class="bg-white rounded-xl shadow-lg p-6 mb-6">
<!-- 任務(wù)表單 -->
<form
id="task-form"
hx-post="/api/tasks"
hx-target="#task-list"
hx-ext="json-enc"
hx-swap="none"
hx-on="htmx:afterRequest:fetchTasks()"
class="flex space-x-3"
>
<input
type="text"
name="title"
placeholder="Add a new task..."
required
class="flex-1 px-4 py-2 border border-gray-300 rounded-lg focus:outline-none focus:ring-2 focus:ring-primary/50"
>
<button type="submit" class="btn btn-primary">
<i class="fa fa-plus mr-2"></i> Add
</button>
</form>
<!-- 任務(wù)列表容器 -->
<div id="task-list" class="space-y-3 mt-4">
<div class="text-center text-gray-500 py-8">
<i class="fa fa-spinner fa-spin text-primary text-2xl mb-2"></i>
<p>Loading tasks...</p>
</div>
</div>
</div>
<footer class="text-center text-gray-500 text-sm">
<p>Spring Boot + HTMX + doT.js Task Manager</p>
</footer>
</div>
<!-- doT.js 任務(wù)列表模板 -->
<script type="text/template" id="task-list-template">
{{~ it.tasks:task:index }}
<div class="task-item {{? task.completed}}task-item-complete{{?}} animate-fade-in" style="animation-delay: {{= index * 50}}ms">
<div class="flex items-center space-x-3">
<button
class="task-toggle-btn w-6 h-6 rounded-full border-2 flex items-center justify-center cursor-pointer transition-all
{{= task.completed ? 'bg-secondary border-secondary' : 'border-gray-300 hover:border-secondary hover:bg-secondary/10'}}
"
hx-put="/api/tasks/{{= task.id}}/toggle"
hx-target="closest .task-item"
hx-swap="none"
hx-on="htmx:afterRequest:fetchTasks()"
>
<i class="fa fa-check text-white text-xs"></i>
</button>
<span class="task-title font-medium">{{= task.title}}</span>
</div>
<button
class="delete-task-btn text-gray-400 hover:text-danger transition-colors"
hx-delete="/api/tasks/{{= task.id}}"
hx-target="closest .task-item"
hx-swap="delete"
hx-confirm="Are you sure you want to delete this task?"
hx-on="htmx:beforeRequest:this.closest('.task-item').classList.add('animate-fade-out')"
>
<i class="fa fa-trash-o text-lg"></i>
</button>
</div>
{{~}}
{{? it.tasks.length === 0 }}
<div class="text-center text-gray-500 py-8">
<i class="fa fa-check-circle text-secondary text-3xl mb-2"></i>
<p>No tasks yet. Add your first task!</p>
</div>
{{?}}
</script>
<script>
// 編譯 doT.js 模板
const taskListTemplate = doT.template(document.getElementById('task-list-template').text);
// 頁面加載后獲取任務(wù)列表
document.addEventListener('DOMContentLoaded', fetchTasks);
// 獲取任務(wù)列表
function fetchTasks() {
htmx.ajax('GET', '/api/tasks', {
handler: function(d,xhr) {
if (xhr.xhr.status === 200) {
try {
const tasks = JSON.parse(xhr.xhr.responseText);
renderTasks(tasks);
} catch (e) {
showError('Failed to parse tasks');
}
} else {
showError('Failed to load tasks');
}
}
});
}
// 渲染任務(wù)列表
function renderTasks(tasks) {
const taskList = document.getElementById('task-list');
const html = taskListTemplate({ tasks });
taskList.innerHTML = html;
htmx.process(taskList);
}
// 顯示錯誤信息
function showError(message) {
const taskList = document.getElementById('task-list');
taskList.innerHTML = `
<div class="text-center text-danger py-8">
<i class="fa fa-exclamation-circle text-danger text-2xl mb-2"></i>
<p>${message}</p>
</div>
`;
}
// 表單提交后清空輸入框
document.getElementById('task-form').addEventListener('htmx:afterRequest', function() {
this.reset();
});
</script>
</body>
</html>
這個前端實現(xiàn)了完整的任務(wù)管理界面,包括:
- 任務(wù)列表的展示和動態(tài)加載
- 添加新任務(wù)的表單
- 任務(wù)狀態(tài)的切換
- 任務(wù)的刪除功能
所有這些功能都是通過 HTMX 的屬性直接實現(xiàn)的,無需編寫大量 JavaScript 代碼。
當用戶執(zhí)行操作時,HTMX 會自動發(fā)送 AJAX 請求到后端 API,并根據(jù)響應(yīng)更新頁面。
前后端交互流程
整個應(yīng)用的交互流程如下:
- 頁面加載時,HTMX 發(fā)送 GET 請求到
/api/tasks獲取任務(wù)列表 - 用戶提交新任務(wù)表單時,HTMX 發(fā)送 POST 請求到
/api/tasks - 后端創(chuàng)建新任務(wù)并返回,HTMX 將新任務(wù)添加到列表
- 用戶點擊任務(wù)的完成狀態(tài)時,HTMX 發(fā)送 PUT 請求到
/api/tasks/{id}/toggle - 后端更新任務(wù)狀態(tài)并返回,HTMX 更新任務(wù)列表
- 用戶刪除任務(wù)時,HTMX 發(fā)送 DELETE 請求到
/api/tasks/{id} - 后端刪除任務(wù),HTMX 從 DOM 中移除任務(wù)項
部署與運行
要運行這個應(yīng)用,你需要
1. 創(chuàng)建一個 Spring Boot 項目
2. 將上述后端代碼復(fù)制到 src/main/java/com/example/taskmanager 目錄
3. 將前端代碼保存至 src/main/resources/static/index.html
4. 運行 Spring Boot 應(yīng)用
5. 在瀏覽器中訪問 http://localhost:8080/index.html
總結(jié)
通過結(jié)合 Spring Boot 和 HTMX,我們實現(xiàn)了一個高效、簡潔的前后端分離應(yīng)用。
這種架構(gòu)既保留了 Spring Boot 強大的后端處理能力,又通過 HTMX 簡化了前端開發(fā),避免了復(fù)雜的前端框架和構(gòu)建流程。
對于中小型項目或者需要快速迭代的應(yīng)用來說,這種組合是一個非常不錯的選擇。
如果你正在尋找一種輕量級、高效的 Web 開發(fā)解決方案,不妨嘗試一下 Spring Boot 與 HTMX 的組合。
到此這篇關(guān)于SpringBoot結(jié)合HTMX實現(xiàn)高效Web開發(fā)實戰(zhàn)的文章就介紹到這了,更多相關(guān)SpringBoot Web開發(fā)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
由淺到深帶你詳談Java實現(xiàn)數(shù)組擴容的三種方式
這篇文章主要詳細介紹了Java實現(xiàn)數(shù)組擴容的三種方式,新建一個數(shù)組,把原來數(shù)組的內(nèi)容搬到新數(shù)組中,使用system.arraycopy(),使用java.util.Arrays.copyOf()這三種方式,具有一定的參考價值,需要的朋友可以借鑒一下2023-06-06
關(guān)于SpringBoot使用Redis空指針的問題(不能成功注入的問題)
這篇文章主要介紹了關(guān)于SpringBoot使用Redis空指針的問題(不能成功注入的問題),本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-11-11
使用shardingsphere對SQLServer坑的解決
本文主要介紹了使用shardingsphere對SQLServer坑的解決,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2022-03-03
Java實現(xiàn)Excel與HTML互轉(zhuǎn)
Excel是一種電子表格格式,而HTM則是一種用于創(chuàng)建網(wǎng)頁的標記語言,雖然兩者在用途上存在差異,但有時我們需要將數(shù)據(jù)從一種格式轉(zhuǎn)換為另一種格式,下面我們就來看看具體實現(xiàn)方法吧2025-01-01
idea創(chuàng)建springboot項目,java版本只能選擇17和21的解決方案
這篇文章主要介紹了idea創(chuàng)建springboot項目,java版本只能選擇17和21的解決方案,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2025-04-04
Java實現(xiàn)添加,讀取和刪除Excel圖片的方法詳解
本文介紹在Java程序中如何添加圖片到excel表格,以及如何讀取、刪除excel表格中已有的圖片。文中的示例代碼講解詳細,感興趣的可以學習一下2022-05-05
java實現(xiàn)圖片上加文字水印(SpringMVC + Jsp)
這篇文章主要為大家詳細介紹了java實現(xiàn)在圖片上加文字水印的方法,水印可以是圖片或者文字,操作方便,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2016-05-05

