代码编写完毕在编译的过程中计算机的内存中会开辟一个空间来存储代码,这个空间就相当于是进程,可以将进程类比于工厂的厂房,但代码相当于原材料,但仅有厂房和原材料无法生产,还需要工人进行加工,工人则类比于线程
- 线程(厂房):程序运行的环境
- 线程(工人):线程
程序的运行分为同步和异步两种;
同步指的是事情按照顺序执行;例如早上起床,我们先刷牙,再洗脸,最后吃早餐,讲究的是事情先后的执行顺序
通常情况下我们的程序时按照同步执行的,即按照顺序一条一条执行,一条执行完毕执行下一条,如下所示:
/*
同步
- 通常情况下代码都是自上而下的
*/
console.log("哈哈")
console.log("嘻嘻")
console.log("嘿嘿")
通过快捷键F5通过NodeJs在控制台中打印输出内容为以上顺序执行的结果
哈哈
嘻嘻
嘿嘿
下面的代码示例就是非常典型的同步结构的代码:
function sum(a, b){
return a + b
}
console.log("第一行打印")
let result = sum(123, 234)
console.log(result)
console.log("第二行打印")
同步的特点是逻辑清晰、结构简单、容易理解;
同步:
- 通常情况下代码都是自上而下一行一行执行的
- 前边的代码不执行后边的代码也不会执行
- 同步的代码执行会出现阻塞的情况
- 一行代码执行慢会影响到整个代码的执行
解决同步的问题:
- java python
- 通过多线程来解决
- node.js
- 通过异步的方式来解决
同步最大的问题就是阻塞;
所谓的阻塞就是按顺序执行的代码,前面不执行完毕后面的代码不会执行;
function sum(a, b){
let begin = Date.now()
while(Date.now() - begin < 10000){
}
return a + b
}
console.log("第一行打印")
let result = sum(123, 234)
console.log(result)
console.log("第二行打印")
上述sum()执行的时候会停顿10秒,10秒后才会返回结果,由于是同步执行代码,所以sum()会阻碍后面所有代码的执行,导致整个程序的执行速度变慢
- 一段代码的执行不会影响到其他程序
- 异步的问题:
- 异步的代码无法通过return来设置返回值
- 特点:
1. 不会阻塞其他代码的执行
2. 需要通过回调函数来返回结果
- 基于回调函数的异步带来的问题
1. 代码的可读性差
2. 可调试性差
- 解决问题:
- 需要一个东西,可以代替回调函数来给我们返回结果
- Promise是一个可以用来存储数据的对象
Promise存储数据的方式比较特殊;
这种特殊的方式使得Promise可以用来存储异步调用的数据
在程序中,有些代码的执行速度很快,比如说:打印一个内容、接受一个请求、发送一个响应这些都是简单且能快速反应的操作,但例如:读写硬盘中的文件(I/O)操作这些就会非常的慢,如果在这些非常耗时的操作影响到能快速反应的操作,这是我们不希望看到的
对于其他语言,例如java,处理这类问题的方式简单粗暴,就是直接开启多线程,每一个线程处理一个事务,避免相互干扰;
但是对于Node.js来说,它本身就是单线程的,那么如何处理这种情况,答案就是采用异步;当我们去读取一个比较大的文件时,node先将指令发给计算机,然后再由计算机去读取文件,此时node的线程不是等待计算机数据返回,而是继续向下执行其他的操作,那么何时去获取计算机读取到的数据呢?等数据返回了我们再去读取,这样既不影响其他操作也可以正常的读取到计算机上返回的数据;
function sum(a, b){
setTimeout(()=>{
return a + b
}, 10000)
}
console.log("第一行打印")
let result = sum(123, 234)
console.log(result)
console.log("第二行打印")
上述代码中,我们将计算操作放到了setTimeout中,同样是等待10s,但是setTimeout不会阻塞其他代码的执行,而是在10s之后将函数放到了任务队列,这样一来就可以很好的解决阻塞问题。
但由于函数的返回值return到了setTimeout的回调函数中,此时我们再调用sum() 就无法获取到函数的计算结果了,因此后面调用sum()传入实参的时候计算结果为undefined;
那么如何获取异步代码的执行结果呢,只有通过回调函数了,异步代码通常都需要一个回调函数作为参数,当异步代码执行完毕取得结果时候便可以将结果作为回调函数的参数进行传递,这样我们便可以在回调函数中来读取结果,并完成后续操作
// 回调函数:将函数作为参数传递
function sum(a, b, cb){
setTimeout(()=>{
cb(a + b)
}, 10000)
}
console.log("第一行打印")
sum(123, 234, result => {
console.log(result)
})
console.log("第二行打印")
异步通过回调函数解决运算结果的传递问题,但最大的问题也来自于回调函数,由于是异步执行,回调函数无法直接通过返回值来返回执行结果,想要取得结果必须通过回调函数,
这样就带来一个问题:如果我们有两个异步操作需要先后执行,一个异步操作依赖于上一个异步操作的结果,那么我们只能采取嵌套措施了
// 回调函数:将函数作为参数传递
function sum(a, b, cb){
setTimeout(()=>{
cb(a + b)
}, 10000)
}
console.log("11111")
sum(123, 234, result => {
// 计算777与前两个数的加和
sum(result, 7, result => {
// 拿结果再进行求和操作
sum(result, 8, result => {
sum(result, 9, result => {
})
})
})
})
上述的示例中,调用了4次sum,每一次都调用了之前的计算结果,后一个是依赖前运算结果的;
这样子就是“回调地狱",又名死亡金字塔,这还只是4次,现实的代码可能比这更加复杂;
总之,异步提高了代码运行的效率,同样也增加了代码的复杂程度;