Introduction
Web Workers are a part of the HTML5 specification - they enable the creation of background JavaScript threads for CPU intensive tasks, without blocking the UI or any scripts that handle user interaction.
This acts as a multi-threading feature and provides parallel execution.
Support Detection
The first step is to check if the browser supports the Web Worker API. If the browser support is present, there will be a Worker property on the global window object. If your browser doesn't support the Web Worker API, the Worker property will be undefined. Note that the Worker object has the W in upper case!
Parallel Job
The next step is to create the "Worker" script that is going to perform the CPU intensive tasks we want to run in a parallel thread. This script is saved in a separate .js file.
Manage the flow
The communication is initiated by passing relevant information in a message from the main script to the worker script using the postMessage method. The worker script responds to the "onmessage" event by fetching the required information from the event and performing the requested task. When the task has completed, the worker posts a message back to the main thread. This message can be a simple variable or a JSON object. The main thread reacts in a similar fashion by handling the onmessage event which was raised by the worker and using the output result as needed. During the time that the worker script was executing, user interaction and other tasks are not blocked and the user is able to interact with the application.
You also have the capability to terminate a worker from the main thread, if needed.
Do note that web workers only have access to a subset of JavaScript's features to support the concurrent execution.
Error handling
Of course, we never have any errors in our code ;-) But we do have to make provisions. If an error occurs while a worker is executing, the ErrorEvent is fired.
Sample
We will try a simple sample now, to take a look at the reality of the implementation. This article is admittedly contrived to provide a working scenario. You can extend the concepts to real-world scenarios involving CPU intensive operations.
In the sample, we have a slow process - the code implements a "do nothing" loop (which is purely to demonstrate the concept in this article) and after a delay, returns the user input and the time that the function completes execution.
We have a fast process - the code just returns the user input and the execution timestamp.
When I tried to run the sample, I entered a string in the userName "slow". Clicked on the slow button which led the delay loop to execute. I could, in the meantime, interact with the application, update the string in the input text box and click the second button. The second button also spawned a worker which returned almost instantly, displaying the updated input and the time.
At no point did the application block, pending for any of the threads to return.
Take a look at the code now:
Code: Worker JavaScript - named as myWorker.js
-
-
- function ReallySlowCode(millis)
- {
- var date = new Date();
- var curDate =
- null;
- do {
- curDate = new Date();
- }
- while (curDate - date < millis);
- }
- function slowfunc(userName) {
- ReallySlowCode(5000);
- return userName +
- "-" + new Date();
- }
- function quickfunc(userName) {
- return
- userName + "-" + new Date();
- }
-
-
-
-
- this.onmessage = function(event) {
- var data = event.data;
- switch (data.op) {
- case 'slow':
- postMessage(slowfunc(data.userName));
- break;
- case 'quick':
- postMessage(quickfunc(data.userName));
- break;
- default:
- postMessage("Wrong
- operation specified ");
- }
- };
Main: named as testww.html
- <!DOCTYPE html>
- <body>
- <table border="0">
- <tr>
- <td>Your
- Name</td>
- <td>
- <input type="text" id="userName"
- />
- </td>
- </tr>
- <tr>
- <td>Slow
- Process</td>
- <td>
- <input type="text" id="slowResult"
- size="100"/>
- </td>
- </tr>
- <tr>
- <td>Quick
- Process</td>
- <td>
- <input type="text" id="quickResult"
- size="100"/>
- </td>
- </tr>
- <tr>
- <td
- colwidth="2">
- <input type="button" id="slowButton" value="Run Long
- process" />
- <input type="button" id="quickButton" value="Run quick
- process" />
- </td>
- </tr>
- </table>
- <script>
-
-
- function getWebWorkerSupport() {
- return (typeof(Worker) !==
- "undefined") ? true:false;
- }
- if(getWebWorkerSupport()
- == true)
- {
- var userName,y,message;
-
-
- slowWorker = new
- Worker("myWorker.js");
- quickWorker = new
- Worker("myWorker.js");
-
-
-
-
- slowWorker.onmessage = function (event)
- {
- document.getElementById("slowResult").value =
- event.data;
- };
- quickWorker.onmessage = function (event)
- {
- document.getElementById("quickResult").value = event.data;
- };
-
-
- document.getElementById("slowButton").onclick = function() {
- userName = document.getElementById("userName").value;
- message = {
- 'op'
- : 'slow',
- 'userName' :
- userName
- };
- slowWorker.postMessage(message);
- }
- document.getElementById("quickButton").onclick
- = function() {
- userName = document.getElementById("userName").value;
- message = {
- 'op'
- : 'quick',
- 'userName' :
- userName
- };
- quickWorker.postMessage(message);
- }
- }
- </script>
- </body>undefined</html>
Sample in Action - open testww.html in a compliant browser (the following results are on Firefox 5)
Conclusion
Web Workers utilize a thread-like message passing mechanism to achieve parallelism. They keep the UI performant and responsive to users. No more "A script on this page is busy or has stopped responding..." errors!!
Happy Coding!