解决 ES9 新增的 SharedArrayBuffer 在浏览器端的性能问题

阅读时长 6 分钟读完

ES9 中新增的 SharedArrayBuffer,可以实现多个 JavaScript 线程在共享一个内存空间上进行操作,从而提高了代码的并发性和性能。然而,SharedArrayBuffer 也带来了一些安全漏洞,基于此,最新版本的 Chrome 和 Safari 浏览器已经禁止了其在默认环境下的使用,需要手动开启才可以使用。另外,SharedArrayBuffer 在某些特定情况下也会带来一些性能问题,在此,本文将探讨如何解决 SharedArrayBuffer 在浏览器端的性能问题。

SharedArrayBuffer 的使用

在使用 SharedArrayBuffer 之前,需要先了解一下 Web Worker。

Web Worker 是 HTML5 的一个新特性,它可以将 JavaScript 代码在主线程之外的线程中运行,与主线程并行处理任务。另外,Web Worker 与主线程之间采用的是异步通信方式,因此不会影响主线程的运行。但是,Web Worker 之间的通信又是非常困难的,因为各个线程之间并没有共享内存空间。

SharedArrayBuffer 就是为了解决这个问题而诞生的。它通过共享内存空间,在多个 JavaScript 线程之间进行交互。具体来说,SharedArrayBuffer 可以被多个线程访问,每个线程都可以在其中获取和修改数据,而不必通过异步通信的方式进行协调。

SharedArrayBuffer 是一个 ArrayBuffer 实例对象的 “兄弟类”,两者都可以共享内存。然而,SharedArrayBuffer 是被设计成可以被多个线程之间共享的。

SharedArrayBuffer 相比 ArrayBuffer 新增了以下两个主要特性:

  1. Atomics:在 SharedArrayBuffer 上提供了一组原子操作的 API,可以实现对共享内存空间的原子读写,从而防止出现竞争条件和数据不一致的问题。

  2. Lock-free:多个线程可以无锁的并发操作同一份数据,避免了传统的互斥锁机制带来的不良效果。

SharedArrayBuffer 的使用方法与 ArrayBuffer 类似,具体可以参考以下代码:

-- -------------------- ---- -------
-- -- -----------------
----- --- - --- ------------------------

-- -- ----------------- --------
----- ---- - --- ---------------- -- ---- -- ---------
----- -- - --- ---------------- -- ---- - ------------

-- -----------
------- - -----
----- - ----

-- -----------
--------------------- -- -- ----
------------------- -- -- ---

SharedArrayBuffer 的性能问题

尽管 SharedArrayBuffer 可以提高 JavaScript 代码的并发性和性能,但是在某些情况下也会带来性能问题。具体来说,浏览器会为每一个 JavaScript 线程分配一个单独的线程,如果线程数过多,就会使浏览器的性能受到影响。

以 Chrome 浏览器为例,由于其采用的是基于线程的渲染机制,每个标签页中最多能同时执行 6 个 JavaScript 线程(最近版本更新,已经变成了 8 个线程)。

因此,在使用 SharedArrayBuffer 时,需要注意避免线程数过多的问题。常见的解决方法如下:

  1. 共享内存空间的数据节点数量控制在较小的数量。

  2. 提高程序运行效率,尽可能的利用单线程的执行能力。

  3. 采用类似线程池模式的算法设计,避免线程的创建和销毁的过度浪费。

SharedArrayBuffer 性能问题的解决方法

除了上述的实现方式之外,为了更好的解决 SharedArrayBuffer 在浏览器端的性能问题,可以采用以下的优化方案:

  1. 使用 Atomics 的 API:Atomics 提供了一组原子操作的 API,可以实现对共享内存空间的原子读写,从而避免了竞争条件和数据不一致的问题。例如:
-- -------------------- ---- -------
-- -------
----- ------ - --- ------------------------
-- - -- -----------
----- --- - --- ---------------------

-- -------
------------------ -- -----

-- ------
-------- ---------------- -------- -
  ---------------- ------ ---------
-

-- ----
--- ---- - - -- - - --- ---- -
  ------------- -- -
    ------------ -----
  ---
-
  1. 优化线程的数量和工作负载:通过合理的控制线程数和工作负载,减少线程的创建和销毁的过度浪费。例如:
-- -------------------- ---- -------
-- -------------------
-------- -------------- -
  --- ------ - --
  --- ---- - - -- - - --------- ---- -
    ------ -- ----------------
  -
  ------ -------
-

-- ------
-------- ---------------------- ------ ---- -
  --- ---- - - ------ - - ---- ---- -
    -------- - ------------------
  -
-

-- ----------
----- ------------ - --
----- --- - -------

-- -----
----- ------ - --- --------------------- - ---
----- ----- - --- -------------------
--- ---- - - -- - - ---- ---- -
  -------- - ------------------------ - --------
-

-- -----------------
----- --------- - --- - -------------
----- -------- - ---
--- ---- - - -- - - ------------- ---- -
  ----- ----- - - - ----------
  ----- --- - ----- - ----------
  --------------
    --- ----------------- -- -
      ------------- -- -
        ---------------------- ------ -----
        ----------
      ---
    --
  --
-

----------------------------- -- -
  -- ----
  --- ----- - --
  --- ---- - - -- - - ---- ---- -
    ----- -- ---------
  -
  -------------------
---

总结

SharedArrayBuffer 的出现可以提高 JavaScript 代码的并发性和性能,但是也有一些潜在的安全和性能问题。通过合理的使用和优化,可以避免 SharedArrayBuffer 在浏览器端的性能问题,提升 JavaScript 代码的性能与应用体验。

以上就是本文分享的内容,希望对读者有所帮助。

来源:JavaScript中文网 ,转载请注明来源 https://www.javascriptcn.com/post/6590d49aeb4cecbf2d619dfa

纠错
反馈