0

0

React.js 中实现产品列表按分类筛选的完整解决方案

花韻仙語

花韻仙語

发布时间:2026-03-09 19:37:03

|

140人浏览过

|

来源于php中文网

原创

React.js 中实现产品列表按分类筛选的完整解决方案

本文详解如何在 React 前端安全、可逆地实现产品分类过滤,避免因状态覆盖导致的“二次筛选失效”问题,并提供前后端协同优化建议。

本文详解如何在 react 前端安全、可逆地实现产品列表按分类筛选,避免因状态覆盖导致的“二次筛选失效”问题,并提供前后端协同优化建议。

在 React 应用中对产品列表(ProductList)进行前端分类过滤时,一个常见却隐蔽的陷阱是:直接用 setProductList(filteredProducts) 覆盖原始产品列表状态。这会导致后续筛选始终基于上一次的子集执行——例如先选 “Roll” 得到 5 个产品,再切回 “Sashimi”,系统却在仅含 Roll 的 5 条数据中查找 Sashimi,结果为空。根本原因在于:原始全量数据丢失,过滤逻辑失去基准

✅ 正确做法:分离「源数据」与「视图数据」

应将产品数据状态拆分为两部分:

亿众购物系统
亿众购物系统

一套设计完善、高效的web商城解决方案,独有SQL注入防范、对非法操作者锁定IP及记录功能,完整详细的记录了非法操作情况,管理员可以随时查看网站安全日志以及解除系统自动锁定的IP等前台简介:  1)系统为会员制购物,无限会员级别。  2)会员自动升级、相应级别所享有的折扣不同。  3)产品可在缺货时自动隐藏。  4)自动统计所有分类中商品数量,并在商品分类后面显示。  5)邮件列表功能,可在线订阅

下载
  • originalProducts:只读的原始全量列表(从 Context 或 API 初始加载一次,永不修改);
  • filteredProducts:仅用于渲染的派生列表(每次筛选重新计算,不修改源数据)。

以下是重构后的 Filter 组件完整实现:

import React, { useState, useContext, useMemo } from 'react';
import { ContextProductList } from '../app';
import { Dropdown } from 'primereact/dropdown';

function Filter() {
  const [category, setCategory] = useState('All'); // 默认显示全部
  const [originalProducts] = useContext(ContextProductList); // ✅ 只读取,不设为 setter

  // 预定义分类选项(含 "Tutto" 映射为 "All")
  const categoryList = [
    { label: 'Tutto', value: 'All' },
    { label: 'Roll', value: 'Roll' },
    { label: 'Sashimi', value: 'Sashimi' },
    { label: 'Uramaki', value: 'Uramaki' },
    { label: 'Bevande', value: 'Bevande' },
    { label: 'Dessert', value: 'Dessert' },
    { label: 'Ramen', value: 'Ramen' },
    { label: 'Speciali', value: 'Special' },
    { label: 'Altro', value: 'Altro' }
  ];

  // ✅ 使用 useMemo 缓存过滤结果,提升性能 & 避免重复计算
  const filteredProducts = useMemo(() => {
    if (!originalProducts || originalProducts.length === 0) return [];
    if (category === 'All') return originalProducts;
    return originalProducts.filter(product => product.category === category);
  }, [originalProducts, category]);

  const handleCategoryChange = (e: { value: string }) => {
    setCategory(e.value);
  };

  return (
    <form>
      <label htmlFor="categorySelect">Seleziona una categoria:</label>
      <Dropdown
        id="categorySelect"
        value={category}
        options={categoryList}
        placeholder="Seleziona una categoria"
        onChange={handleCategoryChange}
        className="w-full md:w-20rem"
      />
      {/* 可选:显示当前筛选结果数 */}
      <div className="mt-2 text-sm text-gray-600">
        Mostrati {filteredProducts.length} prodotti
      </div>
    </form>
  );
}

export default Filter;

⚠️ 关键注意事项

  • 不要修改原始状态:ContextProductList 应只提供 originalProducts(数组),而非 [products, setProducts] —— 若必须共享 setter,请在 Context 提供方封装 resetToOriginal() 方法。
  • 后端 category === "all" 处理需严格一致:你服务端代码中使用了 equalsIgnoreCase("all"),但前端传的是 "All"(首字母大写)。建议统一为小写(如 "all"),或在 Controller 中标准化:
    @GetMapping("/products/{id}/{category}")
    public List<Product> getProductsByCategory(
        @PathVariable String id, 
        @PathVariable String category) {
        String normalizedCategory = StringUtils.defaultString(category).toLowerCase();
        return productService.getAllProductsByCategoryForUserShop(id, normalizedCategory);
    }
  • 空结果处理:当前 findByCategoryAndUserShop 返回 null 而非空集合,会引发 NPE。务必在 Repository 层确保返回 Collections.emptyList():
    @Query("SELECT p FROM Product p WHERE p.userShop.id = :userShopId AND p.category = :category")
    List<Product> findByCategoryAndUserShop(@Param("userShopId") String id, @Param("category") String category);
    // Spring Data JPA 默认返回空列表,无需额外判空

? 前后端协作建议

场景 推荐方案 说明
少量产品( ✅ 纯前端过滤 响应快、降低请求压力,适合本例中的餐厅菜单场景
海量产品或需搜索/分页 ? 后端分页 + 过滤 调用 /products/{id}?category=roll&page=1&size=20,避免传输冗余数据
多租户隔离(userShop) ✅ 已正确实现 所有接口均携带 userShopId,保障数据边界安全

通过分离源数据与视图状态、合理使用 useMemo、统一前后端参数规范,即可彻底解决“筛选一次后失效”的问题,构建健壮、可维护的产品过滤功能。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

253

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

1069

2024.03.01

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1876

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

636

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

2382

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

47

2026.01.19

js正则表达式
js正则表达式

php中文网为大家提供各种js正则表达式语法大全以及各种js正则表达式使用的方法,还有更多js正则表达式的相关文章、相关下载、相关课程,供大家免费下载体验。

530

2023.06.20

js获取当前时间
js获取当前时间

JS全称JavaScript,是一种具有函数优先的轻量级,解释型或即时编译型的编程语言;它是一种属于网络的高级脚本语言,主要用于Web,常用来为网页添加各式各样的动态功能。js怎么获取当前时间呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

556

2023.07.28

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

59

2026.03.06

热门下载

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

相关下载

更多

精品课程

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

共58课时 | 5.9万人学习

国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1万人学习

React核心原理新老生命周期精讲
React核心原理新老生命周期精讲

共12课时 | 1.1万人学习

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

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