实现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/1532779.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月21日 00:01:30
下一篇 2025年12月21日 00:01:43

相关推荐

  • 为什么前端固定定位会发生移动问题?

    前端固定定位为什么会出现移动现象? 在进行前端开发时,我们经常会使用CSS中的position属性来控制元素的定位。其中,固定定位(position: fixed)是一种常用的定位方式,它可以让元素相对于浏览器窗口进行定位,保持在页面的固定位置不动。 然而,有时候我们会遇到一个问题:在使用固定定位时…

    2025年12月24日
    000
  • 深入剖析Ajax技术:揭开其核心技术原理与应用

    深入了解Ajax技术:探索其核心技术原理与应用Ajax(Asynchronous JavaScript and XML)是一种在Web开发中广泛应用的技术,它通过使用异步通信和JavaScript的技术手段,实现了在不刷新整个网页的情况下与服务器进行数据交互。在本文中,我们将深入了解Ajax技术的核…

    2025年12月24日
    000
  • 了解AJAX所需的参数是什么?

    深入了解AJAX的参数:您需要掌握哪些参数? 引言: 在现代Web开发中,AJAX(Asynchronous JavaScript and XML)是一个被广泛使用的技术,它可以实现异步加载数据,从而提升用户体验。AJAX的核心是通过发送HTTP请求与服务器进行交互,并将响应的数据动态地展示在页面上…

    2025年12月24日
    000
  • 深入解析AJAX参数:它们的重要性何在?

    AJAX的参数详解:为什么它们如此重要? 随着Web应用的复杂性不断增加,用户对于实时响应和无刷新的交互体验的需求也越来越高。在这样的背景下,AJAX(Asynchronous JavaScript and XML)成为了前端开发中的必备技术。它可以实现异步数据交互,从服务器请求数据并将其无缝地展示…

    2025年12月24日
    000
  • 通过使用Ajax函数实现异步数据交换的方法

    如何利用Ajax函数实现异步数据交互 随着互联网和Web技术的发展,前端与后端之间的数据交互变得十分重要。传统的数据交互方式,如页面刷新和表单提交,已经不能满足用户的需求。而Ajax(Asynchronous JavaScript and XML)则成为了实现异步数据交互的重要工具。 Ajax通过使…

    2025年12月24日
    000
  • Ajax技术:传统与现代的发展与演进

    从传统到现代:Ajax技术的发展与演进 引言:随着互联网的发展,网页设计与开发也在不断演进。传统的网页通过用户与服务器之间的页面刷新来传递和展示数据,这种方式存在诸多的不便和效率问题。而Ajax(Asynchronous JavaScript and XML)技术的出现,彻底改变了传统网页的工作方式…

    2025年12月24日
    000
  • 使用Ajax技术实现实时数据交互的有效方法

    利用Ajax技术实现无刷新数据交互的实用方法 在Web开发中,数据的实时交互是一个非常重要的功能。传统的浏览器请求刷新页面的方式已经不能满足用户的需求,因此,Ajax技术应运而生。Ajax(Asynchronous JavaScript and XML)是一种可以在不刷新整个页面的情况下,通过与服务…

    2025年12月24日
    000
  • 了解Ajax框架:探索常见的五种框架

    了解Ajax框架:探索常见的五种框架,需要具体代码示例 引言:在现代Web应用开发中,Ajax是必不可少的技术之一。它以其支持异步数据交互,提升用户体验等特点,成为了前端开发中不可或缺的一部分。为了更好地了解和掌握Ajax框架,本文将介绍五种常见的Ajax框架,并提供具体的代码示例,帮助读者深入了解…

    2025年12月24日
    000
  • 深入理解Ajax函数及其参数用法

    掌握常用的Ajax函数及其参数详解 Ajax(Asynchronous JavaScript and XML)是一种用于在客户端和服务器之间异步传输数据的技术。它能够实现无需刷新整个页面而更新部分内容,提升了用户体验和性能。本文将详细介绍常用的Ajax函数及其参数,并附带具体的代码示例。 一、XML…

    2025年12月24日
    000
  • 前端开发中的应用与实践:使用Ajax函数

    Ajax函数在前端开发中的应用与实践 随着Web应用的快速发展,前端开发变得越来越重要。而Ajax作为一种前端开发技术,能够实现无需刷新页面的数据交互,成为了前端开发中不可或缺的工具。本文将介绍Ajax函数的基本原理,以及在前端开发中的应用与实践,并提供具体的代码示例。 Ajax函数的基本原理Aja…

    2025年12月24日
    000
  • 学会从头开始学习CSS,掌握制作基本网页框架的技巧

    从零开始学习CSS,掌握网页基本框架制作技巧 前言: 在现今互联网时代,网页设计和开发是一个非常重要的技能。而学习CSS(层叠样式表)是掌握网页设计的关键之一。CSS不仅可以为网页添加样式和布局,还可以为用户呈现独特且具有吸引力的页面效果。在本文中,我将为您介绍一些基本的CSS知识,以及一些常用的代…

    2025年12月24日
    200
  • 从初学到专业:掌握这五种前端CSS框架

    CSS是网站设计中重要的一部分,它控制着网站的外观和布局。前端开发人员为了让页面更加美观和易于使用,通常使用CSS框架。这篇文章将带领您了解这五种前端CSS框架,从入门到精通。 Bootstrap Bootstrap是最受欢迎的CSS框架之一。它由Twitter公司开发,具有可定制的响应式网格系统、…

    2025年12月24日
    200
  • 揭秘Web标准涵盖的语言:了解网页开发必备的语言范围

    在当今数字时代,互联网成为了人们生活中不可或缺的一部分。作为互联网的基本构成单位,网页承载着我们获取和分享信息的重要任务。而网页开发作为一门独特的技术,离不开一些必备的语言。本文将揭秘Web标准涵盖的语言,让我们一起了解网页开发所需的语言范围。 首先,HTML(HyperText Markup La…

    2025年12月24日
    000
  • 克服害怕做选择的恐惧症:这五个前端CSS框架将为你解决问题

    选择恐惧症?这五个前端CSS框架能帮你解决问题 近年来,前端开发者已经进入了一个黄金时代。随着互联网的快速发展,人们对于网页设计和用户体验的要求也越来越高。然而,要想快速高效地构建出漂亮的网页并不容易,特别是对于那些可能对CSS编码感到畏惧的人来说。所幸的是,前端开发者们早已为我们准备好了一些CSS…

    2025年12月24日
    200
  • 揭开Web开发的语言之谜:了解构建网页所需的语言有哪些?

    Web标准中的语言大揭秘:掌握网页开发所需的语言有哪些? 随着互联网的快速发展,网页开发已经成为人们重要的职业之一。而要成为一名优秀的网页开发者,掌握网页开发所需的语言是必不可少的。本文将为大家揭示Web标准中的语言大揭秘,介绍网页开发所需的主要语言。 HTML(超文本标记语言)HTML是网页开发的…

    2025年12月24日
    400
  • 常用的网页开发语言:了解Web标准的要点

    了解Web标准的语言要点:常见的哪些语言应用在网页开发中? 随着互联网的不断发展,网页已经成为人们获取信息和交流的重要途径。而要实现一个高质量、易用的网页,离不开一种被广泛接受的Web标准。Web标准的制定和应用,涉及到多种语言和技术,本文将介绍常见的几种语言在网页开发中的应用。 首先,HTML(H…

    2025年12月24日
    000
  • 网页开发中常见的Web标准语言有哪些?

    探索Web标准语言的世界:网页开发中常用的语言有哪些? 在现代社会中,互联网的普及程度越来越高,网页已成为人们获取资讯、娱乐、交流的重要途径。而网页的开发离不开各种编程语言的应用和支持。在这个虚拟世界的网络,有许多被广泛应用的标准化语言,用于为用户提供优质的网页体验。本文将探索网页开发中常用的语言,…

    2025年12月24日
    000
  • 深入探究Web标准语言的范围,涵盖了哪些语言?

    Web标准是指互联网上的各个网页所需遵循的一系列规范,确保网页在不同的浏览器和设备上能够正确地显示和运行。这些标准包括HTML、CSS和JavaScript等语言。本文将深入解析Web标准涵盖的语言范围。 首先,HTML(HyperText Markup Language)是构建网页的基础语言。它使…

    2025年12月24日
    000
  • 深入理解CSS框架与JS之间的关系

    深入理解CSS框架与JS之间的关系 在现代web开发中,CSS框架和JavaScript (JS) 是两个常用的工具。CSS框架通过提供一系列样式和布局选项,可以帮助我们快速构建美观的网页。而JS则提供了一套功能强大的脚本语言,可以为网页添加交互和动态效果。本文将深入探讨CSS框架和JS之间的关系,…

    2025年12月24日
    000
  • 项目实践:如何结合CSS和JavaScript打造优秀网页的经验总结

    项目实践:如何结合CSS和JavaScript打造优秀网页的经验总结 随着互联网的快速发展,网页设计已经成为了各行各业都离不开的一项技能。优秀的网页设计可以给用户留下深刻的印象,提升用户体验,增加用户的黏性和转化率。而要做出优秀的网页设计,除了对美学的理解和创意的运用外,还需要掌握一些基本的技能,如…

    2025年12月24日
    200

发表回复

登录后才能评论
关注微信