ARTICLE

Introduction to HTML5 Web Worker

Posted by Dipal Choksi Articles | HTML 5 July 20, 2011
In this article we have a look at HTML5 Web Workers which enable building of responsive web applications.
Reader Level:

Introduction

Web Workers are a part of the HTML5 specification - they enable 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 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 has 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 scenario 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 time stamp.

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

 

/* myWorker.js */
//only for demo purpose
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();
}
 
/* The onmessage event listener responds to the event raised from the main script, invokes the appropriate function based on the data passed.
*/
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>
 
/* Check if Web Workers are supported */
function getWebWorkerSupport() {
return (typeof(Worker) !== "undefined") ? true:false;
}
 
 
if(getWebWorkerSupport() == true)
{
var userName,y,message;
 
/* Create the new workers. each instance runs in parallel */
slowWorker = new Worker("myWorker.js");
quickWorker = new Worker("myWorker.js");
 
/* Bind an event listener for the onmessage function for each of the workers. this will be invoked by the worker thread when the execution has completed.
*/
slowWorker.onmessage = function (event) {
document.getElementById("slowResult").value = event.data;
};
 
quickWorker.onmessage = function (event) {
document.getElementById("quickResult").value = event.data;
};
/* Register events for buttons */
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>
</html>

Sample in Action - open testww.html in a compliant browser (the following results are on Firefox 5)

webworker1.jpg 

Conclusion


Web Workers utilize a thread-like message passing mechanism to achieve parallelism. They keep the UI performant and responsive for users. No more "A script on this page is busy or has stopped responding..." errors!!

Happy Coding!

Login to add your contents and source code to this article
post comment
     

Isn't it such an amazing feature? I haven't yet used web workers in PROD apps. There is a good deal of information on the concepts but not much info on benchmarks. Here's an article with some advanced real-like samples: "10 web workers": http://www.whatwg.org/specs/web-apps/current-work/complete/workers.html

Posted by Dipal Choksi Jul 24, 2011

Thanks Dipal. I didn't know HTML can can multiple threads. Nice :) I can really use this feature in some of our new features like processing some background data. How's the performance? Wonder if you or anybody here have used in real applications?

Posted by Mahesh Chand Jul 21, 2011
COMMENT USING
PREMIUM SPONSORS
DynamicPDF™ product line allows you to dynamically generate PDF documents, merge PDF documents and add new content to existing PDF documents from within your applications.
SPONSORED BY
  • PDF reports have never been easier to create. With our included WYSIWYG Designer, you can layout your reports, set up your data source and let DynamicPDF ReportWriter do the rest.
Get Career Advice from Experts