重新认识async、await

以下是一些事件循环机制例子

0.两个例子先测下实力

async function async1 () {
    await new Promise((resolve, reject) => { // 右边是Promise无需等待
        resolve()
    })
    console.log('A')
}async1()new Promise((resolve) => {
    console.log('B')
    resolve()
}).then(() => {
    console.log('C')
}).then(() => {
    console.log('D')
})
// 最终结果👉: B A C D

解析:调用async1,有await等待一个promise没有返回值,跳出,console.log(‘A’)不执行。打印B,有.then把C放入微任务。现在执行await后面的打印A。之后在取出微任务的打印C,最后是D。结果就是B A C D

async function async1 () {
    await async2()
    console.log('A')
}async function async2 () {
    return new Promise((resolve, reject) => { // 从async视角等待2个then
        resolve()
    })
}async1()new Promise((resolve) => {
    console.log('B')
    resolve()
}).then(() => {
    console.log('C')
}).then(() => {
    console.log('D')
})// 最终结果👉: B C D A

解析:执行async1, async1中await了async2, async2返回了Promise但是也是没有明确的值,跳出,console.log(‘A’)不执行。来到下面打印B,之后微任务放入C,(解释一下这个为什么没有打印A,跟上面哪里不一样?)取出微任务中的,打印C,之后打印D,最后打印A。⚠️这里与上面输出不同是因为,第二个await后面跟的是一个async函数,是async函数就要根据它返回值的类型进行不同数目的微任务第二个例子中async2返回了一个Promise,根据结论,返回值为一个Promise的需要等待 2个then的时间,那么等待then©、then(D)两个后才会打印A。

1.async 函数返回值

在讨论 await 之前,先聊一下async 函数处理返回值的问题,它会像 Promise.prototype.then 一样,会对返回值的类型进行辨识。

👉根据返回值的类型,引起 js引擎 对返回值处理方式的不同

📑结论:async函数在抛出返回值时,会根据返回值类型开启不同数目的微任务
return结果值:非thenable、非promise(不等待)
return结果值:thenable(等待 1个then的时间)
return结果值:promise(等待 2个then的时间)

例1:

async function testA () {
    return 1; // 非thenable、非promise
}testA().then(() => console.log(1));
Promise.resolve()
    .then(() => console.log(2))
    .then(() => console.log(3));// (不等待)最终结果👉: 1 2 3

例子2:

async function testB () {
    return {
        then (cb) { // thenable
            cb();
        }
    };
}testB().then(() => console.log(1));
Promise.resolve()
    .then(() => console.log(2))
    .then(() => console.log(3));// (等待一个then)最终结果👉: 2 1 3

例子3:

async function testC () {
    return new Promise((resolve, reject) => { // promise
        resolve()
    })
}

testC().then(() => console.log(1));
Promise.resolve()
    .then(() => console.log(2))
    .then(() => console.log(3));
    
// (等待两个then)最终结果👉: 2 3 1




async function testC () {
    return new Promise((resolve, reject) => { // promise
        resolve()
    })
} 

testC().then(() => console.log(1));
Promise.resolve()
    .then(() => console.log(2))
    .then(() => console.log(3))
    .then(() => console.log(4))

// (等待两个then)最终结果👉: 2 3 1 4

看了这三个🌰是不是对上面的结论有了更深的认识
稍安勿躁,来试试一个经典面试题👇

async function async1 () {
    console.log('1')
    await async2()
    console.log('AAA')
}async function async2 () {
    console.log('3')
    return new Promise((resolve, reject) => {
        resolve()
        console.log('4')
    })
}
​
console.log('5')setTimeout(() => {
    console.log('6')
}, 0);async1()new Promise((resolve) => {
    console.log('7')
    resolve()
}).then(() => {
    console.log('8')
}).then(() => {
    console.log('9')
}).then(() => {
    console.log('10')
})
console.log('11')// 最终结果👉: 5 1 3 4 7 11 8 9 AAA 10 6

如果根据上面的结论来,这题应该没问题。
5 1 3 4 ,这里的promise需要等待2个then的时间。7也没问题。遇见.then8放进微任务, 打印11。从微任务队列取出打印8,9放进微任务,取出打印9,10放进微任务。至此等待的2个then已经完成。回到async1中将await async2()后面的打印AAA。取出打印10,最后是宏任务setTimeout中的6。

步骤拆分👇:
0.先执行同步代码,输出5
1.执行setTimeout,是放入宏任务异步队列中
2.接着执行async1函数,输出1
3.执行async2函数,输出3
4.Promise构造器中代码属于同步代码,输出4

async2函数的返回值是Promise,等待2个then后放行,所以AAA暂时无法输出

5.async1函数暂时结束,继续往下走,输出7
6.同步代码,输出11
7.执行第一个then,输出8
8.执行第二个then,输出9
9.终于等到了两个then执行完毕,执行async1函数里面剩下的,输出AAA
10.再执行最后一个微任务then,输出10
11.执行最后的宏任务setTimeout,输出6

2. await 右值类型区别

2.1、非 thenable(会立即向微任务队列添加一个微任务then,但不需等待)

🌰1:

async function test () {
    console.log(1);
    await 1;
    console.log(2);
}test();
console.log(3);
// 最终结果👉: 1 3 2

🌰2:

function func () {
    console.log(2);
}async function test () {
    console.log(1);
    await func();
    console.log(3);
}test();
console.log(4);// 最终结果👉: 1 2 4 3

🌰3:

async function test () {
    console.log(1);
    await 123 // 非 thenable,无需等待
    console.log(2);
}test();
console.log(3);
​
Promise.resolve()
    .then(() => console.log(4))
    .then(() => console.log(5))
    .then(() => console.log(6))
    .then(() => console.log(7));// 最终结果👉: 1 3 2 4 5 6 7

Note:
await后面接非 thenable 类型,会立即向微任务队列添加一个微任务then,但不需等待

2.2、thenable类型(等待一个then 的时间之后执行)

async function test () {
    console.log(1);
    await {
        then (cb) { // thenable需等待一个then时间
            cb();
        },
    };
    console.log(2);
}test();
console.log(3);
​
Promise.resolve()
    .then(() => console.log(4))
    .then(() => console.log(5))
    .then(() => console.log(6))
    .then(() => console.log(7));// 最终结果👉: 1 3 4 2 5 6 7

Note:
await 后面接 thenable 类型,需要等待一个 then 的时间之后执行

2.3、Promise类型(同非thenable无需等待)

async function test () {
    console.log(1);
    await new Promise((resolve, reject) => { // 无需等待,但正常是2个then时间
        resolve()
    })
    console.log(2);
}test();
console.log(3);
​
Promise.resolve()
    .then(() => console.log(4))
    .then(() => console.log(5))
    .then(() => console.log(6))
    .then(() => console.log(7));// 最终结果👉: 1 3 2 4 5 6 7

❓为什么表现的和非 thenable 值一样呢?为什么不等待两个 then 的时间呢?

Note:

  • TC 39(ECMAScript标准制定者) 对await 后面是 promise
    的情况如何处理进行了一次修改,移除了额外的两个微任务,在早期版本,依然会等待两个 then 的时间
  • 掘金上有大佬翻译了官方解释:更快的async 函数和 promises, 但在这次更新中并没有修改 thenable 的情况

这样做可以极大的优化 await 等待的速度👇

async function func () {
    console.log(1);
    await 1; // 非 thenable,无需等待1个then
    console.log(2);
    await 2;
    console.log(3);
    await 3;
    console.log(4);
}async function test () {
    console.log(5);
    await func();
    console.log(6);
}test();
console.log(7);
​
Promise.resolve()
    .then(() => console.log(8))
    .then(() => console.log(9))
    .then(() => console.log(10))
    .then(() => console.log(11));// 最终结果👉: 5 1 7 2 8 3 9 4 10 6 11

前面5 1 7没问题,正常要从微任务对列中拿出8,但是await func();中await 1; 非 thenable,无需等待1个then,所以先打印出了2。之后走的打印8,放入9。本来要打印9的,但是又到了await 2; 非 thenable,无需等待1个then,所以先打印出了3。之后取出打印9。以此类推,到await func();全部执行完。打印6,最后是11。

Note:
await Promise.prototype.then 虽然很多时候可以在时间顺序上能等效,但是它们之间有本质的区别

1.test 函数中的 await 会等待 func 函数中所有的 await 取得 恢复函数执行 的命令并且整个函数执行完毕后才能获得取得 恢复函数执行的命令;
2.test 函数的 await 此时不能在时间的顺序上等效 then,而要等待到 func 函数完全执行完毕;
3.比如这里的数字6很晚才输出,如果单纯看成then的话,在下一个微任务队列执行时6就应该作为同步代码输出了才对。

所以我们可以合并两个函数的代码👇

async function test () {
    console.log(5);
	// 原来​func()中的
    console.log(1);
    await 1;
    console.log(2);
    await 2;
    console.log(3);
    await 3;
    console.log(4);
    await null;
    
    console.log(6);
}test();
console.log(7);
​
Promise.resolve()
    .then(() => console.log(8))
    .then(() => console.log(9))
    .then(() => console.log(10))
    .then(() => console.log(11));// 最终结果👉: 5 1 7 2 8 3 9 4 10 6 11

因为将原本的函数融合,此时的 await 可以等效为Promise.prototype.then,又完全可以等效如下代码👇

async function test () {
    console.log(5);
    console.log(1);
    Promise.resolve()
        .then(() => console.log(2))
        .then(() => console.log(3))
        .then(() => console.log(4))
        .then(() => console.log(6))
}test();
console.log(7);
​
Promise.resolve()
    .then(() => console.log(8))
    .then(() => console.log(9))
    .then(() => console.log(10))
    .then(() => console.log(11));// 最终结果👉: 5 1 7 2 8 3 9 4 10 6 11

以上三种写法在时间的顺序上完全等效,所以你 完全可以将 await后面的代码可以看做在 then 里面执行的结果,又因为 async 函数会返回 promise 实例,所以还可以等效成👇

async function test () {
    console.log(5);
    console.log(1);
}test()
    .then(() => console.log(2))
    .then(() => console.log(3))
    .then(() => console.log(4))
    .then(() => console.log(6))
​
console.log(7);
​
Promise.resolve()
    .then(() => console.log(8))
    .then(() => console.log(9))
    .then(() => console.log(10))
    .then(() => console.log(11));// 最终结果👉: 5 1 7 2 8 3 9 4 10 6 11

可以发现,test 函数全是走的同步代码…

所以👉:async/await 是用同步的方式,执行异步操作

3.例子

例1:

async function async2 () {
    new Promise((resolve, reject) => { // async返回值是非thenable,无需等待
        resolve()
    })
}async function async3 () {
    return new Promise((resolve, reject) => { // async返回值是promise需要等待2个then
        resolve()
    })
}async function async1 () {
    // 方式一:最终结果:B A C D
    // await new Promise((resolve, reject) => { // 同非thenable无需等待,最新规定已经移除了等待2个then
    //     resolve()
    // })// 方式二:最终结果:B A C D
    // await async2()// 方式三:最终结果:B C D A
    await async3()
​
    console.log('A')
}async1()new Promise((resolve) => {
    console.log('B')
    resolve()
}).then(() => {
    console.log('C')
}).then(() => {
    console.log('D')
})

大致思路👇:
首先,async函数的整体返回值永远都是Promise,无论值本身是什么

  • 方式一:await的是Promise,无需等待
  • 方式二:await的是async函数,但是该函数的返回值本身是非thenable,无需等待
  • 方式三:await的是async函数,且返回值本身是Promise,需等待两个then时间

例2: 又搞不懂了?

function func () {
    console.log(2);// 方式一:1 2 4  5 3 6 7
    // Promise.resolve() //拿到确切的func函数返回值undefined
    //     .then(() => console.log(5))
    //     .then(() => console.log(6))
    //     .then(() => console.log(7))// 方式二:1 2 4  5 6 7 3
    return Promise.resolve() // 拿到func函数返回值,但是并未获得具体的结果
        .then(() => console.log(5))
        .then(() => console.log(6))
        .then(() => console.log(7))
}async function test () {
    console.log(1);
    await func(); // 方式一 拿到确切值undefined等待了一个then
    console.log(3);
}test();
console.log(4); 

步骤拆分👇:
方式一:
同步代码输出12,接着将log(5)处的then1加入微任务队列,await拿到确切的func函数返回值undefined

将后续代码console.log(3);放入微任务队列(then2,可以这样理解)

执行同步代码输出4,到此,所有同步代码完毕
执行第一个放入的微任务then1输出5,产生log(6)的微任务then3

执行第二个放入的微任务then2输出3

然后执行微任务then3,输出6,产生log(7)的微任务then4
执行then4,输出7

方式二:
同步代码输出12await拿到func函数返回值,但是并未获得具体的结果(由Promise本身机制决定),暂停执行当前async函数内的代码(跳出、让行)
输出4,到此,所有同步代码完毕
await一直等到Promise.resolve().then...执行完成,再放行输出3

方式二没太明白❓

function func () {
    console.log(2);return Promise.resolve()
        .then(() => console.log(5))
        .then(() => console.log(6))
        .then(() => console.log(7))
}async function test () {
    console.log(1);
    await func() // 没拿到确切值,await一直等到Promise.resolve().then...执行完成
    console.log(3);
}test();
console.log(4);new Promise((resolve) => {
    console.log('B')
    resolve()
}).then(() => {
    console.log('C')
}).then(() => {
    console.log('D')
})// 最终结果👉: 1 2 4    B 5 C 6 D 7 3

还是没懂?

继续👇

async function test () {
    console.log(1);
    await Promise.resolve() // 怎么算有确切值呢?
        .then(() => console.log(5))
        .then(() => console.log(6))
        .then(() => console.log(7))
    console.log(3);
}test();
console.log(4);new Promise((resolve) => {
    console.log('B')
    resolve()
}).then(() => {
    console.log('C')
}).then(() => {
    console.log('D')
})// 最终结果👉: 1 4    B 5 C 6 D 7 3

这里也是懵
Note:
综上,await一定要等到右侧的表达式有确切的值才会放行,否则将一直等待(阻塞当前async函数内的后续代码),不服看看这个👇

function func () {
    return new Promise((resolve) => {
        console.log('B')
        // resolve() 故意一直保持pending
    })
}async function test () {
    console.log(1);
    await func()
    console.log(3);
}test();
console.log(4);
// 最终结果👉: 1 B 4 (永远不会打印3)
​
​
// ---------------------或者写为👇-------------------
async function test () {
    console.log(1);
    await new Promise((resolve) => {
        console.log('B')
        // resolve() 故意一直保持pending
    })
    console.log(3);
}test();
console.log(4);
// 最终结果👉: 1 B 4 (永远不会打印3)

例3:

async function func () {
    console.log(2);
    return { // 等待一个then
        then (cb) { 
            cb()
        }
    }
}async function test () {
    console.log(1);
    await func();
    console.log(3);
}test();
console.log(4);new Promise((resolve) => {
    console.log('B')
    resolve()
}).then(() => {
    console.log('C')
}).then(() => {
    console.log('D')
})// 最终结果👉: 1 2 4 B C 3 D

步骤拆分👇:
同步代码输出1、2
await拿到func函数的具体返回值thenable,将当前async函数内的后续代码放入微任务then1(但是需要等待一个then时间)
同步代码输出4、B,产生log(C)的微任务then2
由于then1滞后一个then时间,直接执行then2输出C,产生log(D)的微任务then3
执行原本滞后一个then时间的微任务then1,输出3
执行最后一个微任务then3输出D

其他例子:
setTimeout+Promise+Async输出顺序?很简单呀!

4.总结

4.1 async函数返回值

📑结论:async函数在抛出返回值时,会根据返回值类型开启不同数目的微任务
return结果值:非thenable、非promise(不等待)
return结果值:thenable(等待 1个then的时间)
return结果值:promise(等待 2个then的时间)

4.2 await右值类型区别

  • 接非 thenable 类型,会立即向微任务队列添加一个微任务then,但不需等待
  • thenable 类型,需要等待一个 then 的时间之后执行
  • Promise类型(有确定的返回值),会立即向微任务队列添加一个微任务then,但不需等待
    • TC 39 对await 后面是 promise 的情况如何处理进行了一次修改,移除了额外的两个微任务,在早期版本,依然会等待两个 then 的时间

作者:Squirrel_
链接:https://juejin.cn/post/7194744938276323384
来源:稀土掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/918406.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

.NET桌面应用架构Demo与实战|WPF+MVVM+EFCore+IOC+DI+Code First+AutoMapper

目录 .NET桌面应用架构Demo与实战|WPFMVVMEFCoreIOCDICode FirstAutoPapper技术栈简述项目地址:功能展示项目结构项目引用1. 新建模型2. Data层,依赖EF Core,实现数据库增删改查3. Bussiness层,实现具体的业务逻辑4. Service层&am…

c++学习第三天

创作过程中难免有不足&#xff0c;若您发现本文内容有误&#xff0c;恳请不吝赐教。 提示&#xff1a;以下是本篇文章正文内容&#xff0c;下面案例可供参考 一、内联函数 1.铺垫 #include<iostream>//实现一个ADD的宏函数 //#define ADD(x, y) xy //错解1 //#defin…

html 图片转svg 并使用svg路径来裁剪html元素

1.png转svg 工具地址: Vectorizer – 免费图像矢量化 打开svg图片,复制其中的path中的d标签的路径 查看生成的svg路径是否正确 在线SVG路径预览工具 - UU在线工具 2.在html中使用svg路径 <svg xmlns"http://www.w3.org/2000/svg" width"318px" height…

Solana应用开发常见技术栈

编程语言 Rust Rust是Solana开发中非常重要的编程语言。它具有高性能、内存安全的特点。在Solana智能合约开发中&#xff0c;Rust可以用于编写高效的合约代码。例如&#xff0c;Rust的所有权系统可以帮助开发者避免常见的内存错误&#xff0c;如悬空指针和数据竞争。通过合理利…

23种设计模式-访问者(Visitor)设计模式

文章目录 一.什么是访问者模式&#xff1f;二.访问者模式的结构三.访问者模式的应用场景四.访问者模式的优缺点五.访问者模式的C实现六.访问者模式的JAVA实现七.代码解释八.总结 类图&#xff1a; 访问者设计模式类图 一.什么是访问者模式&#xff1f; 访问者模式&#xff08;…

RecyclerView详解——(四)缓存复用机制

稍微看了下源码和部分文章&#xff0c;在此做个小小的总结 RecyclerView&#xff0c;意思为可回收的view&#xff0c;那么相对于listview&#xff0c;他的缓存复用肯定是一大优化。 具体而言&#xff0c;当一个列表项被移出屏幕后&#xff0c;RecyclerView并不会销毁其视图&a…

C++设计模式行为模式———迭代器模式

文章目录 一、引言二、迭代器模式三、总结 一、引言 迭代器模式是一种行为设计模式&#xff0c; 让你能在不暴露集合底层表现形式 &#xff08;列表、 栈和树等&#xff09; 的情况下遍历集合中所有的元素。C标准库中内置了很多容器并提供了合适的迭代器&#xff0c;尽管我们不…

【网络云计算】2024第48周-技能大赛-初赛篇

文章目录 1、比赛前提2、比赛题目2.1、 修改CentOS Stream系统的主机名称&#xff0c;写出至少3种方式&#xff0c;并截图带时间戳和姓名&#xff0c;精确到秒&#xff0c;否则零分2.2、 创建一个名为你的名字的拼音的缩写的新用户并设置密码&#xff0c;将用户名添加到 develo…

【汇编语言】数据处理的两个基本问题(三) —— 汇编语言的艺术:从div,dd,dup到结构化数据的访问

文章目录 前言1. div指令1.1 使用div时的注意事项1.2 使用格式1.3 多种内存单元表示方法进行举例1.4 问题一1.5 问题一的分析与求解1.5.1 分析1.5.2 程序实现 1.6 问题二1.7 问题二的分析与求解1.7.1 分析1.7.2 程序实现 2. 伪指令 dd2.1 什么是dd&#xff1f;2.2 问题三2.3 问…

R语言数据分析案例45-全国汽车销售数据分析(可视化与回归分析)

一、研究背景 随着经济的发展和人们生活水平的提高&#xff0c;汽车已经成为人们日常生活中不可或缺的交通工具之一。汽车市场的规模不断扩大&#xff0c;同时竞争也日益激烈。对于汽车制造商和经销商来说&#xff0c;深入了解汽车销售数据背后的规律和影响因素&#xff0c;对…

【算法】【优选算法】前缀和(下)

目录 一、560.和为K的⼦数组1.1 前缀和1.2 暴力枚举 二、974.和可被K整除的⼦数组2.1 前缀和2.2 暴力枚举 三、525.连续数组3.1 前缀和3.2 暴力枚举 四、1314.矩阵区域和4.1 前缀和4.2 暴力枚举 一、560.和为K的⼦数组 题目链接&#xff1a;560.和为K的⼦数组 题目描述&#x…

论文 | Learning to Transfer Prompts for Text Generation

1. 总结与提问 论文摘要总结&#xff1a; 论文提出了一种创新的PTG&#xff08;Prompt Transfer Generation&#xff09;方法&#xff0c;旨在通过迁移提示的方式解决传统预训练语言模型&#xff08;PLM&#xff09;在数据稀缺情况下微调的问题。通过将一组已在源任务中训练好…

TON商城与Telegram App:生态融合与去中心化未来的精彩碰撞

随着区块链技术的快速发展&#xff0c;去中心化应用&#xff08;DApp&#xff09;逐渐成为了数字生态的重要组成部分。而Telegram作为全球领先的即时通讯应用&#xff0c;不仅仅满足于传统的社交功能&#xff0c;更在区块链领域大胆探索&#xff0c;推出了基于其去中心化网络的…

自动驾驶系列—探索自动驾驶数据管理的核心技术与平台

&#x1f31f;&#x1f31f; 欢迎来到我的技术小筑&#xff0c;一个专为技术探索者打造的交流空间。在这里&#xff0c;我们不仅分享代码的智慧&#xff0c;还探讨技术的深度与广度。无论您是资深开发者还是技术新手&#xff0c;这里都有一片属于您的天空。让我们在知识的海洋中…

【技术解析】Dolphinscheduler实现MapReduce任务的高效管理

MapReduce是一种编程模型&#xff0c;用于处理和生成大数据集&#xff0c;主要用于大规模数据集&#xff08;TB级数据规模&#xff09;的并行运算。本文详细介绍了Dolphinscheduler在MapReduce任务中的应用&#xff0c;包括GenericOptionsParser与args的区别、hadoop jar命令参…

数据结构哈希表-(开放地址法+二次探测法解决哈希冲突)(创建+删除+插入)+(C语言代码)

#include<stdio.h> #include<stdlib.h> #include<stdbool.h> #define M 20 #define NULLDEL -1 #define DELDEY -2typedef struct {int key;int count; }HashTable;//创建和插入 void Insert(HashTable ha[], int m, int p, int key) {int i, HO, HI;HO key…

【android USB 串口通信助手】stm32 源码demo 单片机与手机通信 Android studio 20241118

android 【OTG线】 接 下位机STM32【USB】 通过百度网盘分享的文件&#xff1a;USBToSerialPort.apk 链接&#xff1a;https://pan.baidu.com/s/122McdmBDUxEtYiEKFunFUg?pwd8888 提取码&#xff1a;8888 android 【OTG线】 接 【USB转TTL】 接 【串口(下位机 SMT32等)】 需…

大数据技术Kafka详解 ① | 消息队列(Messages Queue)

目录 1、消息队列的介绍 2、消息队列的应用场景 2.1、应用耦合 2.2、异步处理 2.3、限流削峰 2.4、消息驱动的系统 3、消息队列的两种模式 3.1、点对点模式 3.2、发布/订阅模式 4、常用的消息队列介绍 4.1、RabbitMQ 4.2、ActiveMQ 4.3、RocketMQ 4.4、Kafka 4.…

一家餐饮企业,「闯入」AI阵地

作者| 皮爷 出品|产业家 “我们需要用AI来帮助我们门店破除内卷的状态。”一位连锁餐饮品牌告诉产业家&#xff0c;“这也是我们想尽快把AI用起来的原因&#xff0c;看看能不能带来一些帮助。” 这种情况正发生在一众餐饮企业中。 与这种情况对应的一个背景是&#xff0c…

MySQL的编程语言

一、MySQL基础 使用系统的全局变量@@VERSION查看当前使用的MySQL的版本信息,SQL语句如下: select @@version; 将局部变量varl声明为char的类型,长度值为10,并为其赋值为“程菲” begin declare var1 char(10); set @var1="程菲"; end 通过局部变量查看d_eams数…