python自动化测试实战 —— 单元测试框架

软件测试专栏

感兴趣可看:软件测试专栏
                     自动化测试学习部分源码
python自动化测试相关知识:
       【如何学习Python自动化测试】—— 自动化测试环境搭建
       【如何学习python自动化测试】—— 浏览器驱动的安装 以及 如何更新driver
       【如何学习Python自动化测试】—— 页面元素定位
       【如何学习Python自动化测试】—— 时间等待
       【如何学习Python自动化测试】—— 浏览器操作
       【如何学习Python自动化测试】—— 鼠标键盘操作
       【如何学习Python自动化测试】—— 多层窗口定位
       【如何学习Python自动化测试】—— 警告框处理
       【如何学习Python自动化测试】—— Cookie 处理
       【如何学习Python自动化测试】—— expected_conditions
       【如何学习Python自动化测试】—— Python 的 unittest 框架
       【如何学习Python自动化测试】—— HTMLTestRunner 生成测试报告


单元测试框架

  • 软件测试专栏
  • 实战相关知识
    • 1.unittest框架
    • 2.Pytest测试框架
  • 实战内容
    • 1.“百度翻译”的页面UI测试
      • (1)利用unittest框架完成
      • (2)利用Pytest框架完成
    • 2.“新浪微博”的两个页面UI测试
      • (1)利用unittest框架完成
      • (2)利用Pytest框架完成
  • 操作异常问题与解决方案
  • 总结
  • 附录

实战相关知识

1.unittest框架

      unittest框架是Python语言内置的单元测试框架,Python编写的Web UI自动化测试脚本可以借助该框架来组织和执行。

(1)UnitTest组成部分

  • TestFixture(测试固件):测试用例的准备和销毁。
  • TestCase(测试用例):一个TestCase的实例就是一个测试用例。
  • TestSuite(测试套件):将多个测试用例集合在一起就是一个TestSuite。
  • TestRunner(测试运行器):使用TextTestRunner提供的 run( )方法执行测试用例。

(2)使用UnitTest测试框架注意:

  • 首先需要导入unittest包:import unittest。
  • 导入包的语句和定义测试类中间要隔两个空行。
  • 新建测试类的名称,建议每个单词的首字母大写,编写顺序建议保持团队一致。
  • 测试类必须继承unittest.TestCase
  • 接下来可以编写setUp,setUp方法不是必须需要的。
  • 接下来可以编写测试用例,测试用例名称以”test_”开头。
           self.assertXXX是UnitTest提供的断言方法
  • 用例写完,就可以编写tearDown,tearDown也不是必须要有。
          tearDown写完后空两行,就可以使用unittest.main()进行测试了。

(3)用HTMLTestRunner模块生成可视化测试报告。

2.Pytest测试框架

      Pytest测试框架在当今自动化测试中更受欢迎,是一个非常流行且成熟的全功能的Python测试框架,适用于单元测试、UI测试、接口测试。

(1)Pytest规则

  • 文件命名:默认以“test_”开头或者以”_test”结尾。
  • 测试类(class)命名:默认以“Test”开头
  • 测试方法(函数)命名:默认以“test_”开头
  • 断言:直接使用Python语言的断言assert

(2)Pytest测试固件

  • 如果测试文件中没有定义class,而是直接定义的函数,那么使用setup_module、teardown_module 和 setup_function、teardown_function
  • 如果测试文件中定义了class,就使用setup_class、teardown_class 和 setup_method、teardown_method
  • 不管是否定义class,都可以使用setup、teardown来实现在每个方法(或函数)的前后执行。
  • 建议在一个项目中约定好是定义class来组织测试用例,还是直接定义函数来组织用例。

(3)测试用例
      Pytest和unittest的框架风格基本一致,但有几点要注意:

  • 注意函数或方法名以“test_”开头
  • 直接通过函数定义测试用例的话,def后面的括号中没有self
  • 通过class中的方法定义测试用例的话,def后面的括号中有self。
  • 断言用的是Python的断言方式。

(4)用第三方插件完成测试报告的生成。

实战内容

1.“百度翻译”的页面UI测试

      将“百度翻译”的页面UI测试(至少包含翻译功能这个测试用例)的自动化测试线性脚本,分别用unittest和Pytest框架完成,并生成测试报告。

(1)利用unittest框架完成

首先需要导入包
在这里插入图片描述

新建测试类的名称,测试类必须继承unittest.TestCase
在这里插入图片描述

编写setUp
在这里插入图片描述

编写测试用例,测试用例名称以”test_”开头。

  • 用例1:输入的文字与语言不一致
    在这里插入图片描述
    使用assertTRUE断言方法,断言网页会输出与输入文字匹配的语言,如下图
    在这里插入图片描述

  • 用例2:输入的文字与语言一致
    在这里插入图片描述
    使用assertEqual断言方法,断言输出文本框结果与预期结果一致,如下图
    在这里插入图片描述
    用例写完,编写tearDown
    在这里插入图片描述
    tearDown写完后空两行,使用unittest.main()进行测试
    在这里插入图片描述
    用第三方插件完成测试报告的生成
    将HTMLTestRunner.py和run.py以及__init__.py文件放置与脚本文件同级目录下,如下图:
    在这里插入图片描述
    编写run.py文件
    在这里插入图片描述
    运行run.py文件
    运行结果如下
    在这里插入图片描述
    出现如上结果,说明两个测试用例通过
    测试报告如下
    在这里插入图片描述

(2)利用Pytest框架完成

首先需要导入包
在这里插入图片描述
新建测试类的名称,默认以“Test”开头
在这里插入图片描述

编写setUp
在这里插入图片描述
编写测试用例,测试用例名称以”test_”开头。

  • 用例1:输入的文字与语言不一致
    在这里插入图片描述
    使用Python语言的断言assert,断言网页会输出与输入文字匹配的语言

  • 用例2:输入的文字与语言一致
    在这里插入图片描述
    使用Python语言的断言assert,断言输出文本框结果与预期结果一致

  • 用例写完,编写teardown
    在这里插入图片描述
    全局设置
    创建一个配置文件:pytest.ini
    该文件要和需要执行的测试文件所在的目录文件在同一级
    如下图所示:
    在这里插入图片描述
    pytest.ini内容如下:
    在这里插入图片描述
    通过addopts来设置命令行参数;-v 监控、失败重试的次数、重试的时间间隔、按标签来执行、生成测试报告,多个参数之间用空格分隔

  • 接着在py文件teardown后两行,使用pytest.main()进行测试
    在这里插入图片描述
    运行结果如下
    在这里插入图片描述
    出现passed说明两个测试用例通过
    测试报告如下
    在这里插入图片描述

2.“新浪微博”的两个页面UI测试

      将“新浪微博”的两个页面UI测试(至少包含登录账号、发文字微博两个测试用例)的自动化测试线性脚本,分别用unittest和Pytest框架完成,并生成测试报告。

(1)利用unittest框架完成

首先需要导入包
在这里插入图片描述

新建测试类的名称,测试类必须继承unittest.TestCase
在这里插入图片描述

编写setUp,因为用例二需要登录才能进行测试,所以登录需要写进setUp
在这里插入图片描述

编写测试用例,测试用例名称以”test_”开头。

  • 用例1:登录账号,这里仅测试是否通过
    在这里插入图片描述
    使用assertIn断言方法,断言用户名‘111Nuyoah111’会出现在网页中
    在这里插入图片描述

  • 用例2:发布微博文字
    使用assertIn断言方法,断言发布内容成功,如下图
    在这里插入图片描述

  • 用例写完,编写tearDown
    在这里插入图片描述

  • tearDown写完后空两行,使用unittest.main()进行测试
    在这里插入图片描述
    用第三方插件完成测试报告的生成
    将HTMLTestRunner.py和run.py以及__init__.py文件放置与脚本文件同级目录下,如下图:
    在这里插入图片描述

  • 编写run.py文件
    在这里插入图片描述
    运行run.py文件
    运行结果如下
    在这里插入图片描述
    出现如上结果,说明两个测试用例通过
    测试报告如下
    在这里插入图片描述

(2)利用Pytest框架完成

首先需要导入包
在这里插入图片描述

新建测试类的名称,默认以“Test”开头
在这里插入图片描述

编写setUp
在这里插入图片描述

编写测试用例,测试用例名称以”test_”开头。

  • 用例1:登录账号,这里仅测试是否通过
    在这里插入图片描述

使用assertIn断言方法,断言用户名‘xxx’会出现在网页中

  • 用例2:发布微博文字
    在这里插入图片描述
    使用assertIn断言方法,断言发布内容成功

  • 用例写完,编写teardown
    在这里插入图片描述
    全局设置
    创建一个配置文件:pytest.ini
    该文件要和需要执行的测试文件所在的目录文件在同一级
    如下图所示:
    在这里插入图片描述
    pytest.ini内容如下:
    在这里插入图片描述
    通过addopts来设置命令行参数;-v 监控、失败重试的次数、重试的时间间隔、按标签来执行,多个参数之间用空格分隔
    运行生成测试报告,在py文件teardown后两行,使用pytest.main()进行测试
    在这里插入图片描述

  • 运行结果如下
    在这里插入图片描述
    出现passed说明两个测试用例通过
    测试报告如下
    在这里插入图片描述

操作异常问题与解决方案

  • 问题1:pytest框架脚本运行失败
  • 解决方法:通过查询,发现是pytest环境问题,所以将pytest及相关插件卸载后在全局目录下重新下载

  • 问题2:allure在成功使用后,再次使用会自动跳过测试用例
    在这里插入图片描述

  • 解决方法:allure环境不稳定,更换测试报告方法,使用pytest-html完成测试报告


  • 问题3:微博无法使用账号密码登录,测试用例无法完成登录的试错
  • 解决方法:修改代码,直接在setup固件进行手动登录,最后加入一个登录的测试用例进行断言

总结

      Unittest和Pytest是Python中常用的两个测试框架,用于编写和执行单元测试。

      Unittest是Python的内置测试框架之一,可以通过导入unittest模块来使用。Unittest提供了一组用于编写测试用例的类和方法,测试用例是通过继承unittest.TestCase类来创建的。测试方法以test_开头,并且可以使用断言方法(如assertEqual()、assertTrue()等)来验证预期行为。Unittest提供了丰富的功能和工具,如测试套件、测试装置(setUp()和tearDown()方法)、测试发现等。可以使用命令行工具或集成开发环境(IDE)来运行Unittest测试。

      Pytest是一个第三方的Python测试框架,可以通过安装pytest库来使用。Pytest提供了更简洁、灵活和可扩展的方式来编写测试用例。不需要继承特定的基类,可以使用普通的函数定义测试用例,用assert语句来断言结果。Pytest具有丰富的插件生态系统和许多附加功能,例如自动发现测试文件、参数化测试、夹具(fixtures)等。Pytest支持使用命令行工具来运行测试,并提供了丰富的输出和报告选项。

      unitest和pytest的区别是:语法风格:Unittest使用类和方法的方式来组织测试用例,而Pytest使用函数定义测试用例。灵活性:Pytest提供了更灵活和简洁的语法,没有像Unittest那样的约束。Pytest的编写方式更为简单,减少了样板代码的编写。插件生态系统:Pytest具有丰富的插件生态系统,提供了许多附加功能和扩展选项,而Unittest相对较少。自动发现测试:Pytest具有自动发现测试文件和用例的功能,而Unittest需要手动设置测试套件和加载用例。夹具支持:Pytest提供了强大的夹具(fixtures)机制,用于管理测试数据和环境设置。Unittest也支持夹具,但Pytest的夹具功能更为灵活和强大。

      总体而言,Pytest相对于Unittest提供了更简洁、灵活且功能更丰富的测试框架,能够简化测试代码的编写和维护,并提供更好的测试发现和报告功能。当然,选择使用哪个框架取决于个人偏好和项目需求。

附录

部分源码(与上面截图有所不同,仅供参考)

Baidu unittest

from selenium import webdriver
import unittest
from time import sleep
from selenium.webdriver.common.by import Byclass TestBaidu(unittest.TestCase):def setUp(self):self.driver = webdriver.Chrome()self.driver.maximize_window()self.driver.implicitly_wait(20)# 访问登录页self.driver.get('https://fanyi.baidu.com/')# 移除广告弹窗self.driver.find_element(By.CLASS_NAME,'app-guide-close').click()def test_001(self):# 用例一:输入的文字与选择的语言不一致# 选择语言阿拉伯语language = self.driver.find_element(By.CLASS_NAME,'select-from-language')language.click()language_text=self.driver.find_element(By.XPATH,'//*[@id="lang-panel-container"]''/div/div[5]/div[1]/div[1]/div/span[1]')language_text.click()#输入翻译文本英语intext =self.driver.find_element(By.CLASS_NAME,'textarea')intext.send_keys('love')sleep(2)# 输入不匹配文字后的提示信息self.assertTrue(self.driver.find_element(By.LINK_TEXT,'英语'))def test_002(self):# # 用例二:输入的文字与选择的语言一致# 选择语言英语language = self.driver.find_element(By.CLASS_NAME, 'select-from-language')language.click()language_text = self.driver.find_element(By.XPATH,'//*[@id="lang-panel-container"]''/div/div[5]/div[1]/div[21]/div/span[1]')language_text.click()# 输入翻译文本英语intext = self.driver.find_element(By.CLASS_NAME, 'textarea')intext.send_keys('love')sleep(2)# 输入不匹配文字后的提示信息outtext = self.driver.find_element(By.CLASS_NAME, 'output-bd').textself.assertEqual(outtext,'爱')def tearDown(self):self.driver.quit()if __name__ == '__main__':unittest.main()

Baidu pytest

from time import sleepimport pytest
from selenium import webdriver
from selenium.webdriver.common.by import Byclass TestBaidu():def setup(self):self.driver = webdriver.Chrome()self.driver.maximize_window()self.driver.implicitly_wait(20)# 访问登录页self.driver.get('https://fanyi.baidu.com/')# 移除广告弹窗self.driver.find_element(By.CLASS_NAME,'app-guide-close').click()@pytest.mark.L1def test_001(self):# 用例一:输入的文字与选择的语言不一致# 选择语言阿拉伯语language = self.driver.find_element(By.CLASS_NAME,'select-from-language')language.click()language_text=self.driver.find_element(By.XPATH,'//*[@id="lang-panel-container"]''/div/div[5]/div[1]/div[1]/div/span[1]')language_text.click()#输入翻译文本英语intext =self.driver.find_element(By.CLASS_NAME,'textarea')intext.send_keys('love')sleep(2)# 输入不匹配文字后的提示信息assert '英语' == self.driver.find_element(By.LINK_TEXT,'英语').text@pytest.mark.L2def test_002(self):# # 用例二:输入的文字与选择的语言一致# 选择语言英语language = self.driver.find_element(By.CLASS_NAME, 'select-from-language')language.click()language_text = self.driver.find_element(By.XPATH,'//*[@id="lang-panel-container"]''/div/div[5]/div[1]/div[21]/div/span[1]')language_text.click()# 输入翻译文本英语intext = self.driver.find_element(By.CLASS_NAME, 'textarea')intext.send_keys('love')sleep(2)# 输入不匹配文字后的提示信息outtext = self.driver.find_element(By.CLASS_NAME, 'output-bd').textassert outtext=='爱'def teardown(self):self.driver.quit()if __name__ == '__main__':pytest.main(["-s", "./test_baidu.py"])

Weibo unittest

from selenium import webdriver
import time,unittest
# 通过时间戳,构造唯一project name
from selenium.webdriver.common.by import Byproject_name = 'project_{}'.format(time.time())class TestNewProject(unittest.TestCase):def setUp(self):self.driver = webdriver.Chrome()self.driver.maximize_window()self.driver.implicitly_wait(20)# 登录self.driver.get('https://weibo.com')self.driver.find_element(By.CLASS_NAME, 'LoginCard_btn_Jp_u1').click()time.sleep(8)  # 手动扫码登录def test_new_project(self):# 测试搜索框输入框self.driver.find_element(By.CLASS_NAME,'Form_input_2gtXx').send_keys('123')time.sleep(1)self.driver.find_element(By.CLASS_NAME,'Visible_angle_MP2Km').click()time.sleep(1)self.driver.find_element(By.XPATH,'//*[@id="homeWrap"]/div[1]/div/div[4]/div/div[3]/div/div/div[4]').click()# 测试发布self.driver.find_element(By.CLASS_NAME,'Tool_btn_2Eane').click()time.sleep(1)self.assertIn('123', self.driver.page_source)def tearDown(self):self.driver.quit()if __name__ == '__main__':unittest.main()

Weibo pytest

import time
import pytest
from selenium import webdriver
from selenium.webdriver.common.by import Byclass TestWeibo  ():def setup(self):self.driver = webdriver.Chrome()self.driver.maximize_window()self.driver.implicitly_wait(20)# 登录self.driver.get('https://weibo.com')self.driver.find_element(By.CLASS_NAME, 'LoginCard_btn_Jp_u1').click()time.sleep(8)  # 手动扫码登录@pytest.mark.L1def test_new_project(self):assert 'xxx' in self.driver.page_source#此处自行修改“xxx”# 测试搜索框输入框self.driver.find_element(By.CLASS_NAME,'Form_input_2gtXx').send_keys('123')time.sleep(1)self.driver.find_element(By.CLASS_NAME,'Visible_angle_MP2Km').click()time.sleep(1)self.driver.find_element(By.XPATH,'//*[@id="homeWrap"]/div[1]/div/div[4]/div/div[3]/div/div/div[4]').click()# 测试发布self.driver.find_element(By.CLASS_NAME,'Tool_btn_2Eane').click()time.sleep(1)assert '123' in self.driver.page_sourcedef teardown(self):self.driver.quit()if __name__ == '__main__':pytest.main(["-s", "test_weibo.py", "--html=./report.html"])

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

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

相关文章

swing快速入门(五)

注释很详细,直接上代码 上一篇 本篇新增内容: 1.布局管理器BorderLayout 2.自适应尺寸方法pack() import java.awt.*; public class swing_test_3 {public static void main(String[] args) {Frame framenew Frame("演示BorderLayout");//…

cordic 算法学习记录

参考:b站教学视频FPGA:Cordic算法介绍与实现_哔哩哔哩_bilibili FPGA硬件实现加减法、移位等操作比较简单,但是实现乘除以及函数计算复杂度高且占用资源多,常见的计算三角函数/平方根的求解方式有①查找表:先把函数对应…

JVM面试连环炮:你准备好迎接挑战了吗?

在Java开发领域,JVM面试一直是一个热门话题。作为一名优秀的开发者,你是否已经准备好迎接这场挑战了呢?今天,我们就来深度解析一下JVM面试的热点问题,帮助你更好地应对面试,一举拿下offer! 1、…

vim + ctags 跳转, 查看函数定义

yum install ctags Package ctags-5.8-13.el7.x86_64 already installed and latest version 创建 /home/mzh/pptp-master/tags.sh #!/usr/bin/shWORKDIR/home/mzh/pptp-masterfind ${WORKDIR} -name "*.[c|h]" | xargs ctags -f ${WORKDIR}/tags find /usr/inclu…

mysql数据库学习笔记(1)

今天开始学mysql数据库,为什么要学这个呢,因为数据库可结构化存储大量的数据信息,方便用户进行有效的检索和访问。数据库可有效地保持数据信息的一致性、完整性、降低数据冗余。数据库可满足应用的共享和安全方面的要求,把数据放在…

java--认识异常、自定义异常

1.异常体系 Error:代表的系统级别错误(属于严重问题),也就是说系统一旦出现问题,sun公司会把这些问题封装成Error对象给出来,说白了,Error是给sun公司自己用的,不是给我们程序员用的,因此我们开…

大数据技术2:大数据处理流程

前言:下图是一个简化的大数据处理流程图,大数据处理的主要流程包括数据收集、数据存储、数据处理、数据应用等主要环节。 1.1 数据收集 大数据处理的第一步是数据的收集。现在的中大型项目通常采用微服务架构进行分布式部署,所以数据的采集需…

MySQL慢SQL优化思路

MySQL慢SQL优化思路 具体思路: 1、慢查询日志记录慢 SQL 2、explain 分析 SQL 的执行计划 3、profile 分析执行耗时 4、Optimizer Trace 分析详情 5、确定问题并采用相应的措施 1、查看慢日志 1.1 使用命令查询慢日志配置 mysql> show variables like s…

mysql 5.7.34升级到5.7.44修补漏洞

mysql 5.7.34旧版本,漏扫有漏洞,升级到最新版本 旧版本5.7.34在 /home/mysql/mysql中安装 备份旧版本数据还有目录 数据库备份升级 tar -xf mysql-5.7.44-el7-x86_64.tar #覆盖旧版本数据库文件 #注意看看文件是否和你起服务的用户一样 \cp -r mysql-5…

decomposition-based multi-objective algorithm4SPDPTW

关键词 文章概述 研究背景 多目标选择性接送和配送问题(PDPs):研究涉及多目标选择性接送和配送问题,这些问题传统上从单一目标角度进行探讨,以寻找最具盈利性的请求集合,同时遵守一系列限制条件。 经济和…

基于OpenCV+CNN+IOT+微信小程序智能果实采摘指导系统——深度学习算法应用(含python、JS工程源码)+数据集+模型(五)

目录 前言总体设计系统整体结构图系统流程图 运行环境Python环境TensorFlow 环境Jupyter Notebook环境Pycharm 环境微信开发者工具OneNET云平台 模块实现1. 数据预处理2. 创建模型并编译3. 模型训练及保存4. 上传结果5. 小程序开发1)查询图片2)查询识别结…

【收获】成长之路

目录 一、前言二、计算机方面三、专业知识方面四、总结 一、前言 四年,对于一个人的成长来说,是一个相当重要的阶段。在这段时间里,我经历了许多挑战、收获了许多成就,也在不断地成长和改变。回首这四年的点点滴滴,我深…

安装LLaMA-Factory微调chatglm3,修改自我认知

安装git clone https://github.com/hiyouga/LLaMA-Factory.git conda create -n llama_factory python3.10 conda activate llama_factory cd LLaMA-Factory pip install -r requirements.txt 之后运行 CUDA_VISIBLE_DEVICES0 python src/train_web.py,按如下配置…

市场全局复盘 20231211

昨日回顾: SELECT TOP 10000 CODE,成交额排名,净流入排名,代码,名称,DDE大单金额,涨幅,所属行业,主力净额,DDE大单净量,CONVERT(DATETIME, 最后涨停时间, 120) AS 最后涨停时间 FROM dbo.全部A股20231208_ALL WHERE 连板天 > 1AND DDE大单净量 > …

什么是零拷贝

什么是零拷贝? 快速理解 快速理解 要想理解零拷贝,首先要了解操作系统的IO流程,因为有内核态和用户态的区别,为了保证安全性和缓存,普通的读写流程如下: (对于Java程序而言,还会多了一个堆外内…

java的long类型超过9位报错:the literal 987654321000 of type int is out of range

java的long类型超过9位报错 1、报错提示2、报错截图3、解决办法4、参考文章 1、报错提示 the literal 987654321000 of type int is out of range 2、报错截图 3、解决办法 long类型是一种用于表示较大整数的数据类型,范围比int类型更广泛。然而,即使…

用PHP和HTML做登录注册操作数据库Mysql

用PHP和HTML做登录注册操作数据库Mysql 两个HTML页面&#xff0c;两个PHP,两个css,两张图片&#xff0c;源码资源在上方。 目录 HTML页面 login.html <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta nam…

IDEA卡顿,进行性能优化设置(亲测有效)——情况二

问题背景与现象 IDEA今天突然显示到期&#xff0c;于是从同事那边搞到一个很好用的破解方式&#xff0c;说实话&#xff0c;非常方便&#xff08;后续在安前码后中分享&#xff09; 破解之后呢&#xff0c;香了一阵子&#xff0c;但是突然显示开始卡顿&#xff0c;界面几乎是…

Word插件-好用的插件-PPT 素材该怎么积累-大珩助手

PPT 素材该怎么积累&#xff1f; 使用大珩助手中的素材库功能&#xff0c;将Word中的&#xff0c;或系统中的文本文件、图片、其他word文档、pdf&#xff0c;所有见到的好素材&#xff0c;一键收纳。 步骤&#xff1a;选中文件&#xff0c;按住鼠标左键拖到素材库界面中&…

【软考】信息系统项目管理师论文方向猜想

报喜不报忧&#xff0c;每天都在为鸡零狗碎推诿扯皮&#xff0c;属实是有辱师门。 通过软考&#xff0c;目前算是真正有意义的事情。 虽然都说高项的论文是个玄学&#xff0c;但是道听途说了一些通关感想还是蛮有启发的。 文件要求 参考了一份广西省高级工程师评审的文件&am…