Postman接口自动化之postman脚本编写

news2024/11/15 20:01:20

这是之前搞的接口自动化方案,已经在业务测试中实现了使用postman编写接口脚本,通过GitHub+Jenkins+email +html report实现了接口自动化,现在分块整理一下。

postman脚本编写

1、创建集合 和 目录:

一条业务线下的接口可以放到一个集合里,例如,xxxOneAPI;按照接口的业务分类,创建文件夹,例如,user、dev、event、home、settings等;一个接口的不同场景或者用例可以创建子文件夹,这样就形成了postman接口用例的层级目录:

 2、请求URL:

指定请求方式后就可以编辑请求的url了,url的格式一般为:

https://{{host}}/v1/user/register?uuid={{$guid}}&t={{$timestamp}}

(1)因为在测试过程中会有不同的环境,例如,测试环境、准生产环境、生产环境等,或者不同的大区,例如,中国区、北美区、欧洲区等,这样把域名作为一个变量,例如,host,使用双{}来引用;

(2)query部分包含uuid和时间戳,uuid是一个请求的唯一标识,以及该接口在处理过程中请求其他相关的接口都会是相同的uuid,查日志时会用到它,时间戳是发生请求的时间。postman提供了生成uuid和时间戳的方法,引用方式为:{{$guid}}和{{$timestamp}}

(3)跟URL设置相关的是Params,例如:

  3、编辑Headers:

根据接口文档添加指定的header,例如:

 如果,多个接口使用的header是相同的,可以使用Presets->Manage Presets添加公共header 

 

 

这样在编辑后面的接口脚本时就可以直接选择已经编辑好的公共接口了。

注意:header中一定要指定Content-Type,例如,application/x-www-form-urlencoded

4、编辑Body:

(1)指定传输格式,例如,x-www-form-urlencoded

(2)设置环境变量,并通过{{}}引用

5、编写Test:

(1)添加断言,断言有不用的方式,例如:

//断言状态code是200

pm.test("status code 是200",function(){

    pm.response.to.have.status(200);

});

//断言整个返回的json串,适用于当返回的json内容较少时

pm.test("Json串返回502 用户已存在", function () {

    pm.response.to.have.body({"errno":502,"errmsg":"用户已存在","data":{"countryAbbr":"CN","countryCode":"86"}});

});

//断言json串中的Email是登录的Email

pm.test("json串中返回的Email是登录的Email", function () {

    var jsonData = pm.response.json();

    pm.expect(jsonData.data.email).to.eql(pm.environment.get("username"));

});

//断言返回的Json串中包含字段:email、sid_、xxxxUid_、 xxxxUsername_、uid_字段

pm.test("Json串中返回的字段包含:email、sid_、xxxxUid_、 xxxxUsername_、uid_", function () {

    pm.expect(pm.response.text()).to.include("email");

    pm.expect(pm.response.text()).to.include("sid_");

    pm.expect(pm.response.text()).to.include("tuyaUid_");

    pm.expect(pm.response.text()).to.include("tuyaUsername_");

    pm.expect(pm.response.text()).to.include("uid_");

});

pm.test("上传成功,已生成url!", function () {

    pm.expect(pm.response.text()).to.include("url", "https://d4fp4ynbqtcz4d.cloudfront.net/xxxxx/feedback");

});

(2)添加postman控制台的日志:

console.log("注册的email字段是:" +pm.environment.get("unregisteredEmail"));

(3)提取返回值:

    //提取uid_、sid_、xxxxUid_、xxxxUsername_

    pm.environment.set("uid_", jsonData.data.uid_);

    pm.environment.set("sid_", jsonData.data.sid_);

    pm.environment.set("xxxxUid_", jsonData.data.xxxxUid_);

    pm.environment.set("xxxxUsername_", jsonData.data.xxxxUsername_);

});

(4)添加 if...else if...判断,用于场景兼容:

if(pm.response.to.have.body({"errno":510,"errmsg":"新密码与原密码一样","data":{}})){

//断言

    pm.test("Json返回510 新密码与原密码一样", function () {

        pm.response.to.have.body({"errno":510,"errmsg":"新密码与原密码一样","data":{}});

})

}else if(pm.response.to.have.body({"errno":112,"errmsg":"重复操作过于频繁,请休息","data":{}})){

//断言

    pm.test("Json返回112 重复操作过于频繁,请休息", function () {

        pm.response.to.have.body({"errno":112,"errmsg":"重复操作过于频繁,请休息","data":{}});

})

}

(5)添加逻辑_1:

var resData = JSON.parse(responseBody);

tests["判断 errorno 是0"] = resData.errno === 0

var xxxxxflag = 0   //有mm设备标志

var listEnd = resData.data.list.length   //设备列表长度,即设备数量

if(listEnd == 0){

    pm.environment.set("devSn", "1000");   //该用户未绑定任何设备

    console.log("该用户未绑定任何设备,devSn设置为1000")

}

else{

    for(var i = 0;i<listEnd;i++){   //遍历设备列表role=0的设备可以编辑设备名称,role=1的设备设置按钮置灰

        if(resData.data.list[i].pid == "xxxxx" && resData.data.list[i].devType == 1 && resData.data.list[i].role == 0){

            xxxxxflag = 1;   //有mm设备并且role=0,标识设置为1

            pm.environment.set("devSn", resData.data.list[i].sn);   //获取mm产品的设备序列号,作为修改设备名称接口的入参

            pm.environment.set("devName", resData.data.list[i].name);   //获取mm产品的设备名字,作为修改设备名称接口的入参

            tests["设备的pid是xxxxx"] = resData.data.list[i].pid === "xxxxx"   //有mm设备再去断言,该值对于mm设备固定

            tests["设备的devType是1"] = resData.data.list[i].devType === 1   //该值对于mm设备固定

            tests["设备的role是0"] = resData.data.list[i].role === 0   //自由设备

            break;

        }

    }

    if(xxxxxflag == 0){

        pm.environment.set("devSn", "2000");   //该用户没有绑定mm设备的时候为该值,修改设备名称是用作判断条件

        console.log("该用户没有绑定mm设备或者没有自主设备,不能修改设备名称,devSn设置为2000")

    }

}

console.log("设备的sn_是:" +pm.environment.get("devSn"));

(6)添加逻辑_2:

//三种事件基本类型(RD提供):

//TYPE_MM_EVENT_STAY = "event_stay" // 【事件】DL250

//TYPE_MM_EVENT_PUSH = "event_push" // 【事件】按mm

//TYPE_MM_EVENT_DAS = "event_das" // 【事件】QQ

pm.test("校验返回的列表数据如下:", function () {

    var jsonData = pm.response.json();

    if(jsonData.data.list.length == 0){   //没有事件返回的情况

        tests["没有该时间段的das事件,事件列表返回为空"] = jsonData.errmsg == "成功";

    }else{   //有事件的情况

        var rightFlagBase = 0;

        var rightFlagType = 0;

        

        for(var i = 0;i<jsonData.data.list.length;i++){   //有事件情况,每个事件都应该符合要求,pid是xxxxx、devTyep是1、sn是指定的设备

            if(jsonData.data.list[i].pid=="xxxxx"&&jsonData.data.list[i].devType==1&&jsonData.data.list[i].sn==pm.environment.get("devSn")){

                rightFlagBase = 1;

                for(var j=0;j<jsonData.data.list[i].uniformType.length;j++){

                    if(["das","person","emergency"].includes(jsonData.data.list[i].uniformType[j])&&jsonData.data.list[i].eventType=="event_das"){   //基本类型也要正确,如业务上有调整,变化此处。QQ的eventType是"event_das"

                        rightFlagType = 1;

                        continue

                    }else{

                        rightFlagType = 0;

                        break;

                    }

                }

                if(rightFlagType == 1){

                    continue;

                }else{

                    tests["第"+(i+1)+"条数据不符合要求,统一类型错误:"+jsonData.data.list[i].uniformType] = jsonData.data.errmsg === "失败";

                    console.log("统一类型是:"+jsonData.data.list[i].uniformType+",不在指定列表当中");

                    break;

                }

                

            }else{

                rightFlagBase = 0;

                tests["第"+(i+1)+"条数据不符合要求,断言失败"] = jsonData.data.errmsg === "失败";

                console.log("第"+(i+1)+"条数据不符合要求,断言失败,信息如下:");

                console.log("pid是:"+jsonData.data.list[i].pid);

                console.log("devType是:"+jsonData.data.list[i].devType);

                console.log("sn是:"+jsonData.data.list[i].sn);

                break;

            }

        }

        

        if(rightFlagBase == 1 && rightFlagType == 1){   //完全信任正确标志,在遍历完事件列表后如果返回数据正确,该标志位都为1,有一条数据不符合要求,该标志位为0

            tests["pid是:xxxxx"] = jsonData.errmsg === "成功";

            tests["devType是:1"] = jsonData.errmsg === "成功";

            tests["sn是:" +pm.environment.get("devSn")] = jsonData.errmsg === "成功";

            tests["返回的统一类型在指定列表当中"+["das","person","emergency"]] = jsonData.errmsg ==="成功";

            tests["QQ对应的eventType是event_das"] = jsonData.errmsg ==="成功";

        }else{

            console.log("基本标志位是:"+rightFlagBase);

            console.log("类型标志位是:"+rightFlagType);

        }

    }

});

(7)添加逻辑_3:

pm.test("断言信息如下:", function () {

    var jsonData = pm.response.json();

    tests["返回结果是:成功"] = jsonData.errmsg ==="成功";

    if(jsonData.data.list.length != 0){

        var rightFlag = 0;

        //获取当前年、月,与返回的日期对比

        var date = new Date();

        var year = date.getFullYear();

        if((date.getMonth()+1) < 10){

            var month = "0"+(date.getMonth()+1);   //1-9月份需要修改月份格式,例如,07

        }else{

            var month = date.getMonth()+1   //10、11、12月不用修改格式

        }

        //console.log("year:"+year);

        //console.log("month:"+month);

        for(var i = 0;i<jsonData.data.list.length;i++){

            if(jsonData.data.list[i].date.substr(0,4) == year && jsonData.data.list[i].date.substr(5,2) == month && jsonData.data.list[i].count != 0){

                //console.log("截取的年份是:"+jsonData.data.list[i].date.substr(0,4));

                //console.log("截取的月份是:"+jsonData.data.list[i].date.substr(5,2));

                //console.log("数量是:"+jsonData.data.list[i].count)

                rightFlag = 1;

                continue;

            }else{

                rightFlag = 0;

                break;

            }

        }

        if(rightFlag == 1){

            tests["返回月份正确,并且该月事件数量不为0"] = jsonData.errmsg ==="成功";

        }

    }else{

        tests["返回list是空,前一个月没有事件"] = jsonData.errmsg ==="成功";

    }

    //pm.expect(jsonData.value).to.eql(100);

});

(8)添加逻辑_4:

//获取当前时间戳,并生成4小时又2分钟后的时间戳,作为设置值传给后端

var timestamp = new Date().getTime();   //当前时间戳

timestamp = timestamp + 4*60*60*1000 + 2*60*1000;   //4小时又2分钟后的时间戳

console.log("当前时间戳:"+timestamp)

pm.environment.set("timeStamp", timestamp);

//pm.globals.set("globalTemp", timestamp);

(9)定义方法、发送请求:

//校验返回json串

pm.test("返回json串正确", function () {

    pm.response.to.have.body({"errno":0,"errmsg":"成功","data":{}});

});

//调用 获取设置-根据key获取-notify 断言设置的 值 是否正确

function uuid() {

    var s = [];

    var hexDigits = "0123456789abcdef";

    for (var i = 0; i < 36; i++) {

        s[i] = hexDigits.substr(Math.floor(Math.random() * 0x10), 1);

    }

    s[14] = "4";  // bits 12-15 of the time_hi_and_version field to 0010

    s[19] = hexDigits.substr((s[19] & 0x3) | 0x8, 1);  // bits 6-7 of the clock_seq_hi_and_reserved to 01

    s[8] = s[13] = s[18] = s[23] = "-";

    var uuid = s.join("");

    return uuid;

}

var tempL = pm.environment.get("host");

var tempUuid = uuid();

console.log("tempGuid:"+tempUuid)

var tempP = pm.environment.get("pid");

const getRegStatus = {

    url:'https://'+tempL+'/v1/appSetting/getByKey?uuid='+tempUuid,

    method: 'POST',

    header: ['Content-Type:application/x-www-form-urlencoded',

             'zz-Pid:' +pm.environment.get("pid_"),

             'zz-Uid:' +pm.environment.get("uid_"),

             'zz-Sid:' +pm.environment.get("sid_"),

             ],

    body: { mode: 'urlencoded', urlencoded: 'sn='+pm.environment.get("devSn")+'&devType=1&key=radar'}

};

pm.sendRequest(getRegStatus, function (err, response) {   //发送请求

    console.log(response.json());

    var jsonData = response.json()

    pm.test("bearing值是111.0", function () {

        pm.expect(jsonData.data.list[0].value).to.eql("111.0");

    });

    pm.test("locate值是{\"longitude\":116.56605774760248,\"latitude\":39.97220103883733}", function () {

        pm.expect(jsonData.data.list[1].value).to.eql("{\"longitude\":116.56605774760248,\"latitude\":39.97220103883733}");

    });

    pm.test("distance值是914cm即30ft", function () {

        pm.expect(jsonData.data.list[2].value).to.eql("914");

    });

});

6、编写Pre-request Script

(1)调用方法:

//生成当前时间戳

var timeNow = new Date().getTime();

pm.environment.set("timeLt", timeNow);   //当前时间

console.log("当前时间戳是:"+timeNow);

pm.environment.set("timeGt", 0);   //当前日期的0点

//生成md5

var pwdTemp = pm.environment.get("pwd");

var md5Pwd = CryptoJS.MD5(pwdTemp).toString();

pm.environment.set('pwdAfterMd5', md5Pwd);

(2)定义方法,例如,生成邮箱,用于注册使用:

//生成随机邮箱前缀

function randomString(e) {    

    e = e || 32;

    var t = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789",

    a = t.length,

    n = "";

    for (i = 0; i < e; i++) n += t.charAt(Math.floor(Math.random() * a));

    return n

}

pm.environment.set("unregisteredEmail", "gz-test-" + randomString(10)+"@qq.com");

(3)添加逻辑判断:

var tempSn = pm.environment.get("devSn");

var tempNa = pm.environment.get("devName");

if(tempSn == "1000"){

    console.log("该用户未xx任何设备");

}

else if(tempSn == "2000"){

    console.log("该用户没有绑定mm设备或者没有自主设备,不能修改设备名称");

}

else{

    console.log("该用户绑定了mm,sn_是:"+tempSn+"name是:"+tempNa);

    var tempNaEdit = tempNa.reverse();   //对原有的设备名字进行反转,作为修改后的名字

    pm.environment.set("devName", tempNaEdit);   

}

(4)生成预置数据_1:

//生成当前时间戳

var timeNow = new Date().getTime();

pm.environment.set("timeLt", timeNow);   //当前时间

console.log("当前时间戳是:"+timeNow);

//获取当日0点时间戳

var startT = new Date();

startT.setHours(0);

startT.setMinutes(0);

startT.setSeconds(0);

startT.getTime();

console.log("startT is: "+startT.getTime());

pm.environment.set("timeGt", startT.getTime());

(5)生成预置数据_2:

var date = new Date();

//获取前月第一天的时间戳,例如,7月1日是:1625068800000

var start = new Date(date.getFullYear(),(date.getMonth()-1),date.getDate(),"00","00","00");

var startOfMonth = start.setDate(1);

console.log("前月的第一天是:"+startOfMonth);

pm.environment.set("sOm", startOfMonth);

//获取前月的天数,例如,7月份是31天

var countOfMonth = new Date(date.getFullYear(),date.getMonth(),0).getDate();

console.log("前月一共有"+countOfMonth+"天");

//获取前月最后一天的时间戳,例如,7月31日是:1627747199000

var end  = new Date(date.getFullYear(),(date.getMonth()-1),date.getDate(),"23","59","59");

var endOfMonth = end.setDate(countOfMonth);

console.log("前月最后一天是:"+endOfMonth);

pm.environment.set("eOm", endOfMonth);

(6)发送请求:

同Test模块

总结:接口主要是对数据进行处理(通过服务程序对数据操作,增删改查等),那么数据的准确性包括字段的准确性成为验证的重点。为了代替人眼对接口返回数据的校验,那么在接口自动化中就会使用代码对数据进行校验,这样就产生了Pre-request Script和Test模块的大量使用并需要编写适合场景的校验代码。

8、其他注意:

1、首先创建自己的postman账号!

2、创建团队,免费的是3个人。

3、辑脚本过程要及时的Save,避免写了大半天的脚本没保存。
 

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

ACL2023 | 大模型如何快速构建指令遵循数据集?self-instruct:用175条种子数据追上InstructGPT001效果

一、概述 title&#xff1a;SELF-INSTRUCT: Aligning Language Models with Self-Generated Instructions 论文地址&#xff1a;https://arxiv.org/abs/2212.10560 代码&#xff1a;GitHub - yizhongw/self-instruct: Aligning pretrained language models with instruction…

chatgpt赋能python:Python怎么输出Unicode值

Python怎么输出Unicode值 Python 是一种高级编程语言&#xff0c;因其简单易学和快速开发已成为许多开发者的首选。Python 可以输出多种数据类型&#xff0c;包括字符串和数字。 在许多情况下&#xff0c;输出 Unicode 值是必需的&#xff0c;本文将介绍在 Python 中如何输出 …

shardingsphere第二课-shardingsphere-jdbc的基本使用及各种分片策略

第一章介绍 一、ShardingJDBC客户端分库分表 ShardingSphere-JDBC 定位为轻量级 Java 框架&#xff0c;在 Java 的 JDBC 层提供的额外服务。 它使用客户端直连数据库&#xff0c;以 jar 包形式提供服务&#xff0c;无需额外部署和依赖&#xff0c;可理解为增强版的 JDBC 驱动…

chatgpt赋能python:Python中如何输入一个列表

Python中如何输入一个列表 输入一个列表是Python编程的基本任务之一。列表可以看做是一种序列&#xff0c;其中包含多个元素&#xff0c;用逗号隔开&#xff0c;并用方括号括起来。在Python中&#xff0c;列表是一种非常常见的数据类型&#xff0c;常用于存储和处理一系列相关…

断言操作符介绍

目录 1.延时操作符&#xff08;##&#xff09; 1.1 ##m 1.2 ##[m:n] 2.蕴含操作符&#xff08;|>,|->&#xff09; 2.1 |>操作符 2.2 |->操作符 3 重复操作符 ([*m][->m][m]) 3.1 连续重复操作符&#xff08;[*m][*m:n]&#xff09; 3.2 跟随重复操作…

Java选择题刷题记录5

Java堆栈 图片来自https://www.cnblogs.com/cici-new/p/14963762.html 数组、String都在堆里 枚举类 1.枚举类可以实现一个或多个接口&#xff0c;使用enum定义的枚举类默认继承java.lang.Enum类&#xff0c;而不是默认继承Object类&#xff0c;其中 java.lang.Enum类实现了…

机器视觉硬件的选择-标定板

康耐视智能相机Insight-缺陷检测 一>棋盘格的作用 a>畸变校正 径向畸变,径向畸变就是沿着透镜半径方向分布的畸变,产生原因是光线在原理透镜中心的地方比靠近中心的地方更加弯曲,这种畸变在短焦镜头中表现更加明显,径 向畸变主要包括桶形畸变和枕形畸变两种。以下分别…

chatgpt赋能python:Python输入π的方法及其应用

Python输入π的方法及其应用 Python是一门强大的编程语言&#xff0c;其支持的数学函数功能能够帮助用户完成各种复杂的计算操作。当我们需要在Python代码中使用π值时&#xff0c;不同的场景需要不同的处理方法。本文将详细介绍如何在Python中输入π值&#xff0c;并且探讨它…

Ansible 的脚本 --- playbook 剧本

playbook的相关知识 playbook 的简介 playbook是 一个不同于使用Ansible命令行执行方式的模式&#xff0c;其功能更强大灵活。 简单来说&#xff0c;playbook是一个非常简单的配置管理和多主机部署系统&#xff0c; 不同于任何已经存在的模式&#xff0c;可作为一个适合部署…

【C++】STL之string类(2)

个人主页&#xff1a;平行线也会相交&#x1f4aa; 欢迎 点赞&#x1f44d; 收藏✨ 留言✉ 加关注&#x1f493;本文由 平行线也会相交 原创 收录于专栏【C之路】&#x1f48c; 本专栏旨在记录C的学习路线&#xff0c;望对大家有所帮助&#x1f647;‍ 希望我们一起努力、成长&…

Vue中如何进行分布式任务调度与任务监控

Vue中如何进行分布式任务调度与任务监控 在复杂的系统中&#xff0c;如何有效地进行任务调度和监控是一个非常重要的问题。分布式系统中&#xff0c;任务调度和监控则更加复杂。Vue是一款流行的前端框架&#xff0c;本文将介绍如何在Vue中进行分布式任务调度和监控。 什么是分…

QML基础

从 Qt 4.7 开始&#xff0c;Qt 引入了一种声明式脚本语言&#xff0c;称为 QML&#xff08;Qt Meta Language 或者 Qt Modeling Language&#xff09;&#xff0c;作为 C 语言的一种替代。而 Qt Quick 就是使用 QML 构建的一套类库。 QML 是一种基于 JavaScript 的声明式语言。…

Qt下存储读写应用程序设置的三种方法

一、简介 List item 用户对应用程序经常有这样的要求&#xff1a;要求它能记住它的settings&#xff0c;比如窗口大小、位置和密码等等。有三种方法可以实现&#xff1a; 使用注册表&#xff1b;使用配置文件(.ini)&#xff1b;使用自定义文件(例如.txt)。 二、使用注册表 …

C盘空间不足清理方法 之 Google Chrome 浏览器用户数据迁移和Windows10 默认浏览器路径失效修复

原理分析 将原来C盘的目录拷贝到其他盘&#xff0c;然后用mklink建立一个联接&#xff0c;这里贴下ChatGPT对于三种链接的解释 # 在Windows 10中&#xff0c;mklink是一个命令行工具&#xff0c;用于创建符号链接&#xff08;symbolic link&#xff09;或者硬链接&#xff08…

chatgpt赋能python:Python异常过滤教程:如何正确处理和过滤Python中的异常

Python异常过滤教程&#xff1a;如何正确处理和过滤Python中的异常 介绍 Python是一种非常流行的编程语言&#xff0c;广泛应用于各种应用程序和领域中。在我们编写Python程序的过程中&#xff0c;通常要处理各种异常情况&#xff0c;比如用户输入错误&#xff0c;文件读取错…

CSS查缺补漏之《Web字体、2D/3D变换》

文章略长&#xff0c;慢慢享用&#xff5e; Web字体 css3新增了字体&#xff0c;使得用户不必局限在本计算机中安装的字体&#xff0c;可以使用多种字体&#xff1b; 需要在style中定义font-face规则&#xff1b; font-face { font-family: xxx名字; /* 必选项&#xff0c;自…

ROS2 入门应用 创建启动文件(Python)

ROS2 入门应用 创建启动文件&#xff08;Python&#xff09; 1. 创建功能包2. 添加依赖关系3. 添加启动文件4. 创建启动文件4.1. Python4.2. XML4.3. YAML 5. 编译和运行 1. 创建功能包 用Python、XML或YAML编写的启动文件可以启动和停止不同的节点&#xff0c;以及触发和处理…

设计模式(二十三):行为型之解释器模式

设计模式系列文章 设计模式(一)&#xff1a;创建型之单例模式 设计模式(二、三)&#xff1a;创建型之工厂方法和抽象工厂模式 设计模式(四)&#xff1a;创建型之原型模式 设计模式(五)&#xff1a;创建型之建造者模式 设计模式(六)&#xff1a;结构型之代理模式 设计模式…

chapter8:SpringBoot启动配置原理

尚硅谷SpringBoot顶尖教程 1. 启动流程简介 SpringBoot应用从主启动类启动后的运行流程主要包含下面几个要点&#xff1a; &#xff08;1&#xff09;准备环境 执行ApplicationContextInitializer#initialize()方法&#xff1b;监听器SpringApplicationRunListener回调cont…

ADC 读取电位器旋钮,用回差消除临界值档位跳动

就是比如&#xff0c;用电位器当旋钮做风扇调速&#xff0c;划分出10 个速度档位&#xff0c;对应10 个ADC 转换结果的阈值。如果直接比较阈值&#xff0c;当旋钮拧到临近阈值的地方时&#xff0c;ADC 结果的微小跳动会导致风扇档位在两个级别之间不停左右横跳&#xff0c;因此…