0

0

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

聖光之護

聖光之護

发布时间:2025-10-26 13:45:17

|

277人浏览过

|

来源于php中文网

原创

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

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

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

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

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

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

奇布塔
奇布塔

基于AI生成技术的一站式有声绘本创作平台

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

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

user.ejs 改造:

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



    
    
    
    User


    
    

SEARCH HERE




<% if(search_results && search_results.length > 0) { %> <% search_results.forEach((result) => { %>

<%= result.id %>

<%= result.name %>

<%= result.description %>

<% }) %> <% } else { %>

No results found

<% } %>

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

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

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

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

418

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

535

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

311

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

77

2025.09.10

ajax教程
ajax教程

php中文网为大家带来ajax教程合集,Ajax是一种用于创建快速动态网页的技术。通过在后台与服务器进行少量数据交换,Ajax可以使网页实现异步更新。这意味着可以在不重新加载整个网页的情况下,对网页的某部分进行更新。php中文网还为大家带来ajax的相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

159

2023.06.14

ajax中文乱码解决方法
ajax中文乱码解决方法

ajax中文乱码解决方法有设置请求头部的字符编码、在服务器端设置响应头部的字符编码和使用encodeURIComponent对中文进行编码。本专题为大家提供ajax中文乱码相关的文章、下载、课程内容,供大家免费下载体验。

160

2023.08.31

ajax传递中文乱码怎么办
ajax传递中文乱码怎么办

ajax传递中文乱码的解决办法:1、设置统一的编码方式;2、服务器端编码;3、客户端解码;4、设置HTTP响应头;5、使用JSON格式。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

117

2023.11.15

ajax网站有哪些
ajax网站有哪些

使用ajax的网站有谷歌、维基百科、脸书、纽约时报、亚马逊、stackoverflow、twitter、hacker news、shopify和basecamp等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

234

2024.09.24

php中文乱码如何解决
php中文乱码如何解决

本文整理了php中文乱码如何解决及解决方法,阅读节专题下面的文章了解更多详细内容。

1

2026.01.28

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 4.2万人学习

TypeScript 教程
TypeScript 教程

共19课时 | 2.5万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号