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

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

本教程将指导您如何在node.js和ejs应用中实现无刷新动态搜索功能。通过利用javascript的dom事件监听和fetch api进行异步请求,我们将优化后端控制器以返回json数据,并在前端实时更新搜索结果,彻底解决传统表单提交导致的页面重载问题,显著提升用户体验。

在现代Web应用中,用户期望在输入搜索关键词或调整筛选条件时,能够即时看到结果更新,而无需点击提交按钮或等待页面刷新。这正是动态搜索的核心价值。本教程将针对一个Node.js Express应用结合EJS模板引擎的场景,详细阐述如何构建一个高效、响应式的动态搜索系统。

1. 问题分析与现有代码审视

在提供的代码中,存在两个主要问题,导致动态搜索功能未能按预期工作:

oninput=”this.form.submit()” 导致的页面重载: 在user.ejs的搜索输入框上,oninput=”this.form.submit()” 属性使得每一次输入都会触发表单提交,导致整个页面刷新。这不仅阻止了JavaScript中fetch请求对#search-results div的局部更新,也可能与req.flash机制产生冲突,因为req.flash通常用于一次性消息,且在重定向后才会清除。后端响应类型不匹配: 前端JavaScript的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请求。

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;      }      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与JavaScript改造:实现实时更新

前端的主要任务是移除导致页面重载的表单提交,并确保JavaScript能够监听所有相关输入的变化,然后通过fetch API向新的后端API端点发送请求,并将返回的JSON数据动态渲染到页面上。

user.ejs 改造:

移除 oninput=”this.form.submit()”: 这是最关键的一步,它将阻止页面的自动刷新。移除表单 action=”/search” method=”GET”: 因为我们将通过JavaScript进行异步请求,不再需要传统的表单提交。为所有输入字段设置初始值: 如果您希望在页面刷新后保留用户的筛选条件,可以在EJS中设置输入字段的value属性。确保 search-results div 存在: 这是JavaScript将更新的区域。

                User        

SEARCH HERE

<input type="search" name="searchedValue" id="searchedValue" value="">
Department <option value="CSE" >CSE <option value="EEE" >EEE <option value="ME" >ME Year <option value="first" >First Year <option value="second" >Second Year <option value="third" >Third Year <option value="fourth" >Fourth Year
<input type="date" id="fromDate" name="fromDate" value=""> <input type="date" id="toDate" name="toDate" value="">
0) { %> { %>

No results found

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 = `

${result.id}

${result.name}

${result.description}

`; searchResultsContainer.appendChild(resultElement); }); } else { searchResultsContainer.innerHTML = '

No results found

'; } }) .catch(error => { console.error('Error fetching search results:', error); searchResultsContainer.innerHTML = `

Error loading results: ${error.message}

`; }); }, 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渲染,不需要在此处再次调用

4. 关键改进点与注意事项

Debouncing (去抖动):在searchedValueInput的input事件监听中,每次按键都会触发updateSearchResults。频繁的AJAX请求会增加服务器负担。通过引入searchTimeout和setTimeout/clearTimeout,我们实现了去抖动。这意味着用户停止输入300毫秒后,才会发送实际的搜索请求。这显著提升了用户体验和系统性能。AJAX请求的URL构造:使用URLSearchParams来构建查询字符串是更健壮和可读的方式,它能自动处理URL编码错误处理:在fetch请求中增加了.then(response => { if (!response.ok) … }) 和 .catch(error => …),用于处理网络错误或服务器返回的非2xx状态码。初始页面加载:getHome函数现在负责在首次访问或页面刷新时渲染EJS,并带上初始的搜索结果。这意味着用户在刷新页面后,不会看到空白的搜索结果区域,而是会显示当前的搜索/筛选条件下的结果。前端JavaScript的updateSearchResults函数在页面加载时不再需要显式调用,因为EJS已经完成了首次渲染。req.flash 的移除:由于AJAX请求直接返回JSON,req.flash机制不再适用于实时更新。它主要用于在Express会话中存储一次性消息,并在重定向后传递给下一个请求。

通过上述改造,您的Node.js和EJS应用将拥有一个功能完善、响应迅速的动态搜索功能,大大提升用户体验。

以上就是实现Node.js与EJS动态搜索:无刷新实时结果更新教程的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/194107.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月1日 17:24:36
下一篇 2025年11月1日 17:25:09

相关推荐

  • 配置PhpStorm代码折叠和展开的规则

    phpstorm 的代码折叠功能可通过设置和快捷键开启或关闭,并支持按语言结构自定义折叠规则,同时提供快捷键与鼠标操作实现高效代码浏览。具体包括:1. 在 settings 中勾选 enable code folding 或使用快捷键切换状态;2. 在 code folding 设置项中启用或禁用不…

    2025年12月10日 好文分享
    000
  • 配置PHPCMS手机端访问的Nginx规则

    要让phpcms在手机上快速运行,关键在于nginx规则配置。1. 通过定义map变量$is_mobile识别移动设备user-agent,实现精准的设备判断;2. 主域名配置中利用$is_mobile进行301重定向至手机站,提升seo与用户体验;3. 手机站与桌面站共用一套代码但分开配置,确保内…

    2025年12月10日 好文分享
    000
  • 优化PHPCMS编辑器的响应速度和稳定性

    优化phpcms编辑器的响应速度和稳定性需从诊断问题开始,明确是加载慢、运行卡顿还是保存异常;1.前端优化包括压缩合并文件、使用cdn加速、延迟加载资源、优化js代码及替换轻量级编辑器;2.后端优化涉及数据库查询优化、启用缓存机制、gzip压缩、高效图片处理、减少冗余数据库操作及异步处理耗时任务;3…

    2025年12月10日 好文分享
    000
  • PHP命名空间:组织代码结构

    php命名空间用于解决类名、函数名等标识符冲突问题,并提升代码可读性与维护性。1.命名空间通过逻辑分组避免冲突,如同不同文件夹允许同名文件;2.使用namespace声明命名空间,如namespace myappmodels;3.引用类时可用fqn或use关键字导入简化;4.支持子命名空间嵌套,如m…

    2025年12月10日 好文分享
    000
  • 配置PhpStorm代码格式化的规则和快捷键

    配置phpstorm的代码格式化规则和快捷键需先选择语言规范并设置代码风格,再自定义细节规则,最后配置快捷键及自动保存选项。首先打开settings进入editor > code style选择对应语言并新建或复制配置方案,可导入.editorconfig或psr-12标准,也可手动调整缩进、…

    2025年12月10日 好文分享
    000
  • PHP怎样获取网页HTML PHP抓取网页内容的3种高效方式分享

    php获取网页html内容主要有以下几种方式:1.file_get_contents()函数,优点是简单易用,代码量少,缺点是功能有限,无法设置请求头、超时时间等,容易被反爬虫机制拦截;2.curl扩展,功能强大,可以设置各种http选项,支持https,但代码相对复杂且需要安装curl扩展;3.g…

    2025年12月10日 好文分享
    000
  • PHP怎么解析JSON数据 PHP处理JSON数据的完整技巧解析

    php解析json数据的核心方法是使用json_decode()函数,它能将json字符串转换为php数组或对象。具体步骤包括:1. 使用json_decode()解码json字符串,通过设置第二个参数为true返回数组形式;2. 利用json_encode()将php变量编码为json字符串,并可…

    2025年12月10日 好文分享
    000
  • PHP如何调用Node.js脚本 调用Node.js的3种实用技巧

    php调用node.js脚本有三种主要方法:1.exec()、shell_exec()、system()函数可直接执行命令,但需注意安全性和异步处理;2.使用消息队列(如rabbitmq、redis)实现解耦和异步任务处理,需配置持久化与确认机制;3.通过http api调用node.js构建的服务…

    2025年12月10日 好文分享
    000
  • PHP模板系统:Blade引擎解析

    blade引擎的优势在于简洁性、可读性和高性能。其使用@符号引导的指令如@if、@foreach,使模板更易理解和维护,同时支持模板继承与组件功能,提升代码复用性和可维护性;此外,blade将模板编译为原生php代码并缓存,显著提高渲染速度。1. blade通过简单语法增强可读性与开发效率;2. 提…

    2025年12月10日 好文分享
    000
  • PHP反射机制:动态代码分析

    php反射机制通过动态分析代码结构实现类、方法、属性等信息的检查与操作,核心是reflectionclass、reflectionmethod等反射类。1. 可获取类名、构造函数参数、方法及属性;2. 支持动态调用方法、设置属性、创建实例;3. 广泛用于依赖注入、orm、单元测试等场景;4. 使用时…

    2025年12月10日 好文分享
    000
  • PHP中的JSON:如何解析和生成JSON数据

    在php中处理json数据的核心操作是解析和生成。使用json_decode()可将json字符串转为php数组或对象,如$jsonstring = ‘{“name”:”john doe”,”age”:30,&#82…

    2025年12月10日 好文分享
    000
  • PHP中的JSON处理:如何在PHP中解析和生成JSON数据

    php处理json数据的内置函数主要有json_encode()和json_decode()。一、json_encode()用于将php数组或对象转换为json字符串,常用于构建api响应,默认中文会被转义,可用json_unescaped_unicode保留中文,使用json_pretty_pri…

    2025年12月10日
    000
  • 如何将整型转换为字符串?

    将整型转换为字符串的方法在不同编程语言中有显著差异。1) 在python中,使用str()函数可将整型转换为字符串,如str(42),并支持格式化保留前导零和处理大数。2) 在java中,可用string.valueof()或integer.tostring()方法进行转换,前者更通用,后者专为整型…

    2025年12月10日
    000
  • PHP中如何分割字符串?

    在php中分割字符串最常用的方法是使用explode函数。1. 使用explode函数简单高效,适合大多数场景。2. 处理多种分隔符时,使用preg_split和正则表达式更灵活,但需注意性能和正确性。3. 限制分割次数时,explode函数的第三个参数很有用。4. 复杂格式可结合explode和p…

    2025年12月10日
    000
  • PHP中如何操作JSON数据?

    在php中操作json数据可以通过json_encode和json_decode函数实现。1. 使用json_encode将php数组转换为json格式。2. 使用json_decode将json字符串转换为php数组或对象。3. 处理复杂的嵌套json数据时,使用循环遍历。4. 诊断json错误时…

    2025年12月10日
    000
  • 如何将数组转换为字符串?

    在python中,将数组转换为字符串可以使用以下方法:1. 使用str.join()方法,如’, ‘.join(map(str, my_list));2. 使用列表推导式,如’, ‘.join([str(item) for item in my_lis…

    2025年12月10日
    000
  • 如何将字符串转换为整型?

    在python中,将字符串转换为整型主要使用int()函数。1) 去除字符串中的空格,使用strip()方法;2) 处理带小数点的字符串,先转浮点数再转整型,或使用round()函数四舍五入;3) 处理带千位分隔符的字符串,使用replace()方法去除分隔符;4) 使用try-except块处理错…

    2025年12月10日
    000
  • 如何从JSON创建PHP多维数组?

    从json创建php多维数组的步骤如下:1) 使用json_decode函数将json字符串转换为php数组,第二个参数设为true以获取关联数组;2) 对于复杂嵌套结构,使用递归函数处理多维数组;3) 处理错误时,使用json_last_error()函数检查错误类型;4) 处理大规模数据时,采用…

    2025年12月10日
    000
  • 如何将PHP多维数组转换为JSON?

    将php多维数组转换为json的方法是使用json_encode函数。1) 使用json_encode($array, json_unescaped_unicode)处理中文字符,避免乱码。2) 使用json_pretty_print选项提高json的可读性,方便调试和阅读。 要将PHP的多维数组转…

    2025年12月10日
    000
  • PHP中explode()和preg_split()的区别?

    explode()适合简单字符串分割,而preg_split()适合需要正则表达式的复杂分割任务。1) explode()使用固定分隔符,适合初学者和简单场景。2) preg_split()使用正则表达式,适合复杂分割,但增加了使用难度和性能考虑。 PHP中explode()和preg_split(…

    2025年12月10日
    000

发表回复

登录后才能评论
关注微信