admin管理员组

文章数量:1345316

I have a client-side JS app that uses IndexedDB to store its state. Works fine. However, it's kind of slow because I am frequently reading from and writing to IndexedDB so that the state does not bee inconsistent when multiple tabs are open.

My idea was... put all DB access stuff inside a Service Worker, and then I can cache values there in memory without worrying that another tab might have altered the database.

That seems to work fine, except some parts of my application take a long time to run. I can municate the status (like "X% done") from the Service Worker to my UI. But both Firefox and Chrome seem to kill the worker if it runs for more than 30 seconds, which is way too short for me.

Is there any way around this limitation? If not, any ideas for achieving something similar? A Shared Worker could do it I think, except browser support is poor and I don't anticipate that improving now with the momentum behind Service Workers.

I have a client-side JS app that uses IndexedDB to store its state. Works fine. However, it's kind of slow because I am frequently reading from and writing to IndexedDB so that the state does not bee inconsistent when multiple tabs are open.

My idea was... put all DB access stuff inside a Service Worker, and then I can cache values there in memory without worrying that another tab might have altered the database.

That seems to work fine, except some parts of my application take a long time to run. I can municate the status (like "X% done") from the Service Worker to my UI. But both Firefox and Chrome seem to kill the worker if it runs for more than 30 seconds, which is way too short for me.

Is there any way around this limitation? If not, any ideas for achieving something similar? A Shared Worker could do it I think, except browser support is poor and I don't anticipate that improving now with the momentum behind Service Workers.

Share Improve this question asked Dec 29, 2016 at 3:19 dumbmatterdumbmatter 9,7037 gold badges45 silver badges87 bronze badges 11
  • Are you trying to update all tabs when change of a value occurs? – guest271314 Commented Dec 29, 2016 at 3:30
  • Sometimes yes, sometimes no. – dumbmatter Commented Dec 29, 2016 at 3:41
  • Have you tried using localStorage, storage event? – guest271314 Commented Dec 29, 2016 at 3:44
  • In this case, I need the features and performance of IndexedDB. – dumbmatter Commented Dec 29, 2016 at 3:49
  • SharedWorker is not supported by ie though supported by other borwsers developer.mozilla/en-US/docs/Web/API/SharedWorker. Would SharedWorker performance be different from ServiceWorker? – guest271314 Commented Dec 29, 2016 at 3:52
 |  Show 6 more ments

3 Answers 3

Reset to default 5 +25

The Google documentation on service workers tells us that using service workers as a memory cache is impossible:

It's terminated when not in use, and restarted when it's next needed, so you cannot rely on global state within a service worker's onfetch and onmessage handlers. If there is information that you need to persist and reuse across restarts, service workers do have access to the IndexedDB API.

My suggestion is to keep using service workers to persist data to the database, and use localStorage to create a shared cache between pages. The tab that is making the change is then responsible for both updating the cache in localStorage and persisting to IndexedDB through the service worker.

I ended up using a Shared Worker. In browsers that don't support Shared Workers, such as Edge and Safari, I fall back to a Web Worker and some hacky code to only let you open the app in one tab at a time.

Since 90% of my users are on Firefox or Chrome anyway, I figure it's not a huge loss.

I also wrote a library to (amongst other things) normalize the APIs used in Shared Workers and Web Workers, so I have the exact same code running in both scenarios, the only difference is whether the worker is initialized with Worker or SharedWorker.

As you said yourself SharedWorkers seems to be exactly what you need. I'm not sure why you think the momentum behind implementing ServiceWorkers prevent browsers from supporting SharedWorkers. They seem to be two different kind of beasts.

As far as I understand ServiceWorkers should be used as a proxy for your requests when your application is offline and heavy stuff should be done in WebWorkers and SharedWorkers

https://developer.mozilla/en-US/docs/Web/API/Service_Worker_API https://developer.mozilla/en-US/docs/Web/API/SharedWorker

本文标签: javascriptLongrunning process inside a Service Worker (or something similar)Stack Overflow