是将数据库添加到JSP页面显示的详细步骤指南,涵盖从环境搭建到数据展示的全流程操作:
准备工作
- 确认依赖库:确保项目中已包含JDBC驱动包(如MySQL的
mysql-connector-java.jar),该JAR文件需放置在WEB-INF/lib目录下,以便Web容器自动加载,若使用Maven管理项目,则应在pom.xml中添加对应依赖项。 - 创建测试用的数据表:例如创建一个名为
users的用户信息表,包含字段id,username,email等,并插入几条示例记录供后续验证效果。
核心实现步骤
建立数据库连接类(DBUtility)
创建一个专门负责获取和关闭数据库连接的工具类,以下是一个典型实现示例:
import java.sql.;
public class DBUtility {
private static final String URL = "jdbc:mysql://localhost:3306/mydb";
private static final String USERNAME = "root";
private static final String PASSWORD = "password";
public static Connection getConnection() throws SQLException {
return DriverManager.getConnection(URL, USERNAME, PASSWORD);
}
public static void closeResources(ResultSet rs, PreparedStatement ps, Connection con) {
try {if(rs != null) rs.close();} catch (SQLException e) {}
try {if(ps != null) ps.close();} catch (SQLException e) {}
try {if(con != null) con.close();} catch (SQLException e) {}
}
}
️ 注意:实际开发中建议通过配置文件存储敏感信息(如用户名/密码),避免硬编码在代码中。
设计数据访问对象(DAO模式)
采用DAO(Data Access Object)模式封装业务逻辑,使代码结构更清晰,例如针对用户表的操作:
public class UserDAO {
public List<User> findAllUsers() {
List<User> users = new ArrayList<>();
String sql = "SELECT id, username, email FROM users";
try (Connection con = DBUtility.getConnection();
PreparedStatement pstmt = con.prepareStatement(sql);
ResultSet rs = pstmt.executeQuery()) {
while (rs.next()) {
User user = new User();
user.setId(rs.getInt("id"));
user.setUsername(rs.getString("username"));
user.setEmail(rs.getString("email"));
users.add(user);
}
} catch (SQLException e) {
e.printStackTrace(); // 生产环境应替换为日志记录
}
return users;
}
}
优势:这种设计符合单一职责原则,便于维护和扩展,每个数据表对应独立的DAO类,避免功能耦合。
编写Servlet控制器处理请求
创建继承自HttpServlet的控制层组件,用于协调模型与视图间的交互:
@WebServlet("/listUsers")
public class UserServlet extends HttpServlet {
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
UserDAO dao = new UserDAO();
List<User> users = dao.findAllUsers();
// 将数据集存入请求作用域
request.setAttribute("userList", users);
// 转发到JSP页面进行渲染
RequestDispatcher dispatcher = request.getRequestDispatcher("/showUsers.jsp");
dispatcher.forward(request, response);
}
}
关键点:通过
request.setAttribute()方法传递数据对象,使得JSP能够直接访问这些绑定到请求范围内的变量。
JSP页面展示数据(两种方式对比)
方案A:脚本式循环输出(基础写法)
<%@ page import="java.util.List, com.example.User" %>
<html>
<head><title>用户列表</title></head>
<body>
<table border="1">
<tr><th>ID</th><th>用户名</th><th>邮箱</th></tr>
<%
List<User> users = (List<User>) request.getAttribute("userList");
for(User u : users) {
%>
<tr>
<td><%= u.getId() %></td>
<td><%= u.getUsername() %></td>
<td><%= u.getEmail() %></td>
</tr>
<% } %>
</table>
</body>
</html>
缺点:Java代码与HTML混杂,可读性差且难以维护,不推荐用于复杂场景。
方案B:JSTL标签库+EL表达式(推荐做法)
首先在页面顶部声明标签库:
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>
然后使用标准标签实现迭代渲染:
<html>
<head><title>用户列表</title></head>
<body>
<table border="1">
<tr><th>ID</th><th>用户名</th><th>邮箱</th></tr>
<c:forEach var="user" items="${userList}">
<tr>
<td>${user.id}</td>
<td>${user.username}</td>
<td>${user.email}</td>
</tr>
</c:forEach>
</table>
</body>
</html>
优点:完全分离业务逻辑与视图层,支持条件判断、循环等高级功能,大幅提升代码可读性和维护效率。
异常处理机制
- 数据库层面:在DAO层捕获
SQLException并转换为自定义异常抛出,防止敏感错误信息泄露至前端。try { // 执行数据库操作... } catch (SQLException e) { throw new DataAccessException("数据库操作失败", e); } - 表现层兜底:在JSP中增加错误提示区域,当出现运行时异常时展示友好的错误消息而非堆栈跟踪。
性能优化建议
| 优化方向 | 具体措施 | 效果说明 |
|---|---|---|
| 连接池复用 | 使用Apache Commons DBCP或HikariCP管理物理连接 | 减少频繁创建/销毁连接开销 |
| 分页加载 | SQL添加LIMIT子句配合前端分页控件 | 降低单次查询的数据量 |
| 预编译语句 | 对固定结构的查询使用PreparedStatement | 提升重复执行时的解析效率 |
| 缓存热点数据 | Redis缓存高频访问的用户信息 | 减轻数据库读压力 |
FAQs相关问答
Q1:为什么推荐使用JSTL而不是直接在JSP中写Java脚本?
A:JSTL标签库提供了标准化的数据遍历、条件判断等功能,其核心优势在于:①强制分离关注点(SoC原则),使前端开发人员无需了解后端逻辑;②内置防XSS攻击机制自动转义特殊字符;③语法简洁易读,团队协作效率更高,而脚本片段混合编码容易导致安全破绽和维护困难。
Q2:如果出现中文乱码该如何解决?
A:需要三处设置字符编码保持一致:①Web应用的web.xml中配置<filter>指定请求响应编码为UTF-8;②数据库表使用utf8mb4字符集;③JDBC连接字符串添加参数characterEncoding=UTF-8。jdbc:mysql://localhost:3306/mydb?useUnicode=true&characterEncoding=UTF-8
