html5---webworker多线程

时间:2023-03-09 18:02:29
html5---webworker多线程

javascript本身就是一个单线程的语言,一开始它的出现是为了简单的网页设计,设计者并没有考虑到多线程的问题,要知道,线程的开销是非常昂贵的。但是随着web开发的潮流化,javascript不是仅仅一门网页脚本语言那么简单了,它可以涉及到socket(websocket),canvas等复杂的操作,这时候单线程的计算远远不够,在此时html5引入了webworker(多线程)对象,用于实现js的多线程操作,虽然本身js还是门单线程的语言,实现多线程是基于Event loop的实现,这里介绍更加详细http://blog.jobbole.com/30445/

1.webworker分为Worker(专用线程) 和 SharedWorker(共享线程)两种,都必须遵守同源策略

同源策略:

为了能和服务器交互,worker必须遵守同源策略(same-origin policy)(译注:可参考国人文章同源策略)。比如,位于http://www.example.com/内的脚本文件不能访问https://www.example.com的脚本。尽管域名相同,但同源策略要求端口也必须一致。通常,这不会成为一个很大的问题。但是你很有可能会同一个域名编写worker和client,所以知道这点对你总是有所帮助。

(1)Worker(专用线程)一般用于交换信息,而不是共享信息,而SharedWorker则用于共享信息

(2)因为worker一般都是在后台运行的,所以不建议在worker中访问DOM,并进行一系列的DOM操作

2.在这里为了粗略介绍,我们只讲Worker(专用线程),有兴趣了解ShareWorker的可以在这里参考

我们将会做一个加载图片的demo,其中有三个文件:test.html,Fthread.js(父线程),Sthread.js(子线程),test.png位于web服务器网站的根目录下

test.html

<!DOCTYPE html>
<html>
<body>
</body>
<script src="Fthread.js"></script>
</html>

Fthread.js

var worker = new Worker("Sthread.js"); 创建一个子进程
worker.addEventListener("message", function(e) {
e = window.event || e; //加入一些js hack
var imgSrc = e.data; //收到子进程传递过来的数据
//创建一个图片便签并在页面显示图片
var img = document.createElement("img");
img.src = imgSrc;
document.body.appendChild(img);
/*由父进程关闭终止子进程*/
//this.terminate();
}); //很重要,不可少,这个将会触发向子进程的请求
worker.postMessage("begin");

Sthread.js

//当收到父进程传递过来的消息时,子进程响应,发回图片url
onmessage = function(e){
e = window.event || e; if (e.data === "begin") { postMessage("test.png");
} }
this.close();//关闭子进程
}

  

需要注意的是,chrome不支持本地测试,最好使用其他浏览器测试。

3.除次之外,我们还可以进行线程嵌套,即子线程再创建一个子进程,只需要在Sthread.js作些修改

//当收到父进程传递过来的消息时,子进程响应,发回图片url
onmessage = function(e){
e = window.event || e; if (e.data === "begin") { var worker = new Worker("SSThread.js");//再创建子进程
worker.onmessage = function(e1) {
e1 = window.event || e1;
postMessage(e1.data); //返回图片url到父进程
}
worker.postMessage("get url");
}
this.close();//关闭子进程
}

SSthread.js

//当收到父进程传递过来的消息时,子进程响应,发回图片url
onmessage = function(e){
e = window.event || e; if (e.data === "get url") { postMessage("test.png"); //返回图片url到父进程
} }
this.close();//关闭子进程
}

关于线程嵌套的内容暂时讲到这里了,小弟也是初学,还有很多需要尝试和学习。若有什么出错的地方,欢迎指出