【超详细实战攻略】Jmeter逻辑控制器中if控制器、模块控制器、测试片段的使用方法

ops/2024/12/16 22:12:19/

【超详细实战攻略】Jmeter逻辑控制器中if控制器、模块控制器、测试片段的使用方法

  • 1 搭建测试对象
    • 1.1 禅道下载
    • 1.2 禅道安装
    • 1.3 运行禅道
    • 1.4 接口查看
    • 1.5 接口选择
  • 2 Jmeter前置操作
    • 2.1 创建Jmeter线程组
    • 2.2 创建信息头管理器和请求默认值
    • 2.3 添加获取token接口
    • 2.4 添加监听器
  • 3 获取token并存入系统变量
  • 4 测试片段
  • 5 模块控制器
  • 6 if控制器
    • 6.1 需求分析
    • 6.2 if控制器说明
    • 6.3 获取系统变量
    • 6.4 添加获取用户列表接口
    • 6.5 提取账号和id
    • 6.6 if控制器-判断用户存在
    • 6.7 if控制器-判断用户不存在
  • 7 测试计划效果

写作目的:
1、本文主要是通过搭建禅道为测试对象,然后介绍Jmeter的if控制器、模块控制器、测试片段的使用方法;
2、仅作为简单的介绍,不作为项目的实际参考依据,具体的依据项目测试为准。

1 搭建测试对象

1.1 禅道下载

  • 直接进入官网,选择开源版进行下载;
  • 根据自身电脑环境选择合适的版本,本文是Windows版本:
    在这里插入图片描述
  • 选择“Windows一键安装”下载即可。

1.2 禅道安装

  • 下载后如下:
    在这里插入图片描述
  • 双击运行,选择安装目录,我们安装在如下目录:
    在这里插入图片描述
  • 点击Extract,等待安装:
    在这里插入图片描述

1.3 运行禅道

  • 安装完后,在安装目录下有个exe文件双击打开:
    在这里插入图片描述

  • 双击后会打开如下页面:
    在这里插入图片描述

  • 点击“访问禅道”,或者输入:http://127.0.0.1/index.php:
    在这里插入图片描述

  • 点击【开源版】,进入登陆页面:
    在这里插入图片描述

1.4 接口查看

  • 首次登录时,登录信息如下:
# 账号:admin
# 密码:123456
  • 首次登录需要修改密码,修改成功后进入系统首页,如下:
    在这里插入图片描述
  • 点击左侧的导航栏中的【后台】,点击【二次开发】:
    在这里插入图片描述
  • 可以看到相关的API接口:
    在这里插入图片描述

1.5 接口选择

  • 我们用以下几个接口来进行说明;
  • 获取用户列表;
  • 首先是获取登录token,用于后续接口登录权限认证;
  • 创建用户接口后获取用户的id;
  • 删除用户接口时根据创建的用户id进行删除;
  • 公共接口地址为:
http://127.0.0.1/zentao/api.php/v1
接口说明
token获取/tokens
创建用户/users
删除用户/users/:id

2 Jmeter前置操作

2.1 创建Jmeter线程组

  • 以此选择【测试计划】-【添加】-【线程(用户)】-【线程组】:
    在这里插入图片描述

  • 命名为“线程组-登录获取token”:

在这里插入图片描述

2.2 创建信息头管理器和请求默认值

  • 创建http信息头管理器,以此为【测试计划】-【添加】-【配置元件】-【HTTP信息头管理器】:
    在这里插入图片描述
  • 创建后如下:
    在这里插入图片描述
  • 点击底部的【添加】按钮,添加头信息:
    在这里插入图片描述
  • 创建http请求默认值,以此为:【测试计划】-【添加】-【配置元件】-【HTTP请求默认值】:
    在这里插入图片描述
  • 创建后如下:
    在这里插入图片描述
  • 设置协议、ip、端口等信息:
    在这里插入图片描述

2.3 添加获取token接口

  • 在线程组下新建http请求,以此为:【线程组-登录获取token】-【添加】-【取样器】-【HTTP请求】:
    在这里插入图片描述
  • 添加后如下,并设置接口信息:
    在这里插入图片描述

2.4 添加监听器

  • 在测试计划下添加监听器,以此为:【测试计划】-【监听器】-【查看结果树】:
    在这里插入图片描述
  • 添加运行后如下:
    在这里插入图片描述

3 获取token并存入系统变量

  • 在以上请求【HTTP请求-登录获取token】下创建【json提取器】;
  • 以此为:【HTTP请求-登录获取token】-【添加】-【后置处理器】-【JSON提取器】:
    在这里插入图片描述
  • 提取规则如下,目的是提取登录后返回的token值:
    在这里插入图片描述
  • 添加后置处理器,将提取的变量token加入系统变量,以此为:【HTTP请求-登录获取token】-【添加】-【后置处理器】-【BeanShell 后置处理程序】:
    在这里插入图片描述
  • 写入如下内容,将token加入系统变量中:
${__setProperty(token,${token},)}

在这里插入图片描述

4 测试片段

  • 测试片段的作用类似分组的功能;

  • 我们可以把一些业务功能单元进行封装起来;

  • 比如这里创建一个名为【用户管理】的测试片段,可以在模块控制器中进行调用;

  • 创建测试片段步骤以此为:【线程组-登录获取token】-【添加】-【测试片段】-【测试片段】:
    在这里插入图片描述

  • 创建后如下:
    在这里插入图片描述

  • 注意:创建的测试片段默认是禁止状态,建议使用默认值,只有当测试片段被模块控制器调用时,才会执行,不调用执行;另外如果测试片段被启用,那么它自身下的接口也会执行(即使模块控制器没有调用测试片段)。

5 模块控制器

  • 以上创建了测试片段,片段下可以放很多的接口数据;
  • 那么如何让测试片段执行,一种是启用测试片段(不建议);另一种就是通过模块控制器进行调用;
  • 那模块控制器就是来调用测试片段的,可以指定对应的测试片段进行执行;
  • 我们创建一个模块控制器,以此为:【线程组-登录获取token】-【添加】-【逻辑控制器】-【模块控制器】:
    在这里插入图片描述
  • 创建后,要在下方选中执行的测试片段,如下:
    在这里插入图片描述

6 if控制器

6.1 需求分析

  • 前边基本把所有的步骤已经进行完了,接下来就是添加需要执行的接口数据;
  • 首先是获取登录后token,传入接口的请求头,这样才有权限;
  • 这里获取token其实就是我们之前设置好的系统变量;
  • 其次是添加需要执行的接口数据,我们的逻辑是:

① 先判断创建的用户名是不是存在;
② 如果存在,先调用删除用户接口;
③ 然后再调用创建用户接口;
④ 如果用户不存在,直接调用创建用户接口。

6.2 if控制器说明

  • i控制器其实就是一个条件判断,和代码中的if语句类似;
  • if控制器在这里的作用就是判断用户是否存在。

6.3 获取系统变量

  • 在测试片段下添加请求头,主要是获取系统变量token;
  • 以此为:【测试片段-用户管理】-【添加】-【配置元件】-【HTTP信息头管理器】:
    在这里插入图片描述
  • 创建后,点击下方的【添加】按钮获取系统变量token:
    在这里插入图片描述

6.4 添加获取用户列表接口

  • 我们在测试片段下先获取用户列表,以此为:【测试片段-用户管理】-【添加】-【取样器】-【HTTP请求】:
    在这里插入图片描述
  • 添加获取用户列表接口数据:
    在这里插入图片描述

6.5 提取账号和id

  • 我们通过获取用户列表接口可以查看系统中有多少个用户;
  • 比如运行上边的获取用户列表接口后,可以看出有两个用户:
admin、NoamaNelson

在这里插入图片描述

  • 用户列表接口的返回值如下:
{"page":1,"total":2,"limit":20,"users":[{"id":2,"dept":0,"account":"NoamaNelson","realname":"\u866b\u65e0\u6daf","role":"","pinyin":"","email":""},{"id":1,"dept":0,"account":"admin","realname":"admin","role":"","pinyin":"admin a","email":""}]
}
  • 添加后置处理器提取账号和id,提取的账号主要是用于判断这个账号是否存在;提取id是如果账号存在,后续通过id来删除用户;
  • 以此为:【HTTP请求-获取用户列表】-【添加】-【后置处理器】-【JSON提取器】:
    在这里插入图片描述
  • 提取规则如下:
$.users[?(@.account=="NoamaNelson")].account

在这里插入图片描述

  • 同样的方法添加提取器,提取id:
$.users[?(@.account=="NoamaNelson")].id

在这里插入图片描述

6.6 if控制器-判断用户存在

  • 添加if控制器,假如用户存在,以此为:【测试片段-用户管理】-【添加】-【逻辑控制器】-【IF控制器】:
    在这里插入图片描述
  • 控制器中添加如下内容:
${__jexl3("${account}" == "NoamaNelson")}

在这里插入图片描述

  • 如果用户存在,我们先通过id删除用户,以此添加删除用户的接口:【IF 控制器-用户存在】-【取样器】-【HTTP请求】:
    在这里插入图片描述

  • 添加删除接口数据:
    在这里插入图片描述

  • 删除用户后再添加用户:
    在这里插入图片描述

  • 添加用户数据为:

{"account":"NoamaNelson","password":"123456","realname":"虫无涯"
}

6.7 if控制器-判断用户不存在

  • 添加if控制器,假如用户不存在,以此为:【测试片段-用户管理】-【添加】-【逻辑控制器】-【IF控制器】;
    在这里插入图片描述

  • 注意这里的用户不存在,判断依据就是提取的用户账号是error,这个error描述是我们在提取用户账号的时候自己写的,比如:
    在这里插入图片描述

  • 那么如果用户用户存在,就直接创建用户就行了:
    在这里插入图片描述

7 测试计划效果

  • 通过以上我们就可以放心的去添加删除用户了;
  • 使用if控制器在这里的好处是:如果用户存在我们先删除后添加,避免重复添加或报错的;
  • 我们最后的整个测试计划如下:
    在这里插入图片描述
  • 或者我们可以设计如下的效果:
    在这里插入图片描述
  • 完整的脚本如下:
<?xml version="1.0" encoding="UTF-8"?>
<jmeterTestPlan version="1.2" properties="5.0" jmeter="5.6.3"><hashTree><TestPlan guiclass="TestPlanGui" testclass="TestPlan" testname="测试计划"><elementProp name="TestPlan.user_defined_variables" elementType="Arguments" guiclass="ArgumentsPanel" testclass="Arguments" testname="用户定义的变量"><collectionProp name="Arguments.arguments"/></elementProp></TestPlan><hashTree><HeaderManager guiclass="HeaderPanel" testclass="HeaderManager" testname="HTTP信息头管理器"><collectionProp name="HeaderManager.headers"><elementProp name="" elementType="Header"><stringProp name="Header.name">content-type</stringProp><stringProp name="Header.value">application/json</stringProp></elementProp></collectionProp></HeaderManager><hashTree/><ConfigTestElement guiclass="HttpDefaultsGui" testclass="ConfigTestElement" testname="HTTP请求默认值"><stringProp name="HTTPSampler.domain">127.0.0.1</stringProp><stringProp name="HTTPSampler.protocol">http</stringProp><elementProp name="HTTPsampler.Arguments" elementType="Arguments" guiclass="HTTPArgumentsPanel" testclass="Arguments" testname="用户定义的变量"><collectionProp name="Arguments.arguments"/></elementProp><stringProp name="HTTPSampler.implementation">HttpClient4</stringProp></ConfigTestElement><hashTree/><ResultCollector guiclass="ViewResultsFullVisualizer" testclass="ResultCollector" testname="查看结果树"><boolProp name="ResultCollector.error_logging">false</boolProp><objProp><name>saveConfig</name><value class="SampleSaveConfiguration"><time>true</time><latency>true</latency><timestamp>true</timestamp><success>true</success><label>true</label><code>true</code><message>true</message><threadName>true</threadName><dataType>true</dataType><encoding>false</encoding><assertions>true</assertions><subresults>true</subresults><responseData>false</responseData><samplerData>false</samplerData><xml>false</xml><fieldNames>true</fieldNames><responseHeaders>false</responseHeaders><requestHeaders>false</requestHeaders><responseDataOnError>false</responseDataOnError><saveAssertionResultsFailureMessage>true</saveAssertionResultsFailureMessage><assertionsResultsToSave>0</assertionsResultsToSave><bytes>true</bytes><sentBytes>true</sentBytes><url>true</url><threadCounts>true</threadCounts><idleTime>true</idleTime><connectTime>true</connectTime></value></objProp><stringProp name="filename"></stringProp></ResultCollector><hashTree/><SetupThreadGroup guiclass="SetupThreadGroupGui" testclass="SetupThreadGroup" testname="setUp 线程组-登录获取token"><intProp name="ThreadGroup.num_threads">1</intProp><intProp name="ThreadGroup.ramp_time">1</intProp><boolProp name="ThreadGroup.same_user_on_next_iteration">true</boolProp><stringProp name="ThreadGroup.on_sample_error">continue</stringProp><elementProp name="ThreadGroup.main_controller" elementType="LoopController" guiclass="LoopControlPanel" testclass="LoopController" testname="循环控制器"><stringProp name="LoopController.loops">1</stringProp><boolProp name="LoopController.continue_forever">false</boolProp></elementProp></SetupThreadGroup><hashTree><HTTPSamplerProxy guiclass="HttpTestSampleGui" testclass="HTTPSamplerProxy" testname="HTTP请求-登录获取token"><stringProp name="HTTPSampler.path">/zentao/api.php/v1/tokens</stringProp><boolProp name="HTTPSampler.follow_redirects">true</boolProp><stringProp name="HTTPSampler.method">POST</stringProp><boolProp name="HTTPSampler.use_keepalive">true</boolProp><boolProp name="HTTPSampler.postBodyRaw">true</boolProp><elementProp name="HTTPsampler.Arguments" elementType="Arguments"><collectionProp name="Arguments.arguments"><elementProp name="" elementType="HTTPArgument"><boolProp name="HTTPArgument.always_encode">false</boolProp><stringProp name="Argument.value">{&quot;account&quot;: &quot;admin&quot;, &quot;password&quot;: &quot;Zb918110&quot;}</stringProp><stringProp name="Argument.metadata">=</stringProp></elementProp></collectionProp></elementProp></HTTPSamplerProxy><hashTree><JSONPostProcessor guiclass="JSONPostProcessorGui" testclass="JSONPostProcessor" testname="JSON提取器-提取token"><stringProp name="JSONPostProcessor.referenceNames">token</stringProp><stringProp name="JSONPostProcessor.jsonPathExprs">$.token</stringProp><stringProp name="JSONPostProcessor.match_numbers">1</stringProp><stringProp name="JSONPostProcessor.defaultValues">error</stringProp></JSONPostProcessor><hashTree/><BeanShellPostProcessor guiclass="TestBeanGUI" testclass="BeanShellPostProcessor" testname="BeanShell 后置处理程序"><stringProp name="filename"></stringProp><stringProp name="parameters"></stringProp><boolProp name="resetInterpreter">false</boolProp><stringProp name="script">${__setProperty(token,${token},)}</stringProp></BeanShellPostProcessor><hashTree/></hashTree></hashTree><ThreadGroup guiclass="ThreadGroupGui" testclass="ThreadGroup" testname="线程组-用户管理"><intProp name="ThreadGroup.num_threads">1</intProp><intProp name="ThreadGroup.ramp_time">1</intProp><boolProp name="ThreadGroup.same_user_on_next_iteration">true</boolProp><stringProp name="ThreadGroup.on_sample_error">continue</stringProp><elementProp name="ThreadGroup.main_controller" elementType="LoopController" guiclass="LoopControlPanel" testclass="LoopController" testname="循环控制器"><stringProp name="LoopController.loops">1</stringProp><boolProp name="LoopController.continue_forever">false</boolProp></elementProp></ThreadGroup><hashTree><HeaderManager guiclass="HeaderPanel" testclass="HeaderManager" testname="HTTP信息头管理器"><collectionProp name="HeaderManager.headers"><elementProp name="" elementType="Header"><stringProp name="Header.name">Token</stringProp><stringProp name="Header.value">${__property(token,,)}</stringProp></elementProp></collectionProp></HeaderManager><hashTree/><ModuleController guiclass="ModuleControllerGui" testclass="ModuleController" testname="模块控制器-用户管理"><collectionProp name="ModuleController.node_path"><stringProp name="869052411">测试计划</stringProp><stringProp name="869052411">测试计划</stringProp><stringProp name="-578308503">线程组-用户管理</stringProp><stringProp name="2078815177">测试片段-用户管理</stringProp></collectionProp></ModuleController><hashTree/><TestFragmentController guiclass="TestFragmentControllerGui" testclass="TestFragmentController" testname="测试片段-用户管理" enabled="false"/><hashTree><HTTPSamplerProxy guiclass="HttpTestSampleGui" testclass="HTTPSamplerProxy" testname="HTTP请求-获取用户列表"><stringProp name="HTTPSampler.path">/zentao/api.php/v1/users</stringProp><boolProp name="HTTPSampler.follow_redirects">true</boolProp><stringProp name="HTTPSampler.method">GET</stringProp><boolProp name="HTTPSampler.use_keepalive">true</boolProp><boolProp name="HTTPSampler.postBodyRaw">false</boolProp><elementProp name="HTTPsampler.Arguments" elementType="Arguments" guiclass="HTTPArgumentsPanel" testclass="Arguments" testname="用户定义的变量"><collectionProp name="Arguments.arguments"/></elementProp></HTTPSamplerProxy><hashTree><JSONPostProcessor guiclass="JSONPostProcessorGui" testclass="JSONPostProcessor" testname="JSON提取器-提取账号"><stringProp name="JSONPostProcessor.referenceNames">account</stringProp><stringProp name="JSONPostProcessor.jsonPathExprs">$.users[?(@.account==&quot;NoamaNelson&quot;)].account</stringProp><stringProp name="JSONPostProcessor.match_numbers">1</stringProp><stringProp name="JSONPostProcessor.defaultValues">error</stringProp></JSONPostProcessor><hashTree/><JSONPostProcessor guiclass="JSONPostProcessorGui" testclass="JSONPostProcessor" testname="JSON提取器-提取id"><stringProp name="JSONPostProcessor.referenceNames">id</stringProp><stringProp name="JSONPostProcessor.jsonPathExprs">$.users[?(@.account==&quot;NoamaNelson&quot;)].id</stringProp><stringProp name="JSONPostProcessor.match_numbers">1</stringProp><stringProp name="JSONPostProcessor.defaultValues">error</stringProp></JSONPostProcessor><hashTree/></hashTree><IfController guiclass="IfControllerPanel" testclass="IfController" testname="IF 控制器-用户存在"><stringProp name="IfController.condition">${__jexl3(&quot;${account}&quot; == &quot;NoamaNelson&quot;)}</stringProp><boolProp name="IfController.evaluateAll">false</boolProp><boolProp name="IfController.useExpression">true</boolProp></IfController><hashTree><HTTPSamplerProxy guiclass="HttpTestSampleGui" testclass="HTTPSamplerProxy" testname="HTTP请求-通过id删除用户"><stringProp name="HTTPSampler.path">/zentao/api.php/v1/users/:${id}</stringProp><boolProp name="HTTPSampler.follow_redirects">true</boolProp><stringProp name="HTTPSampler.method">DELETE</stringProp><boolProp name="HTTPSampler.use_keepalive">true</boolProp><boolProp name="HTTPSampler.postBodyRaw">false</boolProp><elementProp name="HTTPsampler.Arguments" elementType="Arguments" guiclass="HTTPArgumentsPanel" testclass="Arguments" testname="用户定义的变量"><collectionProp name="Arguments.arguments"/></elementProp></HTTPSamplerProxy><hashTree/><HTTPSamplerProxy guiclass="HttpTestSampleGui" testclass="HTTPSamplerProxy" testname="HTTP请求-创建用户"><stringProp name="HTTPSampler.path">/zentao/api.php/v1/users</stringProp><boolProp name="HTTPSampler.follow_redirects">true</boolProp><stringProp name="HTTPSampler.method">POST</stringProp><boolProp name="HTTPSampler.use_keepalive">true</boolProp><boolProp name="HTTPSampler.postBodyRaw">true</boolProp><elementProp name="HTTPsampler.Arguments" elementType="Arguments"><collectionProp name="Arguments.arguments"><elementProp name="" elementType="HTTPArgument"><boolProp name="HTTPArgument.always_encode">false</boolProp><stringProp name="Argument.value">{&quot;account&quot;: &quot;NoamaNelson&quot;, &quot;password&quot;: &quot;123456&quot;, &quot;realname&quot;: &quot;虫无涯&quot;}</stringProp><stringProp name="Argument.metadata">=</stringProp></elementProp></collectionProp></elementProp></HTTPSamplerProxy><hashTree/></hashTree><IfController guiclass="IfControllerPanel" testclass="IfController" testname="IF 控制器-用户不存在"><stringProp name="IfController.condition">${__jexl3(&quot;${account}&quot; == &quot;error&quot;)}</stringProp><boolProp name="IfController.evaluateAll">false</boolProp><boolProp name="IfController.useExpression">true</boolProp></IfController><hashTree><HTTPSamplerProxy guiclass="HttpTestSampleGui" testclass="HTTPSamplerProxy" testname="HTTP请求-创建用户"><stringProp name="HTTPSampler.path">/zentao/api.php/v1/users</stringProp><boolProp name="HTTPSampler.follow_redirects">true</boolProp><stringProp name="HTTPSampler.method">POST</stringProp><boolProp name="HTTPSampler.use_keepalive">true</boolProp><boolProp name="HTTPSampler.postBodyRaw">true</boolProp><elementProp name="HTTPsampler.Arguments" elementType="Arguments"><collectionProp name="Arguments.arguments"><elementProp name="" elementType="HTTPArgument"><boolProp name="HTTPArgument.always_encode">false</boolProp><stringProp name="Argument.value">{&quot;account&quot;: &quot;NoamaNelson&quot;, &quot;password&quot;: &quot;123456&quot;, &quot;realname&quot;: &quot;虫无涯&quot;}</stringProp><stringProp name="Argument.metadata">=</stringProp></elementProp></collectionProp></elementProp></HTTPSamplerProxy><hashTree/></hashTree></hashTree></hashTree></hashTree></hashTree>
</jmeterTestPlan>

http://www.ppmy.cn/ops/142475.html

相关文章

PyTorch中detach() 函数详解

PyTorch detach() 函数详解 在使用 PyTorch 进行深度学习模型的训练中&#xff0c;detach() 是一个非常重要且常用的函数。它主要用于在计算图中分离张量&#xff0c;从而实现高效的内存管理和防止梯度传播。本文将详细介绍 detach() 的作用、原理及其实际应用场景&#xff0c…

3D 生成重建035-DiffRF直接生成nerf

3D 生成重建035-DiffRF直接生成nerf 文章目录 0 论文工作1 论文方法2 实验结果 0 论文工作 本文提出了一种基于渲染引导的三维辐射场扩散新方法DiffRF&#xff0c;用于高质量的三维辐射场合成。现有的方法通常难以生成具有细致纹理和几何细节的三维模型&#xff0c;并且容易出…

最短路----Dijkstra算法详解

简介 迪杰斯特拉&#xff08;Dijkstra&#xff09;算法是一种用于在加权图中找到单个源点到所有其他顶点的最短路径的算法。它是由荷兰计算机科学家艾兹格迪科斯彻&#xff08;Edsger Dijkstra&#xff09;在1956年提出的。Dijkstra算法适用于处理带有非负权重的图。迪杰斯特拉…

Git 命令大全:全面掌握版本控制系统

一、引言 Git 是一款广泛使用的分布式版本控制系统&#xff0c;它在软件开发、项目协作以及代码管理等方面发挥着极为重要的作用。无论是个人开发者独自管理代码库&#xff0c;还是大型团队协同开发复杂项目&#xff0c;Git 都提供了强大而灵活的功能来满足各种需求。通过掌握…

服务器一般装什么系统?

在服务器管理中&#xff0c;操作系统的选择是一个关键因素&#xff0c;它直接影响到服务器的稳定性、性能和可维护性。那么为什么有些服务器选择Linux&#xff0c;而不是Windows&#xff1f;选择合适的操作系统对服务器的性能和安全性有多么重要&#xff1f; 在众多操作系统中…

网络编程之初识网络

1.IP地址 IP地址实际上全称是&#xff08;互联网协议地址&#xff09;&#xff0c;是每一台计算机在网络上面的编号&#xff0c;每个连接Internet 的设备&#xff08;如计算机、路由器、服务器等&#xff09;都被分配一个唯一的 IP 地址&#xff0c;简单来说&#xff0c;IP地址…

Linux下禁止root远程登录访问

开始讲故事 Long long ago&#xff0c; Linux远程访问方式有telnet、ssh两种协议&#xff1b;有人可能还会说vnc和rdp协议方式&#xff0c;后面这两种主要是可视化桌面场景下的&#xff0c;并非主流。 时过境迁&#xff0c;telnet因安全性低逐渐被禁用淘汰&#xff0c;最后就…

前端报错npm ERR cb() never called问题

环境使用node版本v14.21.3&#xff0c;npm版本6.14.18 1.问题描述 1.1使用npm install后报错 npm ERR! cb() never called!npm ERR! This is an error with npm itself. Please report this error at: npm ERR! ? ? <https://npm.community>npm ERR! A complete log…