Momentjs对象可变性导致DatePicker异常(React+Antd)

今天在使用受控模式的RangePicker时遇到的bug

以下是两段完全一样的代码,唯一区别是前者使用momentjs,后者使用dayjs:

js 复制代码
import { useState } from "react";
import { DatePicker } from "antd";
import moment from "moment";

const { RangePicker } = DatePicker;

export const ControlledRangePicker = () => {
    const [selectedRange, setSelectedRange] = useState([
        moment().subtract(24, "hour"),
        moment(),
    ]);
    const handleRangeChange = (dates) => {
        if (dates) {
            setSelectedRange(dates);
        }
    };
    return (
        <div>
            <RangePicker
                value={selectedRange}
                onChange={handleRangeChange}
                showTime={{
                    format: "HH:mm",
                    hourStep: 1,
                    minuteStep: 15,
                }}
                format="YYYY-MM-DD HH:mm"
            />
        </div>
    );
};


import dayjs from "dayjs";
export const ControlledRangePicker2 = () => {
    const [selectedRange, setSelectedRange] = useState([
        dayjs().subtract(24, "hour"),
        dayjs(),
    ]);
    const handleRangeChange = (dates) => {
        if (dates) {
            setSelectedRange(dates);
        }
    };
    return (
        <div>
            <RangePicker
                value={selectedRange}
                onChange={handleRangeChange}
                showTime={{
                    format: "HH:mm",
                    hourStep: 1,
                    minuteStep: 15,
                }}
                format="YYYY-MM-DD HH:mm"
            />
        </div>
    );
};

会发现前者的日期选择面板一打开 日期就会疯狂增长,控制台重复报错:

js 复制代码
Warning: Maximum update depth exceeded. 
This can happen when a component calls setState inside useEffect,
but useEffect either doesn't have a dependency array, 
or one of the dependencies changes on every render. 

原因是 Moment 对象是可变对象(mutable),而dayjs是 不可变(immutable)的。 可以用以下代码测试:

js 复制代码
// moment可变性
const date1 = moment('2023-10-01');
const date2 = date1; // 这不是复制,而是引用同一个对象
// const date2 = date1.clone(); 正确的方式:创建副本

date1.add(1, 'day'); // 修改date1

console.log(date1.format('YYYY-MM-DD')); // 2023-10-02
console.log(date2.format('YYYY-MM-DD')); // 2023-10-02  date2也被改变了
js 复制代码
// dayjs 不可变性
const date1 = dayjs('2023-10-01');
const date2 = date1; 

const date3 = date1.add(1, 'day'); // 返回新的对象,不改变原对象

console.log(date1.format('YYYY-MM-DD')); // 2023-10-01  原对象不变
console.log(date2.format('YYYY-MM-DD')); // 2023-10-01  
console.log(date3.format('YYYY-MM-DD')); // 2023-10-02  新对象

React的渲染机制期望状态更新是可控的,但Moment.js的 mutable 特性会让状态变得不可预测 。

而且momentjs体积比dayjs大,antd v4到v5 也是从momentjs 变成了 dayjs。

相关推荐
OEC小胖胖1 天前
01|从 Monorepo 到发布产物:React 仓库全景与构建链路
前端·react.js·前端框架
2501_944711431 天前
构建 React Todo 应用:组件通信与状态管理的最佳实践
前端·javascript·react.js
困惑阿三1 天前
2025 前端技术全景图:从“夯”到“拉”排行榜
前端·javascript·程序人生·react.js·vue·学习方法
阿里巴啦1 天前
React+go实现AI 图像生成落地实践:文生图、图生图的工程项目
人工智能·react.js·ai作画·七牛云·ai生图·ai图生图
有意义1 天前
用心写好一个登录页:代码、体验与细节的平衡
前端·react.js·交互设计
fe小陈1 天前
聊聊我对 React Hook 不一样的理解
react.js
白兰地空瓶1 天前
React 性能优化的“卧龙凤雏”:useMemo 与 useCallback 到底该怎么用
react.js
白兰地空瓶1 天前
聊聊那个让 React 新手抓狂的“闭包陷阱”:Count 为什么永远是 0?
react.js
time_rg1 天前
深入理解react——1. jsx与虚拟dom
前端·react.js
用户12039112947261 天前
React 性能优化之道:useMemo、useCallback 与闭包陷阱的深度剖析
前端·javascript·react.js