精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

那個幾乎沒人提卻超好用的 React Hook 清理范式

開發 前端
很多開發者在處理 WebSocket、??setInterval?? 計時器,或是不會自己收尾的第三方庫時,都會撞上同樣的問題。我也經歷過:因為漏掉一個清理動作,看著本來順滑的界面逐漸變卡,這種無力感——太熟悉了。

上周我在排查一個讓 React 應用性能直線下滑的內存泄漏時,意外挖到一個很“冷門”的套路。

很多開發者在處理 WebSocket、setInterval 計時器,或是不會自己收尾的第三方庫時,都會撞上同樣的問題。我也經歷過:因為漏掉一個清理動作,看著本來順滑的界面逐漸變卡,這種無力感——太熟悉了。

今天一起過一套能解決99% 內存泄漏的實踐:這是不少專業團隊在用、卻很少系統寫下來的“統一清理策略”。

正在吞噬你應用的內存泄漏

設想一個場景:組件里訂閱了 WebSocket,設了幾個定時器,還掛了第三方分析 SDK。

一切都很順,直到用戶在頁面間來回切換;組件裝載—卸載重復發生,而后臺任務仍在偷偷運行,持續吃掉內存與 CPU。

接著你會見到這條經典警告:

Warning: Can’t perform a React state update on an unmounted component.

這基本就宣告:發生泄漏了。因此,我們需要一種既穩妥又易于復用的收尾方法。

AbortController:萬物可撤銷的“清理總開關”

在折騰 useEffect 的時候,我重新審視了 AbortController,然后它就成了我的新“必帶工具”。

多數人以為它只配合 fetch 用,其實幾乎所有異步/可訂閱操作都能用這把“信號”來管理和取消。

關鍵點在于:一個控制器,統籌多路清理——一次 abort(),多處收尾。

useEffect(() => {
  const ctrl = new AbortController();
  const { signal } = ctrl;

  // WebSocket
  const ws = new WebSocket('wss://api.example.com');

  // 事件監聽:原生支持 signal 選項(現代瀏覽器)
  const handleResize = () => { /* ... */ };
  const handleEscape = (e) => { /* ... */ };
  window.addEventListener('resize', handleResize, { signal });
  document.addEventListener('keydown', handleEscape, { signal });

  // 其它自定義清理項集中登記
  const cleanups = [];

  // 第三方庫
  const analytics = new Analytics();
  analytics.connect();
  cleanups.push(() => analytics.disconnect());

  // 定時器
  const id = setInterval(updateData, 1000);
  cleanups.push(() => clearInterval(id));

  // 統一收尾
  return () => {
    ctrl.abort();               // 一鍵撤銷:事件監聽自動移除
    ws.close();                 // 關閉 WebSocket
    cleanups.forEach(fn => fn()); // 執行自定義清理
  };
}, []);

妙處在于:一次 ctrl.abort() 就能讓帶 signal 的監聽器自動解除;因此,不必為每個 handler 手動 removeEventListener。與此同時,你還能把無法“自動撤銷”的資源統一登記,最后集中回收。

WebSocket 的“可驗證清理”寫法

WebSocket 極易在卸載后殘留回調、繼續觸發狀態更新。下面這段實踐非常耐用:

const useWebSocket = (url) => {
  const [socket, setSocket] = React.useState(null);
  const [messages, setMessages] = React.useState([]);

  React.useEffect(() => {
    const ws = new WebSocket(url);
    const ctrl = new AbortController();

    ws.onopen = () => {
      console.log('WebSocket connected');
      setSocket(ws);
    };

    ws.onmessage = (evt) => {
      // 組件仍然“存活”才更新
      if (!ctrl.signal.aborted) {
        setMessages(prev => [...prev, JSON.parse(evt.data)]);
      }
    };

    ws.onerror = (err) => {
      console.error('WebSocket error:', err);
    };

    return () => {
      ctrl.abort();
      ws.close(1000, 'Component unmounting'); // 優雅關閉
      setSocket(null);
    };
  }, [url]);

  return { socket, messages };
};

關鍵洞見:更新狀態前**檢查 ctrl.signal.aborted**。因此,卸載后的“過期回調”會被自然短路,告別那條煩人的警告。盡管如此,別忘了服務端也要容錯,避免頻繁斷連導致的反復重連。

計時器:用數組一次性回收

計時類任務也是泄漏高發區。管理多路 setInterval 的干凈寫法如下:

const useMultipleTimers = () => {
  const [counts, setCounts] = React.useState({ fast: 0, slow: 0 });

  React.useEffect(() => {
    const timers = [];

    // 快速計數
    timers.push(
      setInterval(() => {
        setCounts(prev => ({ ...prev, fast: prev.fast + 1 }));
      }, 100)
    );

    // 慢速計數
    timers.push(
      setInterval(() => {
        setCounts(prev => ({ ...prev, slow: prev.slow + 1 }));
      }, 1000)
    );

    // 統一清理
    return () => {
      timers.forEach(clearInterval);
    };
  }, []);

  return counts;
};

小技巧:把定時器 ID 放進一個容器,卸載時一網打盡;因此,越多的計時任務,越能體現這種寫法的可擴展性。

第三方庫訂閱:適配任何“可退訂”接口

很多庫會返回 subscription 或 unsubscribe 之類的句柄。我們可以把“訂閱—撤銷”與“存活檢查”統一起來:

const useThirdPartySubscription = (config) => {
  const [data, setData] = React.useState(null);
  const subRef = React.useRef(null);

  React.useEffect(() => {
    const ctrl = new AbortController();
    let sub = null;

    const setup = async () => {
      try {
        const lib = await import('some-analytics-lib');
        sub = lib.subscribe(config, (next) => {
          if (!ctrl.signal.aborted) {
            setData(next);
          }
        });
        subRef.current = sub;
      } catch (e) {
        console.error('Subscription setup failed:', e);
      }
    };

    setup();

    return () => {
      ctrl.abort();
      if (sub && typeof sub.unsubscribe === 'function') {
        sub.unsubscribe();
      }
      subRef.current = null;
    };
  }, [config]);

  return data;
};

這在 Socket.IO、Firebase 等“訂閱型服務”上同樣好用;與此同時,你還能在 catch/finally 分支里保留容錯與重試策略。

進階:建立“清理注冊表”

當一個組件里有多條生命周期資源時,建立“注冊表”能把清理邏輯集中到一個出口,因此幾乎不可能漏項。

const useCleanupRegistry = () => {
  const bucketRef = React.useRef([]);

  const register = React.useCallback((fn) => {
    if (typeof fn === 'function') bucketRef.current.push(fn);
  }, []);

  React.useEffect(() => {
    return () => {
      bucketRef.current.forEach(fn => {
        try { fn(); } catch (e) { console.error('Cleanup failed:', e); }
      });
      bucketRef.current = [];
    };
  }, []);

  return register;
};

// 使用示例
const MyComponent = () => {
  const register = useCleanupRegistry();

  React.useEffect(() => {
    // WebSocket
    const ws = new WebSocket('wss://api.example.com');
    register(() => ws.close());

    // Interval
    const interval = setInterval(() => {/* ... */}, 1000);
    register(() => clearInterval(interval));

    // DOM 事件
    const onClick = () => console.log('clicked');
    document.addEventListener('click', onClick);
    register(() => document.removeEventListener('click', onClick));
  }, [register]);

  return <div>Component content</div>;
};

這種模式把“分散的清理點”抽象成“可登記的回調”,因此結構清晰;然而,請控制嵌套層級,避免過度包裝導致的可讀性下降。

現代 fetch 與 AbortController:所有狀態更新都先“問一嘴”

人人都該掌握的 fetch 清理范式如下。核心就是:**每一次狀態寫入之前,先檢查 aborted**。

const useFetchData = (url) => {
  const [data, setData] = React.useState(null);
  const [loading, setLoading] = React.useState(false);
  const [error, setError] = React.useState(null);

  React.useEffect(() => {
    const ctrl = new AbortController();

    setLoading(true);
    setError(null);

    fetch(url, { signal: ctrl.signal })
      .then(res => res.json())
      .then(json => {
        if (!ctrl.signal.aborted) setData(json);
      })
      .catch(err => {
        if (err.name !== 'AbortError' && !ctrl.signal.aborted) {
          setError(err.message);
        }
      })
      .finally(() => {
        if (!ctrl.signal.aborted) setLoading(false);
      });

    return () => ctrl.abort();
  }, [url]);

  return { data, loading, error };
};

因此,即使出現競態、慢請求或快速切頁,也不會把“過期結果”寫回已經卸載的組件;與此同時,錯誤分支也能保持干凈。

為什么這套能擋住 99% 的泄漏

基于 AbortController 的統一策略之所以有效,是因為它:

  • 集中管理:一個控制器可統籌多路資源與事件;
  • 阻斷陳舊更新:卸載后 signal.aborted 讓過期回調自動失效;
  • 天然可擴展:新增任意異步源,只需登記清理或傳入同一 signal
  • 覆蓋邊角:競態、快速切換與多訂閱并存時,依然能保持一致的收尾路徑。

簡潔之美正在于此:你無需逐條記住“第 N 個監聽要怎么解綁”,只要設好控制器 + 注冊表,其余交給機制自己協調。因此,整個團隊的認知負擔也明顯下降。

最后的要點

許多成熟團隊都在采用這套模式:它穩健、易維護,并且能讓復雜頁面在長時間使用后依舊清爽。 不妨在下一個需求里試試,然后告訴我你的體驗與改進建議;盡管如此,如果你在接入遺留 SDK 時遇到特殊清理邊界,也歡迎留言交流。

感謝閱讀。我們下次繼續聊一個更“好玩”的 React 小技巧。

責任編輯:武曉燕 來源: 大遷世界
相關推薦

2022-04-14 11:50:39

函數組件hook

2023-06-05 11:32:47

LinuxGNU/Linux

2025-05-14 00:00:01

2020-05-28 13:33:30

React Hook前端開發

2024-07-16 09:51:39

HTMLHookReact

2021-05-14 09:49:47

React HookReact應用

2025-08-11 04:00:00

開源項目PR

2022-12-23 08:34:30

HookReact

2022-08-01 07:56:23

React Hook開發組件

2020-10-21 11:34:49

React Hook庫

2023-02-14 06:40:33

React HookReact

2021-03-26 09:00:00

開發框架React

2020-10-28 09:37:08

React代碼數據

2022-05-06 07:31:01

useEventReactHook

2020-07-22 13:50:39

shell命令前端

2024-05-27 07:48:23

2019-07-02 10:36:30

JavaScript硬件開發

2022-05-11 07:50:15

React UI組件庫前端

2020-02-17 15:49:11

Python開發

2024-03-19 00:00:00

ReactJavaScript開發
點贊
收藏

51CTO技術棧公眾號

精品国产欧美一区二区五十路| 狠狠做深爱婷婷久久综合一区| 成人免费福利在线| 欧美日韩在线国产| 蜜桃成人av| 91精品久久久久久久99蜜桃| 91专区在线观看| 调教视频免费在线观看| 国产**成人网毛片九色| 国产97在线播放| 九九视频免费在线观看| 国产成人三级| 日韩欧美视频在线| 91视频免费版污| 国产精品一二三产区| 亚洲欧洲性图库| 久久99精品久久久久久秒播放器| 一本一道精品欧美中文字幕| 国产亚洲毛片| 欧美肥婆姓交大片| 黄色av免费播放| 老司机在线精品视频| 欧美一区欧美二区| 九九视频精品在线观看| 国模私拍一区二区国模曼安| 亚洲婷婷综合色高清在线| 欧美日韩精品一区| 性xxxx18| 99久久99精品久久久久久| 91在线免费网站| 中文在线免费看视频| 免费亚洲网站| 97在线免费观看| 欧美黄色一级网站| 亚洲天天影视网| 日韩在线观看免费全集电视剧网站| 亚洲一区二区观看| 激情小说亚洲图片| 亚洲国产成人精品女人久久久 | 国内精品写真在线观看| 国产精品第10页| 综合网在线观看| 国产欧美一级| 欧美在线视频播放| 五月婷婷亚洲综合| 9久re热视频在线精品| 欧美日本在线视频中文字字幕| 欧美视频一区二区在线| 成人影院在线| 色伦专区97中文字幕| 日韩福利在线视频| 日本一区二区高清不卡| 丝袜情趣国产精品| 粉嫩精品久久99综合一区| 精品久久91| 在线观看欧美日韩国产| 亚洲国产精品一区二区久久hs| 欧美电影《轻佻寡妇》| 精品国产一区av| 中国毛片直接看| 在线成人激情| 久久久久中文字幕| 五月婷婷中文字幕| 日本中文字幕一区| 国产日韩欧美91| 999av视频| 成人毛片在线观看| 久久精品中文字幕一区二区三区| 香蕉视频网站在线| 国产精品麻豆视频| 国产av不卡一区二区| 日韩av官网| 欧美性69xxxx肥| 五月天婷婷亚洲| 亚洲三区欧美一区国产二区| 亚洲精品wwww| 阿v天堂2014| 亚洲免费二区| 91av免费观看91av精品在线| 波多野结衣mp4| 极品尤物av久久免费看| 高清一区二区三区视频| 深夜福利在线视频| 中文字幕中文字幕在线一区| 久久久国内精品| 欧美中文字幕精在线不卡| 欧美疯狂性受xxxxx喷水图片| 亚洲熟女乱综合一区二区| 任你躁在线精品免费| 在线播放日韩av| 久久久精品人妻一区二区三区四 | 亚洲人成影院在线观看| 亚洲不卡中文字幕无码| 欧美激情三区| 亚洲国产欧美久久| 久久噜噜色综合一区二区| 在线日韩av| 国产美女久久精品香蕉69| 免费的黄色av| 成人欧美一区二区三区白人 | 国产酒店精品激情| 欧美婷婷久久| 国产丝袜精品丝袜| 欧美日韩小视频| 国产伦精品一区二区三区精品| 成人情趣视频网站| 97国产真实伦对白精彩视频8| 又污又黄的网站| 26uuu国产电影一区二区| 300部国产真实乱| 97欧美成人| 亚洲精品乱码久久久久久按摩观| 日本午夜在线观看| 日韩国产精品久久| 精品日产一区2区三区黄免费 | 国产成人小视频在线观看| www.色日本| 国产精品伦理一区二区| 漂亮人妻被中出中文字幕| 亚洲电影一区| 超碰97人人做人人爱少妇| 国产99久久久久久免费看| 成人avav影音| 久久亚洲国产成人精品无码区 | 国产精品一级久久久| 日韩在线观看www| 色婷婷国产精品| 黄色在线免费播放| 欧美1区视频| 成人激情视频网| 麻豆电影在线播放| 欧美中文字幕一区二区三区| 国产伦精品一区二区三区妓女| 欧美日韩国产亚洲一区| 91在线中文字幕| 精品视频在线一区二区| 欧美久久久久久久久| 波多野结衣一二三四区| 日韩av一二三| 午夜精品区一区二区三| 欧美三级精品| 亚洲性69xxxbbb| 黄色av网站免费| 久久久国产精华| 久久久久狠狠高潮亚洲精品| 日韩深夜福利| 欧美中文字幕视频| 欧洲综合视频| 色婷婷av一区二区三区软件| 精品无码国产污污污免费网站| 日日夜夜免费精品| 亚洲国产日韩美| 成人豆花视频| 欧美精品制服第一页| 草草视频在线播放| 亚洲二区在线观看| 青青草视频成人| 久久精品动漫| 午夜精品区一区二区三| 色8久久久久| 久久99精品久久久久久噜噜 | www.国产com| 国产亚洲欧美一级| 免费看涩涩视频| 午夜久久美女| 精品欧美一区二区三区久久久| 一区二区电影免费观看| 视频在线观看一区二区| 亚洲av永久无码国产精品久久| 亚洲午夜精品久久久久久久久| 国产一级二级在线观看| 日韩高清不卡一区二区| 女同性恋一区二区| 色婷婷av一区二区三区丝袜美腿 | 久久69av| 国模精品视频一区二区| 国产永久免费高清在线观看视频| 欧美日韩国产片| 久草免费新视频| 久久综合中文字幕| 五月天激情播播| 亚洲人成人一区二区三区| 欧美午夜精品久久久久免费视| 欧美综合影院| 97久久久久久| 色综合久久影院| 亚洲福利视频在线| 亚洲图片小说视频| 亚洲成av人片在线观看| 免费看黄色三级| 成人小视频免费观看| 国产精品天天av精麻传媒| 午夜天堂精品久久久久| 欧美在线3区| 视频免费一区二区| 国产精品www| 97人人爽人人澡人人精品| 中文字幕日韩欧美在线| 日本国产在线观看| 欧美精品视频www在线观看| 国产香蕉视频在线| 日韩一区在线看| 日本丰满少妇裸体自慰| 国产伦精品一区二区三区视频青涩 | 国产精品一区二区久久不卡 | 国产精品自在在线| 99久久国产宗和精品1上映| 国产精品草草| www.黄色网址.com| 精品美女视频| 久久av一区二区三区漫画| 亚洲国产一区二区三区网站| 国产精品露脸av在线| 涩涩涩在线视频| 欧美激情乱人伦一区| 免费a级在线播放| 在线不卡国产精品| 猫咪在线永久网站| 亚洲黄色av女优在线观看 | 欧美爱爱视频| 国产精品精品视频一区二区三区| 老色鬼在线视频| 欧美日韩xxx| 亚洲精品白浆| 美女国内精品自产拍在线播放| 在线观看av的网站| 一区二区三区动漫| 欧洲毛片在线| 日韩成人激情视频| 天天综合网在线观看| 日韩美一区二区三区| 国产又大又黑又粗| 欧美在线免费观看视频| 一级特黄免费视频| 在线观看www91| 黄色污污视频软件| 日本精品一级二级| 老熟妇一区二区三区| 色偷偷久久人人79超碰人人澡| 国产视频91在线| 亚洲v日本v欧美v久久精品| 激情五月婷婷小说| 亚洲成人av中文| 日本熟妇毛茸茸丰满| 欧美日韩免费看| 国产又大又黄又粗| 在线观看视频一区| 中文字幕人妻色偷偷久久| 欧美网站大全在线观看| 91福利免费视频| 91精品国产免费| 精品国产av 无码一区二区三区| 91精品国产品国语在线不卡| 99热这里只有精品在线| 精品人在线二区三区| 日本精品久久久久久| 日韩第一页在线| 九色视频网站在线观看| 中文字幕免费国产精品| 久操视频在线免费播放| 欧美日韩国产成人| 天堂√中文最新版在线| 国产成人精品视频在线| 欧美成人xxxx| 国产精品视频免费观看| 日韩精品丝袜美腿| 午夜精品一区二区在线观看的| 91久久久精品国产| 精品国产av无码一区二区三区| 亚洲一区一卡| 天天操天天干天天做| 粉嫩av一区二区三区粉嫩| 中文字幕在线免费看线人| 中文字幕va一区二区三区| 欧美日韩精品一区二区三区视频播放| 亚洲成av人**亚洲成av**| 国产黄网在线观看| 91精品国产黑色紧身裤美女| 天堂av手机版| 中文字幕在线日韩| а√天堂8资源中文在线| 国产精品狠色婷| 午夜视频在线观看精品中文| 热re99久久精品国产99热| 欧美+亚洲+精品+三区| 欧美少妇性生活视频| 国产一区二区三区精品视频| 99久久人妻无码中文字幕系列| 国产女主播视频一区二区| 国产一级做a爰片在线看免费| 欧美小视频在线观看| av小说天堂网| 国产一区二区三区18| 黄页网站在线| 91精品国产综合久久香蕉的用户体验| 高清一区二区三区| 亚洲欧美日韩综合一区| 中文亚洲字幕| 成人在线短视频| 国产欧美精品国产国产专区 | 欧美色综合久久| 色哟哟中文字幕| 日韩有码在线电影| 中文字幕一区久| 国产精品视频免费一区二区三区| 国产精品麻豆久久| 韩国一区二区av| 成人性生交大片免费看视频在线| 91制片厂在线| 91官网在线观看| 亚洲 国产 欧美 日韩| 欧美久久久精品| 久久er热在这里只有精品66| 精品在线视频一区二区| 午夜久久99| 国内精品国产三级国产aⅴ久| 国产三级精品三级在线专区| 日韩精品国产一区二区| 欧美一卡二卡三卡四卡| 欧美18hd| 国产精品日韩欧美综合| 啪啪亚洲精品| www.四虎成人| 91视频91自| 黄色在线观看国产| 亚洲国产精品系列| 国内精彩免费自拍视频在线观看网址| 97se国产在线视频| 综合久久十次| 欧美性受xxxx黒人xyx性爽| 国产精品欧美精品| 一二三区在线播放| 神马久久桃色视频| 先锋影音网一区二区| 中文一区一区三区免费| 精品一区二区三区影院在线午夜| 国产传媒视频在线| 欧美日韩大陆在线| 麻豆传媒视频在线观看免费| 国产情人节一区| 久久久久久久久丰满| theporn国产精品| 亚洲人成7777| av官网在线观看| 久久福利网址导航| 欧美视频二区欧美影视| 国产一区二区三区在线免费| 成人自拍视频在线| 亚洲黄色一区二区| 亚洲欧美日韩网| 成人在线爆射| 亚洲欧美一区二区原创| 国产在线精品国自产拍免费| 黄视频网站免费看| 日韩精品中文字幕在线不卡尤物| 污的网站在线观看| 精品午夜一区二区| 久久综合九色综合欧美狠狠| 日本污视频网站| 欧美一区二区女人| 91超碰在线播放| 欧美在线播放一区| 久久精品国产一区二区三区免费看| 最新av电影网站| 精品免费一区二区三区| 伊人久久国产| 一区二区三区av在线| 成人一区二区三区在线观看| 丁香六月婷婷综合| 色爱精品视频一区| 中文字幕一区二区三区中文字幕| 欧美三级一级片| 国产精品美女久久久久久久久 | 欧美天堂在线| 99国产精品白浆在线观看免费| www.欧美日韩| 最近中文字幕在线视频| 欧美丰满少妇xxxxx| 一区二区美女| 超碰在线超碰在线| 欧美日韩亚洲一区二| 麻豆视频在线观看免费网站| 国产日韩久久| 久久国产精品第一页| 日本少妇xxxx动漫| 日韩有码在线观看| 要久久爱电视剧全集完整观看| 一级黄色录像在线观看| 午夜精品影院在线观看| 免费av不卡| 久久久影院一区二区三区| 狠狠色狠狠色综合日日91app| 中文字幕在线观看视频网站| 久久亚洲私人国产精品va| 伊人久久综合影院| 国产精品成人99一区无码| 在线电影国产精品| 欧美日韩大片| 人妻无码久久一区二区三区免费| 国产精品传媒视频|