同步和异步

同步和异步

所谓同步,是指一个进程必需等待另一个进程的结果。

所谓异步,是指不需要等待其他进程,继续做自己的事件

静态资源的加载没有异步同步一说(script 例外),浏览器默认都是并行加载静态资源的

<script>标签是同步加载的

加载外部脚本时,浏览器会暂停页面渲染,等待脚本下载并执行完成后,再继续渲染。原因是JavaScript可以修改DOM(比如使用document.write方法),所以必须把控制权让给它。

如果外部脚本加载时间很长(比如一直无法完成下载),就会造成网页长时间失去响应,浏览器就会呈现“假死”状态,这被称为“阻塞效应”。

为了避免这种情况,较好的做法是将<script>标签都放在页面底部,而不是头部

defer属性

浏览器开始解析HTML网页
解析过程中,发现带有defer属性的script标签
浏览器继续往下解析HTML网页,同时并行下载script标签中的外部脚本
浏览器完成解析HTML网页,此时再执行下载的脚本

async属性

async属性的作用是,使用另一个进程下载脚本,下载时不会阻塞渲染。(执行是同步的)

浏览器开始解析HTML网页
解析过程中,发现带有async属性的script标签
浏览器继续往下解析HTML网页,同时并行下载script标签中的外部脚本

**脚本下载完成,浏览器暂停解析HTML网页,开始执行下载的脚本**

脚本执行完毕,浏览器恢复解析HTML网页

async属性可以保证脚本下载的同时,浏览器继续渲染。需要注意的是,一旦采用这个属性,就无法保证脚本的执行顺序。哪个脚本先下载结束,就先执行那个脚本。如果同时使用async和defer属性,后者不起作用,浏览器行为由async属性决定。

nodejs的异步非阻塞和python的同步

Node.js一直都标榜自己是non-blocking I/O,不标榜自己是Asynchronous I/O。
所以异步I/O并不是说Node.js的特色,非阻塞才是。
nodejs是单线程异步

单线程指的是主线程是“单线程”的,所有阻塞的部分交给一个线程池处理,然后这个主线程通过一个队列跟线程池协作,于是对我们写到的js代码部分,不再关心线程问题,代码也主要由一堆callback回调构成,然后主线程在不停的循环过程中,适时调用这些代码。

python可以用多线程实现异步

匿名函数和闭包的误区

匿名函数就是闭包吗,匿名函数和闭包有关系吗,

不是,没有什么关系!

匿名函数?

通常大家所说的匿名函数,准备的说法应该是 IIFE: Immediately Invoked Function Expression
意为立即调用的函数表达式,也就是说,声明函数的同时立即调用这个函数。

实际上,立即执行的匿名函数并不是函数,因为已经执行过了,所以它是一个结果,这个结果是对当前这个匿名函数执行结果的一个引用(函数执行默认return undefined)。

这个结果可以是一个字符串、数字或者null/false/true,也可以是对象、数组或者一个函数(对象和数组都可以包含函数),当返回的结果包含函数时,这个立即执行的匿名函数所返回的结果就是典型的闭包了。

闭包

A closure is the combination of a function and the lexical environment within which that function was declared.

中文解释是:闭包是一个函数和该函数被定义时的词法环境的组合。

按照闭包起的作用来理解它:就是能在一个函数外部执行这个函数内部的定义方法,并访问内部的变量

 function f1(){
    var n=999;
    function f2(){
      alert(n); 
    }
    return f2;
  }
  var result=f1();
  result(); // 999

立即执行函数的应用

for循环的问题及解决方案

还有一个令人感到困惑,工作和学习中也经常遇见的问题是在for循环中:

for(var i = 0;i<5;i++){
  setTimeout(function(){
    console.log(i);
  },100*i);
}

我们希望打印出来0,1,2,3,4,然而打印出来的是5个5。

什么原因引起的这问题呢?这是因为setTimeout的回调函数并不是立即执行的而是要等到循环结束才开始计时和执行(在此对运行机制不伸展)

要说明的一点是js中函数在执行前都只对变量保持引用,并不会真正获取和保存变量的值。所以等循环结束后i的值是已经是5了,因此执行定时器的回调函数会打印出5个5。

怎么解决这个问题?

最常见的解决方法就是给定时器外面加一个立即执行的匿名函数,并把当前循环的i作为实参传入这个立即执行的匿名函数。如下:

for(var i = 0;i<5;i++){
  (function(i){
    setTimeout(function(){
      console.log(i);
    },100*i);
  })(i);
}

可以得到预想的结果:0,1,2,3,4

此时很多人认为这个立即执行的匿名函数就是闭包,其实这么理解是错误的,在错误的理解之上又扩展了好多案例

https://stackoverflow.com/questions/8967214/what-is-the-difference-between-a-closure-and-an-anonymous-function-in-js。

到底这个for循环中的闭包是什么呢,其中的自执行匿名函数又起到什么作用呢?
我们可以试着把这个自执行的匿名函数改写为具名的函数,来测试下结果:

for(var i = 0;i<5;i++){
  function hasNameFn(i){
    setTimeout(function(){
      console.log(i);
    },100*i);
  };
  hasNameFn(i);
}

可以发现结果和使用匿名函数的结果是一样的

另外,使用es6的let或者bind绑定当前的i都可以达到一样的结果

所以说匿名函数和闭包之间并没有什么关系,只不过很多时候在用到匿名函数解决问题的时候恰好形成了一个闭包,就导致很多人分不清楚匿名函数和闭包的关系

websocket技术

WebSocket 是 HTML5 开始提供的一种在单个 TCP 连接上进行全双工通讯的协议。

本质上是一个基于 TCP 的协议。

HTTP 协议是一种无状态的、无连接的、单向的应用层协议。它采用了请求/响应模型。通信请求只能由客户端发起,服务端对请求做出应答处理。

这种通信模型有一个弊端:HTTP 协议无法实现服务器主动向客户端发起消息。

这种单向的请求,如果服务器有连续的状态变化,客户端要获知就非常麻烦。大多数 Web 应用程序将通过频繁的异步JavaScript和XML(AJAX)请求实现长轮询。轮询的效率低,非常浪费资源(因为必须不停连接,或者 HTTP 连接始终打开)。

相对传统http的优势

  • WebSocket是HTML5开始提供的一种在单个 TCP 连接上进行全双工通讯的协议。
  • 在WebSocket API中,浏览器和服务器只需要做一个握手的动作,然后,浏览器和服务器之间就形成了一条快速通道。两者之间就直接可以数据互相传送。
  • 浏览器通过 JavaScript 向服务器发出建立 WebSocket 连接的请求,连接建立以后,客户端和服务器端就可以通过 TCP 连接直接交换数据。
  • 获取 Web Socket 连接后,通过 send() 方法来向服务器发送数据,并通过 onmessage 事件来接收服务器返回的数据。

创建WebSocket对象。

var Socket = new WebSocket(url, [protocol] );

属性

  • Socket.readyState 只读属性 readyState 表示连接状态,可以是以下值:

      • 表示连接尚未建立。
      • 表示连接已建立,可以进行通信。
      • 表示连接正在进行关闭。
      • 表示连接已经关闭或者连接不能打开。
  • Socket.bufferedAmount 只读属性 bufferedAmount 已被 send() 放入正在队列中等待传输,但是还没有发出的 UTF-8 文本字节数。

事件

  • open 连接建立时触发
  • message 客户端接收服务端数据时触发
  • error 通信发生错误时触发
  • close 连接关闭时触发

方法

  • Socket.send() 使用连接发送数据
  • Socket.close() 关闭连接

示例

// 初始化一个 WebSocket 对象
var ws = new WebSocket("ws://localhost:9998/echo");

// 建立 web socket 连接成功触发事件
ws.onopen = function () {
  // 使用 send() 方法发送数据
  ws.send("发送数据");
  alert("数据发送中...");
};

// 接收服务端数据时触发事件
ws.onmessage = function (evt) {
  var received_msg = evt.data;
  alert("数据已接收...");
};

// 断开 web socket 连接成功触发事件
ws.onclose = function () {
  alert("连接已关闭...");
};

socket.io 是基于 WebSocket 的 C-S 实时通信库