新闻中心
实现Node.js与EJS动态搜索:无刷新实时结果更新教程

本教程将指导您如何在node.js和ejs应用中实现无刷新动态搜索功能。通过利用j*ascript的dom事件监听和fetch api进行异步请求,我们将优化后端控制器以返回json数据,并在前端实时更新搜索结果,彻底解决传统表单提交导致的页面重载问题,显著提升用户体验。
在现代Web应用中,用户期望在输入搜索关键词或调整筛选条件时,能够即时看到结果更新,而无需点击提交按钮或等待页面刷新。这正是动态搜索的核心价值。本教程将针对一个Node.js Express应用结合EJS模板引擎的场景,详细阐述如何构建一个高效、响应式的动态搜索系统。
1. 问题分析与现有代码审视
在提供的代码中,存在两个主要问题,导致动态搜索功能未能按预期工作:
- oninput="this.form.submit()" 导致的页面重载: 在user.ejs的搜索输入框上,oninput="this.form.submit()" 属性使得每一次输入都会触发表单提交,导致整个页面刷新。这不仅阻止了J*aScript中fetch请求对#search-results div的局部更新,也可能与req.flash机制产生冲突,因为req.flash通常用于一次性消息,且在重定向后才会清除。
- 后端响应类型不匹配: 前端J*aScript的updateSearchResults函数使用fetch请求,并期望从/search路径获取JSON格式的搜索结果 (.then(response => response.json()))。然而,后端user.controllers.js中的getHome函数(通常处理/或/search路由)最终是调用res.render('../Views/user.ejs', ...)来渲染EJS模板,而不是返回JSON数据。req.flash('search_results', search_results) 也是为EJS渲染准备的,不适合直接作为AJAX响应。
为了解决这些问题,我们需要对前端和后端进行相应的改造。
2. 后端控制器优化:提供JSON API
为了支持前端的异步请求,我们需要一个能够返回JSON格式搜索结果的后端API端点。我们可以修改现有的getHome函数,使其能够根据请求类型(是普通页面加载还是AJAX请求)返回不同的响应,或者创建一个新的专用API函数。考虑到清晰性和职责分离,我们推荐创建一个新的API函数来处理AJAX请求。
user.controllers.js 改造:
我们将保留getHome用于初始页面加载和渲染EJS,并创建一个getSearchResultsAPI函数来专门处理前端的fetch请求。
小爱开放平台
小米旗下小爱开放平台
291
查看详情
const { pool } = require('../config/database.config');
const MiniSearch = require('minisearch');
const userModels = require('../models/user.models'); // 假设 formatDate 在这里
// 辅助函数:执行搜索和过滤逻辑
const performSearchAndFilter = async (searchedValue, deptFilter, yearFilter, fromDate, toDate) => {
const all_results = [];
const client = await pool.connect();
try {
const query = `select * from "users"`;
const result = await client.query(query);
result.rows.forEach((row) => {
all_results.push(row);
});
} catch (err) {
console.error("Database query error:", err);
// 可以在这里抛出错误或返回空数组
return [];
} finally {
client.release(); // 确保释放客户端
}
const minisearch = new MiniSearch({
fields: ['id', 'name', 'description', 'dept', 'year', 'fromDate', 'toDate'],
storeFields: ['id', 'name', 'description', 'dept', 'year', 'fromDate', 'toDate'],
});
minisearch.addAll(all_results);
const filterCriteria = (result, filters) => {
return Object.entries(filters).every(([key, value]) => {
if (!value) {
return true;
}
if (key === 'fromDate') {
const formattedDate = userModels.formatDate(result.fromDate);
return value <= formattedDate;
}
if (key === 'toDate') {
const formattedDate = userModels.formatDate(result.toDate);
return value >= formattedDate;
}
return result[key] !== undefined && result[key] === value;
});
};
const filters = {
dept: deptFilter,
year: yearFilter,
fromDate: fromDate,
toDate: toDate,
};
let results = [];
if (searchedValue) {
results = minisearch.search(searchedValue, {
prefix: true,
fuzzy: 0.4,
filter: (result) => filterCriteria(result, filters),
});
} else {
results = all_results.filter((result) => filterCriteria(result, filters));
}
return results.map((result) => ({
id: result.id,
name: result.name,
description: result.description,
}));
};
// 首页渲染函数 (保持不变,用于首次加载页面)
const getHome = async (req, res) => {
const searchedValue = req.query.searchedValue || '';
const deptFilter = req.query.deptFilter || '';
const yearFilter = req.query.yearFilter || '';
const fromDate = req.query.fromDate || '';
const toDate = req.query.toDate || '';
const search_results = await performSearchAndFilter(searchedValue, deptFilter, yearFilter, fromDate, toDate);
// 渲染EJS模板,包含初始或刷新后的搜索结果
res.render('../Views/user.ejs', {
search_results: search_results,
// 传递当前筛选值,以便前端可以回显
currentFilters: { searchedValue, deptFilter, yearFilter, fromDate, toDate }
});
};
// 新增的API函数,用于处理AJAX请求,返回JSON数据
const getSearchResultsAPI = async (req, res) => {
try {
const searchedValue = req.query.searchedValue || '';
const deptFilter = req.query.deptFilter || '';
const yearFilter = req.query.yearFilter || '';
const fromDate = req.query.fromDate || '';
const toDate = req.query.toDate || '';
const search_results = await performSearchAndFilter(searchedValue, deptFilter, yearFilter, fromDate, toDate);
// 直接返回JSON数据
res.json({ search_results: search_results });
} catch (error) {
console.error("API search error:", error);
res.status(500).json({ error: "Internal Server Error" });
}
};
module.exports = {
getHome,
getSearchResultsAPI, // 导出新的API函数
};路由配置 (app.js 或 routes.js):
确保您的Express应用中,/ 路由映射到getHome,而/search(或/api/search)路由映射到getSearchResultsAPI。
// 示例 Express 路由配置
const express = require('express');
const router = express.Router();
const userController = require('./controllers/user.controllers');
router.get('/', userController.getHome); // 初始页面加载
router.get('/search', userController.getSearchResultsAPI); // AJAX请求
module.exports = router;3. 前端EJS与J*aScript改造:实现实时更新
前端的主要任务是移除导致页面重载的表单提交,并确保J*aScript能够监听所有相关输入的变化,然后通过fetch API向新的后端API端点发送请求,并将返回的JSON数据动态渲染到页面上。
user.ejs 改造:
- 移除 oninput="this.form.submit()": 这是最关键的一步,它将阻止页面的自动刷新。
- 移除表单 action="/search" method="GET": 因为我们将通过J*aScript进行异步请求,不再需要传统的表单提交。
- 为所有输入字段设置初始值: 如果您希望在页面刷新后保留用户的筛选条件,可以在EJS中设置输入字段的value属性。
- 确保 search-results div 存在: 这是J*aScript将更新的区域。
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>User</title>
</head>
<body>
<!-- 移除 form action 和 method,也不再需要 submit 按钮 -->
<div>
<h1>SEARCH HERE</h1>
<input type="search" name="searchedValue" id="searchedValue" value="<%= currentFilters.searchedValue %>">
<br>
<label for="filter">Select a filter:</label>
<select id="deptFilter" name="deptFilter">
<option value="">Department</option>
<option value="CSE" <%= currentFilters.deptFilter === 'CSE' ? 'selected' : '' %>>CSE</option>
<option value="EEE" <%= currentFilters.deptFilter === 'EEE' ? 'selected' : '' %>>EEE</option>
<option value="ME" <%= currentFilters.deptFilter === 'ME' ? 'selected' : '' %>>ME</option>
</select>
<select id="yearFilter" name="yearFilter">
<option value="">Year</option>
<option value="first" <%= currentFilters.yearFilter === 'first' ? 'selected' : '' %>>First Year</option>
<option value="second" <%= currentFilters.yearFilter === 'second' ? 'selected' : '' %>>Second Year</option>
<option value="third" <%= currentFilters.yearFilter === 'third' ? 'selected' : '' %>>Third Year</option>
<option value="fourth" <%= currentFilters.yearFilter === 'fourth' ? 'selected' : '' %>>Fourth Year</option>
</select>
<br>
<label for="fromDate"> From </label>
<input type="date" id="fromDate" name="fromDate" value="<%= currentFilters.fromDate %>">
<label for="toDate"> To </label>
<input type="date" id="toDate" name="toDate" value="<%= currentFilters.toDate %>">
<br>
<!-- 移除 submit 按钮 -->
</div>
<div id="search-results">
<% if(search_results && search_results.length > 0) { %>
<% search_results.forEach((result) => { %>
<div>
<h1><%= result.id %></h1>
<h2><%= result.name %></h2>
<h3><%= result.description %></h3>
</div>
<% }) %>
<% } else { %>
<h1>No results found</h1>
<% } %>
</div>
<script>
const searchedValueInput = document.querySelector('#searchedValue');
const deptFilterInput = document.querySelector('#deptFilter');
const yearFilterInput = document.querySelector('#yearFilter');
const fromDateInput = document.querySelector('#fromDate');
const toDateInput = document.querySelector('#toDate');
const searchResultsContainer = document.querySelector('#search-results');
let searchTimeout; // 用于 debouncing
function updateSearchResults() {
clearTimeout(searchTimeout); // 清除之前的计时器
searchTimeout = setTimeout(() => { // 设置新的计时器
const searchedValue = searchedValueInput.value;
const deptFilter = deptFilterInput.value;
const yearFilter = yearFilterInput.value;
const fromDate = fromDateInput.value;
const toDate = toDateInput.value;
// 构建查询字符串
const queryParams = new URLSearchParams({
searchedValue: searchedValue,
deptFilter: deptFilter,
yearFilter: yearFilter,
fromDate: fromDate,
toDate: toDate
}).toString();
// 发送异步请求到新的API端点
fetch(`/search?${queryParams}`) // 注意这里指向 /search 路由
.then(response => {
if (!response.ok) {
throw new Error(`HTTP error! status: ${response.status}`);
}
return response.json();
})
.then(data => {
searchResultsContainer.innerHTML = ''; // 清空现有结果
if (data.search_results && data.search_results.length > 0) {
data.search_results.forEach(result => {
const resultElement = document.createElement('div');
resultElement.innerHTML = `
<h1>${result.id}</h1>
<h2>${result.name}</h2>
<h3>${result.description}</h3>
`;
searchResultsContainer.appendChild(resultElement);
});
} else {
searchResultsContainer.innerHTML = '<h1>No results found</h1>';
}
})
.catch(error => {
console.error('Error fetching search results:', error);
searchResultsContainer.innerHTML = `<h1>Error loading results: ${error.message}</h1>`;
});
}, 300); // 300毫秒的 debouncing 延迟
}
// 监听所有相关输入字段的变化
searchedValueInput.addEventListener('input', updateSearchResults);
deptFilterInput.addEventListener('change', updateSearchResults);
yearFilterInput.addEventListener('change', updateSearchResults);
fromDateInput.addEventListener('change', updateSearchResults);
toDateInput.addEventListener('change', updateSearchResults);
// 页面加载时执行一次搜索,以防页面刷新后需要重新加载数据
// updateSearchResults(); // 首次加载已由EJS渲染,不需要在此处再次调用
</script>
</body>
</html>4. 关键改进点与注意事项
-
Debouncing (去抖动):
- 在searchedValueInput的input事件监听中,每次按键都会触发updateSearchResults。频繁的AJAX请求
会增加服务器负担。 - 通过引入searchTimeout和setTimeout/clearTimeout,我们实现了去抖动。这意味着用户停止输入300毫秒后,才会发送实际的搜索请求。这显著提升了用户体验和系统性能。
- 在searchedValueInput的input事件监听中,每次按键都会触发updateSearchResults。频繁的AJAX请求
-
AJAX请求的URL构造:
- 使用URLSearchParams来构建查询字符串是更健壮和可读的方式,它能自动处理URL编码。
-
错误处理:
- 在fetch请求中增加了.then(response => { if (!response.ok) ... }) 和 .catch(error => ...),用于处理网络错误或服务器返回的非2xx状态码。
-
初始页面加载:
- getHome函数现在负责在首次访问或页面刷新时渲染EJS,并带上初始的搜索结果。这意味着用户在刷新页面后,不会看到空白的搜索结果区域,而是会显示当前的搜索/筛选条件下的结果。
- 前端J*aScript的updateSearchResults函数在页面加载时不再需要显式调用,因为EJS已经完成了首次渲染。
-
req.flash 的移除:
- 由于AJAX请求直接返回JSON,req.flash机制不再适用于实时更新。它主要用于在Express会话中存储一次性消息,并在重定向后传递给下一个请求。
通过上述改造,您的Node.js和EJS应用将拥有一个功能完善、响应迅速的动态搜索功能,大大提升用户体验。
以上就是实现Node.js与EJS动态搜索:无刷新实时结果更新教程的详细内容,更多请关注其它相关文章!
# 首次
# 河北关键词排名查询
# 烟台seo优化项目
# 唯品会营销推广具体内容
# 上海高端全网营销推广平台
# 桂城网站推广优势
# 迪庆网站建设优化
# 宁夏seo关键词排名优化软件
# 山南seo公司都选火星
# 网站推广多少费用一年啊
# 咸宁本地seo推广哪家好做
# 创建一个
# 您的
# 这是
# 小爱
# 表单
# javascript
# 移除
# 后端
# 搜索结果
# 加载
# edg
# app
# 编码
# node
# ajax
# json
# node.js
# 前端
# js
# html
# java
相关栏目:
【
科技资讯46185 】
【
网络学院92790 】
相关推荐:
J*aScript异步迭代器_j*ascript异步遍历
在FastAPI中利用lifespan与依赖注入高效管理Redis连接池
在哪找SublimeJ远程工具_SFTP插件配置教程
Python实时数据流中的动态最值查找策略
必由学登录入口 必由学官方网站在线访问链接
J*aScript中如何高效提取对象指定属性
vivo手机互传视频怎么操作_vivo手机互传视频详细传输方法
内存疯狂猛猛涨价:主板销量直接腰斩!
苹果手机指南针不准怎么校准 传感器校准方法详解【建议收藏】
c++如何使用折叠表达式(Fold Expressions)_c++17可变参数模板新技巧
大象笔记网页版入口 印象笔记网页版登录入口
J*aScript打印功能_j*ascript输出控制
2026春节假期票务安排_2026春节放假购票指南
Win11怎么查看电脑配置_Win11硬件配置检测工具使用
Tailwind CSS line-clamp 布局问题解析与修复指南
Win11怎么设置鼠标主按键_Win11鼠标左右键功能互换
J*aScript数组对象转换:按指定键分组与值收集
win11如何加载ICC颜色配置文件 Win11校色文件安装与显示器色彩管理【指南】
Go RPC HTTP服务正确实现与常见陷阱解析
如何使用Node.js csv 包按条件移除含空字段的CSV记录
Safari怎么安装扩展程序 浏览器插件安装与管理方法【详解】
J*a应用集成GitHub CLI与API认证指南
汽水音乐车机版8.9下载 汽水音乐车机版8.9版本安装入口
fishbowl官网免费版 fishbowl养鱼网站入口
12306怎么选座位选到安静区_12306选座安静区域选择策略
Win10桌面图标出现小盾牌怎么办 Win10去除UAC图标教程【解决】
Yandex搜索引擎一键访问入口_俄罗斯Yandex官网免登录
不会效仿卡普空!《铁拳》制作人澄清:不采取赛事付费|直播|
QQ邮箱网页版入口页面 QQ邮箱在线登录入口官网
印象笔记如何设提醒任务防漏执行_印象笔记设提醒任务防漏执行【任务提醒】
QQ邮箱网页版登录入口 QQ邮箱官方在线使用平台
小猿搜题在线学习页面在哪_小猿搜题在线学习中心入口
J*a如何使用AtomicInteger控制计数_J*a无锁计数器性能分析
邮政快递包裹最新位置 邮政快递实时追踪入口
qq浏览器打开空白页怎么办 qq浏览器启动后显示白屏的解决教程
sublime如何配置Go语言开发环境_sublime搭建Golang编译运行系统
如何在Promise链中优雅地中断后续then执行
NRF24L01数据传输深度解析:解决大载荷接收异常与分包策略
千牛数据看板网页版_千牛数据看板网页版访问方法
J*a编写用户注册与登录功能_掌握字符串与验证逻辑
夸克AO3官网入口_AO3镜像网站2025推荐
美团外卖商家服务中心入口 美团商家版官网入口
Web Components中自定义开关组件状态同步的常见陷阱与解决方案
痛风发作了怎么办? 快速止痛和后期饮食调理
抖音DOU+怎么投最有效 抖音付费推广的ROI提升技巧
J*aScript map 方法中处理循环元素为空数组的策略
J*aScript DOM操作:高效清空列表元素的策略与实践
腾讯QQ邮箱登录入口_QQ邮箱官方网站使用地址
composer 和 npm/yarn 在管理依赖方面有什么核心思想差异?
微博网页版主页入口 微博官方网站免登录访问


2025-10-26
浏览次数:次
返回列表
会增加服务器负担。