記錄--原生 canvas 如何實現大屏?

来源:https://www.cnblogs.com/smileZAZ/archive/2023/02/01/17083434.html
-Advertisement-
Play Games

這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 前言 可視化大屏該如何做?有可能一天完成嗎?廢話不多說,直接看效果,線上 Demo 地址 lxfu1.github.io/large-scree…。 看完這篇文章(這個項目),你將收穫: 全局狀態真的很簡單,你只需 5 分鐘就能上手 如何 ...


這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助

前言

可視化大屏該如何做?有可能一天完成嗎?廢話不多說,直接看效果,線上 Demo 地址 lxfu1.github.io/large-scree…

看完這篇文章(這個項目),你將收穫:

  1. 全局狀態真的很簡單,你只需 5 分鐘就能上手
  2. 如何緩存函數,當入參不變時,直接使用緩存值
  3. 千萬節點的圖如何分片渲染,不卡頓頁面操作
  4. 項目單測該如何寫?
  5. 如何用 canvas 繪製各種圖表,如何實現 canvas 動畫
  6. 如何自動化部署自己的大屏網站

效果

實現

項目基於 Create React App --template typescript搭建,包管理工具使用的 pnpm ,pnpm 的優勢這裡不多介紹(快+節省磁碟空間),之前在其它平臺寫過相關文章,後續可能會搬過來。由於項目 package.json 裡面有限制包版本(最新版本的 G6 會導致 OOM,官方短時間能應該會修複),如果使用的 yarn 或 npm 的話,改為對應的 resolutions 即可。

 "pnpm": {
    "overrides": {
      "@antv/g6": "4.7.10"
    }
  }
"resolutions": {
  "@antv/g6": "4.7.10"
},

啟動

  1. clone項目
git clone https://github.com/lxfu1/large-screen-visualization.git
  1. pnpm 安裝 npm install -g pnpm
  2. 啟動: pnpm start 即可,建議配置 alias ,可以簡化各種命令的簡寫 eg:p start,不出意外的話,你可以通過 http://localhost:3000/ 訪問了
  3. 測試:p test
  4. 構建:p build

強烈建議大家先 clone 項目!

分析

全局狀態

全局狀態用的 valtio ,位於項目 src/models目錄下,強烈推薦。

優點:數據與視圖分離的心智模型,不再需要在 React 組件或 hooks 里用 useState 和 useReducer 定義數據,或者在 useEffect 里發送初始化請求,或者考慮用 context 還是 props 傳遞數據。

缺點:相容性,基於 proxy 開發,對低版本瀏覽器不友好,當然,大屏應該也不會考慮 IE 這類瀏覽器。

import { proxy } from "valtio";
import { NodeConfig } from "@ant-design/graphs";

type IState = {
  sliderWidth: number;
  sliderHeight: number;
  selected: NodeConfig | null;
};

export const state: IState = proxy({
  sliderWidth: 0,
  sliderHeight: 0,
  selected: null,
});

狀態更新:

import { state } from "src/models";

state.selected = e.item?.getModel() as NodeConfig;

狀態消費:

import { useSnapshot } from "valtio";
import { state } from "src/models";

export const BarComponent = () => {
  const snap = useSnapshot(state);

  console.log(snap.selected)
}

當我們選中圖譜節點的時候,由於 BarComponent 組件監聽了 selected 狀態,所以該組件會進行更新。有沒有感覺非常簡單?一些高級用法建議大家去官網查看,不再展開。

函數緩存

為什麼需要函數緩存?當然,在這個項目中函數緩存比較雞肋,為了用而用,試想,如果有一個函數計算量非常大,組件內又有多個 state 頻繁更新,怎麼確保函數不被重覆調用呢?可能大家會想到 useMemo``useCallback等手段,這裡要介紹的是 React 官方的 cache 方法,已經在 React 內部使用,但未暴露。實現上借鑒(抄襲)ReactCache通過緩存的函數 fn 及其參數列表來構建一個 cacheNode 鏈表,然後基於鏈表最後一項的狀態來作為函數 fn 與該組參數的計算緩存結果。

代碼位於 src/utils/cache

interface CacheNode {
  /**
   * 節點狀態
   *  - 0:未執行
   *  - 1:已執行
   *  - 2:出錯
   */
  s: 0 | 1 | 2;
  // 緩存值
  v: unknown;
  // 特殊類型(object,fn),使用 weakMap 存儲,避免記憶體泄露
  o: WeakMap<Function | object, CacheNode> | null;
  // 基本類型
  p: Map<Function | object, CacheNode> | null;
}

const cacheContainer = new WeakMap<Function, CacheNode>();

export const cache = (fn: Function): Function => {
  const UNTERMINATED = 0;
  const TERMINATED = 1;
  const ERRORED = 2;

  const createCacheNode = (): CacheNode => {
    return {
      s: UNTERMINATED,
      v: undefined,
      o: null,
      p: null,
    };
  };

  return function () {
    let cacheNode = cacheContainer.get(fn);
    if (!cacheNode) {
      cacheNode = createCacheNode();
      cacheContainer.set(fn, cacheNode);
    }
    for (let i = 0; i < arguments.length; i++) {
      const arg = arguments[i];
      // 使用 weakMap 存儲,避免記憶體泄露
      if (
        typeof arg === "function" ||
        (typeof arg === "object" && arg !== null)
      ) {
        let objectCache: CacheNode["o"] = cacheNode.o;
        if (objectCache === null) {
          objectCache = cacheNode.o = new WeakMap();
        }
        let objectNode = objectCache.get(arg);
        if (objectNode === undefined) {
          cacheNode = createCacheNode();
          objectCache.set(arg, cacheNode);
        } else {
          cacheNode = objectNode;
        }
      } else {
        let primitiveCache: CacheNode["p"] = cacheNode.p;
        if (primitiveCache === null) {
          primitiveCache = cacheNode.p = new Map();
        }
        let primitiveNode = primitiveCache.get(arg);
        if (primitiveNode === undefined) {
          cacheNode = createCacheNode();
          primitiveCache.set(arg, cacheNode);
        } else {
          cacheNode = primitiveNode;
        }
      }
    }
    if (cacheNode.s === TERMINATED) return cacheNode.v;
    if (cacheNode.s === ERRORED) {
      throw cacheNode.v;
    }
    try {
      const res = fn.apply(null, arguments as any);
      cacheNode.v = res;
      cacheNode.s = TERMINATED;
      return res;
    } catch (err) {
      cacheNode.v = err;
      cacheNode.s = ERRORED;
      throw err;
    }
  };
};

如何驗證呢?我們可以簡單看下單測,位於src/__tests__/utils/cache.test.ts

import { cache } from "src/utils";

describe("cache", () => {
  const primitivefn = jest.fn((a, b, c) => {
    return a + b + c;
  });

  it("primitive", () => {
    const cacheFn = cache(primitivefn);
    const res1 = cacheFn(1, 2, 3);
    const res2 = cacheFn(1, 2, 3);
    expect(res1).toBe(res2);
    expect(primitivefn).toBeCalledTimes(1);
  });
});

可以看出,即使我們調用了 2 次 cacheFn,由於入參不變,fn 只被執行了一次,第二次直接返回了第一次的結果。

項目裡面在做 circle 動畫的時候使用了,因為該動畫是繞圓周無限迴圈的,當迴圈過一周之後,後的動畫和之前的完全一致,沒必要再次計算對應的 circle 坐標,所以我們使用了 cache ,位於src/components/background/index.tsx。

  const cacheGetPoint = cache(getPoint);
  let p = 0;
  const animate = () => {
    if (p >= 1) p = 0;
    const { x, y } = cacheGetPoint(p);
    ctx.clearRect(0, 0, 2 * clearR, 2 * clearR);
    createCircle(aCtx, x, y, circleR, "#fff", 6);
    p += 0.001;
    requestAnimationFrame(animate);
  };
  animate();

分片渲染

你有審查元素嗎?項目背景圖是通過 canvas 繪製的,並不是背景圖片!通過 canvas 繪製如此多的小圓點,會不會阻礙頁面操作呢?當數據量足夠大的時候,是會阻礙的,大家可以把 NodeMargin 設置為 0.1 ,同時把 schduler 調用去掉,直接改為同步繪製。當節點數量在 500 W 的時候,如果沒有開啟切片,頁面白屏時間在 MacBook Pro M1 上白屏時間大概是 8.5 S;開啟分片渲染時頁面不會出現白屏,而是從左到右逐步繪製背景圖,每個任務的執行時間在 16S 左右波動。

  const schduler = (tasks: Function[]) => {
    const DEFAULT_RUNTIME = 16;
    const { port1, port2 } = new MessageChannel();
    let isAbort = false;

    const promise: Promise<any> = new Promise((resolve, reject) => {
      const runner = () => {
        const preTime = performance.now();
        if (isAbort) {
          return reject();
        }
        do {
          if (tasks.length === 0) {
            return resolve([]);
          }
          const task = tasks.shift();
          task?.();
        } while (performance.now() - preTime < DEFAULT_RUNTIME);
        port2.postMessage("");
      };
      port1.onmessage = () => {
        runner();
      };
    });
    // @ts-ignore
    promise.abort = () => {
      isAbort = true;
    };
    port2.postMessage("");
    return promise;
  };

分片渲染可以不阻礙用戶操作,但延遲了任務的整體時長,是否開啟還是取決於數據量。如果每個分片實際執行時間大於 16ms 也會造成阻塞,並且會堆積,並且任務執行的時候沒有等,最終渲染狀態和預期不一致,所以 task 的拆分也很重要。

單測

這裡不想多說,大家可以運行 pnpm test看看效果,環境已經搭建好;由於項目裡面用到了 canvas 所以需要 mock 一些環境,這裡的 mock 可以理解為“我們前端代碼跑在瀏覽器里運行,依賴了瀏覽器環境以及對應的 API,但由於單測沒有跑在瀏覽器裡面,所以需要 mock 瀏覽器環境”,例如項目裡面設置的 jsdom、jest-canvas-mock 以及 worker 等,更多推薦直接訪問 jest 官網。

// jest-dom adds custom jest matchers for asserting on DOM nodes.
import "@testing-library/jest-dom";

Object.defineProperty(URL, "createObjectURL", {
  writable: true,
  value: jest.fn(),
});

class Worker {
  onmessage: () => void;
  url: string;
  constructor(stringUrl) {
    this.url = stringUrl;
    this.onmessage = () => {};
  }

  postMessage() {
    this.onmessage();
  }
  terminate() {}
  onmessageerror() {}
  addEventListener() {}
  removeEventListener() {}
  dispatchEvent(): boolean {
    return true;
  }
  onerror() {}
}
window.Worker = Worker;

自動化部署

開發過項目的同學都知道,前端編寫的代碼最終是要進行部署的,目前比較流行的是前後端分離,前端獨立部署,通過 proxy 的方式請求後端服務;或者是將前端構建產物推到後端服務上,和後端一起部署。如何做自動化部署呢,對於一些不依賴後端的項目來說,我們可以藉助 github 提供的 gh-pages 服務來做自動化部署,CI、CD 僅需配置對應的 actions 即可,在倉庫 settings/pages 下麵選擇對應分支即可完成部署。

例如項目裡面的.github/workflows/gh-pages.yml,表示當 master 分支有代碼提交時,會執行對應的 jobs,並藉助 peaceiris/actions-gh-pages@v3將構建產物同步到 gh-pages 分支。

name: github pages

on:
  push:
    branches:
      - master # default branch
      
env:
  CI: false
  PUBLIC_URL: '/large-screen-visualization'

jobs:
  deploy:
    runs-on: ubuntu-latest
    steps:
      - uses: actions/checkout@v3
      - run: yarn
      - run: yarn build
      - name: Deploy
        uses: peaceiris/actions-gh-pages@v3
        with:
          github_token: ${{ secrets.GITHUB_TOKEN }}
          publish_dir: ./build

本文轉載於:

https://juejin.cn/post/7165564571128692773

如果對您有所幫助,歡迎您點個關註,我會定時更新技術文檔,大家一起討論學習,一起進步。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • Java實現BP神經網路,內含BP神經網路類,採用MNIST數據集,包含伺服器和客戶端程式,可在伺服器訓練後使客戶端直接使用訓練結果,界面有畫板,可以手寫數字 ...
  • 一 引入 考慮實現一種三軸機器人控制項。 三軸機器人用來將某種工件從一個位置運送到另一個位置。 其X軸為手臂軸,可以正向和反向運動,它處於末端,直接接觸工件; 其T軸為旋轉軸,可以對手臂進行旋轉; 其Z軸為升降軸,可以對手臂和旋轉部分進行升降。 二 RobotControl 定義出機器人的軸動作枚舉, ...
  • 前言 相信大家看過不少講C# async await的文章,博客園就能搜到很多,但還是有很多C#程式員不明白。 如果搞不明白,其實也不影響使用。但有人就會疑惑,為什麼要用非同步?我感覺它更慢了,跟同步有啥區別? 有的人研究深入,比如去研究狀態機,可能會明白其中的原理。但深入研究的畢竟少數。有的人寫一些 ...
  • 疑惑 最近在反覆搭建ceph集群過程中,總是遇到osd創建不成功的問題,疑似硬碟殘留信息,排查中引出了很多陌生的命令,比如vgremove等,於是打算重新瞭解這部分。 LVM是什麼? 邏輯捲管理器(LVM,Logical Volume Manager)是一種把硬碟空間分配成邏輯捲的方法。 看到定義可 ...
  • Windows server 2016 搭建DNS伺服器 環境說明: 1、Windows server 2016標準版 實操步驟: 1、添加DNS伺服器功能 1.1、點擊win圖標打開菜單,點擊打開伺服器管理器。 1.2、點擊“ 管理 ”,點擊“ 添加角色和功能 ” 下一步 基於角色或功能的安裝,下 ...
  • Taier 介紹 Taier 是袋鼠雲開源項目之一,是一個分散式可視化的DAG任務調度系統。 旨在降低ETL開發成本、提高大數據平臺穩定性,大數據開發人員可以在 Taier 直接進行業務邏輯的開發,而不用關心任務錯綜複雜的依賴關係與底層的大數據平臺的架構實現,將工作的重心更多地聚焦在業務之中。 項目 ...
  • 0.前言 MySQL由於開源的原因,有各式各樣的中件間Proxy ,極大的豐富了做高可用或遷移的方案,習慣了MySQL生態圈的靈活和便利,Oracle官方不開源代碼和協議,沒有中間件proxy,顯得很笨重。 比如以下的方案就會很不好辦: 實時抓取Oralce的訪問SQL日誌 慢日誌捕獲和收集 高可用 ...
  • 有相當一部分 iPhone 用戶會拒絕iOS更新最新系統,不管是因為各種BUG還是因為其他優化方面的問題,他們都會選擇一個自己覺得均衡的系統版本,安逸養老。 但是蘋果 iOS 系統如果你不及時更新推送版本的話,就會在手機桌面「設置」上方出現角標數字紅點,系統設置中也會出現紅點提示。強迫症患者表示簡直 ...
一周排行
    -Advertisement-
    Play Games
  • 概述:在C#中,++i和i++都是自增運算符,其中++i先增加值再返回,而i++先返回值再增加。應用場景根據需求選擇,首碼適合先增後用,尾碼適合先用後增。詳細示例提供清晰的代碼演示這兩者的操作時機和實際應用。 在C#中,++i 和 i++ 都是自增運算符,但它們在操作上有細微的差異,主要體現在操作的 ...
  • 上次發佈了:Taurus.MVC 性能壓力測試(ap 壓測 和 linux 下wrk 壓測):.NET Core 版本,今天計劃準備壓測一下 .NET 版本,來測試並記錄一下 Taurus.MVC 框架在 .NET 版本的性能,以便後續持續優化改進。 為了方便對比,本文章的電腦環境和測試思路,儘量和... ...
  • .NET WebAPI作為一種構建RESTful服務的強大工具,為開發者提供了便捷的方式來定義、處理HTTP請求並返迴響應。在設計API介面時,正確地接收和解析客戶端發送的數據至關重要。.NET WebAPI提供了一系列特性,如[FromRoute]、[FromQuery]和[FromBody],用 ...
  • 原因:我之所以想做這個項目,是因為在之前查找關於C#/WPF相關資料時,我發現講解圖像濾鏡的資源非常稀缺。此外,我註意到許多現有的開源庫主要基於CPU進行圖像渲染。這種方式在處理大量圖像時,會導致CPU的渲染負擔過重。因此,我將在下文中介紹如何通過GPU渲染來有效實現圖像的各種濾鏡效果。 生成的效果 ...
  • 引言 上一章我們介紹了在xUnit單元測試中用xUnit.DependencyInject來使用依賴註入,上一章我們的Sample.Repository倉儲層有一個批量註入的介面沒有做單元測試,今天用這個示例來演示一下如何用Bogus創建模擬數據 ,和 EFCore 的種子數據生成 Bogus 的優 ...
  • 一、前言 在自己的項目中,涉及到實時心率曲線的繪製,項目上的曲線繪製,一般很難找到能直接用的第三方庫,而且有些還是定製化的功能,所以還是自己繪製比較方便。很多人一聽到自己畫就害怕,感覺很難,今天就分享一個完整的實時心率數據繪製心率曲線圖的例子;之前的博客也分享給DrawingVisual繪製曲線的方 ...
  • 如果你在自定義的 Main 方法中直接使用 App 類並啟動應用程式,但發現 App.xaml 中定義的資源沒有被正確載入,那麼問題可能在於如何正確配置 App.xaml 與你的 App 類的交互。 確保 App.xaml 文件中的 x:Class 屬性正確指向你的 App 類。這樣,當你創建 Ap ...
  • 一:背景 1. 講故事 上個月有個朋友在微信上找到我,說他們的軟體在客戶那邊隔幾天就要崩潰一次,一直都沒有找到原因,讓我幫忙看下怎麼回事,確實工控類的軟體環境複雜難搞,朋友手上有一個崩潰的dump,剛好丟給我來分析一下。 二:WinDbg分析 1. 程式為什麼會崩潰 windbg 有一個厲害之處在於 ...
  • 前言 .NET生態中有許多依賴註入容器。在大多數情況下,微軟提供的內置容器在易用性和性能方面都非常優秀。外加ASP.NET Core預設使用內置容器,使用很方便。 但是筆者在使用中一直有一個頭疼的問題:服務工廠無法提供請求的服務類型相關的信息。這在一般情況下並沒有影響,但是內置容器支持註冊開放泛型服 ...
  • 一、前言 在項目開發過程中,DataGrid是經常使用到的一個數據展示控制項,而通常表格的最後一列是作為操作列存在,比如會有編輯、刪除等功能按鈕。但WPF的原始DataGrid中,預設只支持固定左側列,這跟大家習慣性操作列放最後不符,今天就來介紹一種簡單的方式實現固定右側列。(這裡的實現方式參考的大佬 ...