Jakarta EE 课程 --- 微型资料投递与分发(Mini Drop-off Box)
摘要:实验基于Jakarta EE 9+(兼容Tomcat 10+)、Maven作为构建工具,并在IntelliJ IDEA 2023.2(Community版免费)中进行。项目使用Maven Archetype WebApp模板生成基础结构,然后升级到Jakarta EE。
这个实验实现了一个简单的文件上传/下载系统,名为HttpExperiment,满足所有功能要求:
- 用户上传文件+描述,生成唯一ID,保存到内存。
- 支持重定向(302)、JSON响应(201 + Location头)、下载(Content-Type等头)、缓存(ETag + 304)、错误处理(413/415)、XSS转义。
- 路径:/drops(上传表单/列表),/drops/{id}(详情/下载)。
本文将提供:
- 实验步骤:从项目创建到测试的详细指导。
- 代码:完整、可复制的pom.xml、Java类(Servlet)、JSP文件、web.xml等。
- 代码注释:每个代码块中添加详细注释,解释实现逻辑、HTTP相关知识和关键点。
实验准备:
- 前提:IntelliJ IDEA 2023.2已安装,JDK 17+配置好(File > Project Structure > SDKs > Add JDK)。Tomcat 10+下载(https://tomcat.apache.org/),解压到本地目录(e.g., C:\Tomcat10)。
- 时间:约1-2小时(包括调试)。
- 测试工具:浏览器(Chrome)、curl命令(Windows CMD安装curl或用Git Bash)。
1. 实验步骤
1.1: 在IntelliJ IDEA中使用Maven Archetype创建项目
- 打开IntelliJ IDEA,点击“File > New > Project”。
- 选择“Maven Archetype”。
- 配置:
- Name:HttpExperiment。
- Location:选择保存目录。
- Archetype:点击“Add Archetype...”,填写:
- GroupId:org.apache.maven.archetypes
- ArtifactId:maven-archetype-webapp
- Version:1.4(稳定版)。
- 点击“OK”,然后“Next”。
- 项目属性:
- GroupId:com.example
- ArtifactId:httpexperiment
- Version:1.0-SNAPSHOT
- 点击“Create”。IntelliJ生成项目,包括pom.xml和src/main/webapp/index.jsp。
1.2: 更新pom.xml到Jakarta EE并添加依赖
- 双击pom.xml,替换为以下内容(升级到Jakarta EE 9,添加必需依赖):
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.example</groupId>
<artifactId>httpexperiment</artifactId>
<version>1.0-SNAPSHOT</version>
<packaging>war</packaging>
<properties>
<maven.compiler.source>17</maven.compiler.source>
<maven.compiler.target>17</maven.compiler.target>
</properties>
<dependencies>
<!-- Jakarta EE Web API (Servlet, JSP等) -->
<dependency>
<groupId>jakarta.platform</groupId>
<artifactId>jakarta.jakartaee-web-api</artifactId>
<version>9.1.0</version>
<scope>provided</scope>
</dependency>
<!-- JSTL for JSP (标签支持) -->
<dependency>
<groupId>jakarta.servlet.jsp.jstl</groupId>
<artifactId>jakarta.servlet.jsp.jstl-api</artifactId>
<version>2.0.0</version>
</dependency>
<dependency>
<groupId>org.glassfish.web</groupId>
<artifactId>jakarta.servlet.jsp.jstl</artifactId>
<version>2.0.0</version>
</dependency>
<!-- Jackson for JSON (可选, 用于JSON响应) -->
<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-databind</artifactId>
<version>2.13.3</version>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-war-plugin</artifactId>
<version>3.3.2</version>
</plugin>
</plugins>
</build>
</project>
- 右键pom.xml > Maven > Reload project(下载依赖)。
1.3: 更新web.xml(兼容Jakarta EE)
在src/main/webapp/WEB-INF/web.xml
中替换为:
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="https://jakarta.ee/xml/ns/jakartaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="https://jakarta.ee/xml/ns/jakartaee https://jakarta.ee/xml/ns/jakartaee/web-app_5_0.xsd"
version="5.0">
<!-- 无需额外配置, Servlet用注解 -->
</web-app>
1.4: 创建Servlet类(DropsServlet.java)
在src/main/java/com/example
下New > Java Class创建DropsServlet.java。
package com.example;
import jakarta.servlet.ServletException;
import jakarta.servlet.annotation.MultipartConfig;
import jakarta.servlet.annotation.WebServlet;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import jakarta.servlet.http.Part;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.PrintWriter;
import java.util.Base64;
import java.util.HashMap;
import java.util.Map;
import java.util.UUID;
import com.fasterxml.jackson.databind.ObjectMapper;
// 注解注册路径: /drops (上传/列表), /drops/* (详情/下载)
@WebServlet(name = "DropsServlet", value = {"/drops", "/drops/*"})
@MultipartConfig // 启用multipart/form-data支持 (文件上传)
public class DropsServlet extends HttpServlet {
// 内存存储: ID -> 文件数据 (Map<String, FileData>)
private Map<String, FileData> fileStore = new HashMap<>();
// 文件数据类 (内部类, 存储描述、内容、类型、ETag)
private static class FileData {
String description;
byte[] content;
String contentType;
String etag; // ETag for缓存
FileData(String desc, byte[] cont, String type) {
de