0

0

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

碧海醫心

碧海醫心

发布时间:2026-03-09 17:37:01

|

756人浏览过

|

来源于php中文网

原创

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

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

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

在 React 应用中对商品列表(ProductList)进行分类筛选时,一个常见但隐蔽的陷阱是:直接修改原始产品列表的状态(setProductList(filtered)),会导致后续筛选失去数据源基础。正如你在代码中观察到的——首次筛选后 productList 状态已变为子集,再次切换分类时若该分类不在当前子集中,结果为空数组,体验中断。

✅ 正确做法:分离「原始数据」与「筛选视图」

不应将筛选结果写回全局产品状态,而应仅维护一个受控的筛选视图状态(如 filteredProducts),同时保留原始产品列表(allProducts)始终不变。推荐结构如下:

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

function Filter() {
  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' }
  ];

  const [category, setCategory] = useState('All');
  const [allProducts] = useContext(ContextProductList); // ✅ 只读取原始数据,不用于 setState

  // ✅ 使用 useMemo 实现高效、可缓存的筛选逻辑
  const filteredProducts = useMemo(() => {
    if (!allProducts || allProducts.length === 0) return [];
    if (category === 'All') return allProducts;
    return allProducts.filter(product => product.category === category);
  }, [allProducts, category]);

  // ✅ 向父组件或展示层传递筛选结果(例如通过 context 或 props)
  // 注意:此处不调用 setProductList!避免污染原始状态

  return (
    <div>
      <label htmlFor="categorySelect">Seleziona una categoria:</label>
      <Dropdown
        id="categorySelect"
        value={category}
        options={categoryList}
        placeholder="Seleziona una categoria"
        onChange={(e) => setCategory(e.value)}
      />
      {/* 示例:向下游组件传递筛选结果 */}
      <ProductListView products={filteredProducts} />
    </div>
  );
}

export default Filter;

? 关键改进点:

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

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

下载
  • useContext(ContextProductList) 仅用于读取原始数据,不再解构出 setProductList;
  • 引入 filteredProducts 作为派生状态(推荐用 useMemo 缓存,避免重复计算);
  • 所有 UI 渲染基于 filteredProducts,原始 allProducts 始终完好无损;
  • 切换分类时,useMemo 自动重新计算,无需手动管理中间状态。

⚠️ 注意事项与最佳实践

  • 不要在筛选逻辑中修改全局状态:setProductList(...) 应仅用于加载/刷新全量数据(如页面初始化、新增商品后),而非筛选过程。
  • 后端筛选更优场景:当商品量大(>1000 条)、分类维度多(如含价格区间、关键词搜索)、或需服务端权限校验时,应调用你已实现的后端接口 /products/{id}/{category}。前端只需传参并更新 filteredProducts:
    useEffect(() => {
      if (category === 'All') {
        setFilteredProducts(allProducts);
      } else {
        fetch(`/api/products/${userShopId}/${category}`)
          .then(res => res.json())
          .then(data => setFilteredProducts(data));
      }
    }, [category, userShopId, allProducts]);
  • 后端 NULL 问题排查:你提到 findByCategoryAndUserShop(id, category) 返回 null,请检查:
    • 数据库中 product.category 字段值是否严格匹配(大小写、空格、拼写);
    • JPA 方法命名是否正确(如 findByCategoryAndUserShopId 而非 findByCategoryAndUserShop);
    • 实体类中 @Column(name = "category") 映射是否准确;
    • 日志中 System.out.println(category) 输出是否与数据库一致(建议改用 SLF4J + %logger{36} 格式化输出)。

✅ 总结

前端筛选的核心原则是:状态不可变(Immutable)+ 视图可派生(Derived UI State)。用 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的相关内容,可以阅读本专题下面的文章。

1089

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号