[PyTest-案例]

article/2025/9/5 11:40:22

接口对象封装

1.requests和pymysql实现ihrm登录接口缺点 : 代码冗余度高,耦合度高,维护成本大

核心思想 : 代码分层

  • 按代码功能划分 :
    • 接口对象层 : 负责发送http请求,访问待测接口,返回响应数据
    • 测试用例层 : 调用接口,按照响应数据,断言完成测试

封装tpshop商城

  • 普通方式实现
# 导包----我的tpshop有问题,故省略url
import requests# 创建session对象
session=requests.Session()
# 发送验证码请求get
session.get(url="")
# 发送登录请求 post
resp=session.post(url="",date={"username":"12345678","password":"123456","varify":"8888"})
# 打印响应结果
print("登录结果:",resp.json())
登录接口对象层
  1. 封装思想 :
    • 将普通方式实现代码中固定不变的,直接写到方法的实现内部
    • 将动态变化的,从参数传入
    • 将响应结果,通过返回值return
  • 示例分析:

在这里插入图片描述
在这里插入图片描述

import requests# 封装tpshop商城,接口对象层
class TpshopLoginApi(object):@classmethod# 获取验证码---普通实现固定不变def get_verify(cls,session):session.get(url="http://tpshop-test.itcast.itheima.net/index.php?m=Home&c=User&a=verify")# 登录@classmethoddef login(cls,session,login_data):resp=session.post(url="http://tpshop-test.itcast.itheima.net/index.php?m=Home&c=User&a=verify",date=login_data)# 返回值return resp# 验证:测试自己封装的接口,功能是否正常
if __name__=='__main__':session=requests.Session()# 发送获取验证码请求TpshopLoginApi.get_verify(session)login_data={"username":"12345678","password":"123456","varify":"8888"}# 发送登录请求response=TpshopLoginApi.login(session,login_data)print("登录结果:",response.json())

具体实现:

1.将普通实现代码中,固定不变的和动态变换的,不变的写到参数内部,动态的通过参数传过去
2.有返回值通过return做返回
3.写测试的代码进行测试
登录接口测试用例层
  • 使用pytest框架,调用自己封装的Api发送请求,获取响应结果,断言,完成接口测试!
  • 优化前
import requestsfrom py_tpshopFZ import TpshopLoginApiclass TestTpshopLogin(object):# 定义测试方法 - 登录成功def test01_login_success(self):# 创建session实例session=requests.Session()# 调用自己封装的api,发送获取验证码请求TpshopLoginApi.get_verify(session)# 准备请求体数据req_data={"username":"12345678","password":"123456","verify":"8888"}# 调用自己封装的API,发送登录请求resp=TpshopLoginApi.login(self,req_data)# 打印响应结果:print("登录成功:",resp.json())# 断言assert 200 == resp.status_code()assert 1 == resp.json().get("status")assert "登陆成功" in resp.json().get("msg")# 定义测试方法 - 密码错误def test02_pwd_err(self):# 创建session实例session = requests.Session()# 调用自己封装的api,发送获取验证码请求TpshopLoginApi.get_verify(session)# 准备请求体数据req_data = {"username": "12345678", "password": "1234561212", "verify": "8888"}# 调用自己封装的API,发送登录请求resp = TpshopLoginApi.login(self, req_data)# 打印响应结果:print("密码错误:", resp.json())# 断言assert 200 == resp.status_code()assert -2 == resp.json().get("status")assert "密码错误" in resp.json().get("msg")# 定义测试方法 - 验证码错误def test03_verify_err(self):# 创建session实例session = requests.Session()# 调用自己封装的api,发送获取验证码请求TpshopLoginApi.get_verify(session)# 准备请求体数据req_data = {"username": "12345678", "password": "123456", "verify": "9999"}# 调用自己封装的API,发送登录请求resp = TpshopLoginApi.login(self, req_data)# 打印响应结果:print("验证码错误:", resp.json())# 断言assert 200 == resp.status_code()assert 0 == resp.json().get("status")assert "验证码错误" in resp.json().get("msg")
  • 优化后
import requestsfrom py_tpshopFZ import TpshopLoginApiclass TestTpshopLogin(object):# 定义为类属性,使用时类属性可以用类名,实例,self,cls均可引用session = None    # 每个方法调用前都执行一次def setup(self):# 创建session实例self.session=requests.Session()# 调用自己封装的API,发送获取验证码请求---索引类属性语法规范TpshopLoginApi.get_verify(self.session)# 定义测试方法 - 登录成功def test01_login_success(self):# 准备请求体数据req_data={"username":"12345678","password":"123456","verify":"8888"}# 调用自己封装的API,发送登录请求resp=TpshopLoginApi.login(self.session,req_data)# 打印响应结果:print("登录成功:",resp.json())# 断言assert 200 == resp.status_code()assert 1 == resp.json().get("status")assert "登陆成功" in resp.json().get("msg")# 定义测试方法 - 密码错误def test02_pwd_err(self):# 准备请求体数据req_data = {"username": "12345678", "password": "1234561212", "verify": "8888"}# 调用自己封装的API,发送登录请求resp = TpshopLoginApi.login(self.session, req_data)# 打印响应结果:print("密码错误:", resp.json())# 断言assert 200 == resp.status_code()assert -2 == resp.json().get("status")assert "密码错误" in resp.json().get("msg")# 定义测试方法 - 验证码错误def test03_verify_err(self):# 创建session实例session = requests.Session()# 调用自己封装的api,发送获取验证码请求TpshopLoginApi.get_verify(session)# 准备请求体数据req_data = {"username": "12345678", "password": "123456", "verify": "9999"}# 调用自己封装的API,发送登录请求resp = TpshopLoginApi.login(self.session, req_data)# 打印响应结果:print("验证码错误:", resp.json())# 断言assert 200 == resp.status_code()assert 0 == resp.json().get("status")assert "验证码错误" in resp.json().get("msg")

搭建自动基础化框架

  • 名称apiTestFramework(package存放带码,dict存放测试报告)
  • api : 定义封装被测系统的接口
  • scripts : 定义测试用例脚本
  • data : 存放测试数据文件
  • report : 存放生成的测试报告
  • common : 存放通用工具类
  • config.py : 定义项目的配置信息
  • run_suite.py : 执行测试套件的入口

封装ihrm登录

  • 准备数据—>调接口—>得响应结果---->设断言
  • 普通实现
import requestsresp=requests.post(url="https://ihrm-java.itheima.net/api/sys/login",headers={"Content-Type":"application/json"},json={"mobile": "13800000002","password": "929itheima.CN032@.20250528"})print("打印结果:",resp.json())
  • 登录接口用例层---------登录接口封装
import requests# 定义ihrm登录接口 类
class IhrmloginApi(object):# 定义方法@classmethoddef login(cls,rep_data):resp=requests.post(url="https://ihrm-java.itheima.net/api/sys/login",json=rep_data)return resp# 测试接口封装是否成功
if __name__=='__main__':data={"mobile": "13800000002","password": "929itheima.CN032@.20250528"}# 传参进行调用resp=IhrmloginApi.login(data)print("登录成功:",resp.json())
  • 登录接口测试用例层
from api.ihrm_loginapi import IhrmloginApiclass TestIhrmLogin(object):# 定义测试方法--登录成功def test_01_login_success(self):# 准备登录测试数据data = {"mobile": "13800000002","password": "929itheima.CN032@.20250528"}# 调用自己封装的api 获取响应结果resp=IhrmloginApi.login(data)# 打印查看print("登陆成功:",resp.json())# 断言assert 200 == resp.status_codeassert True == resp.json().get('success')assert 10000 == resp.json().get("code")assert '操作成功' in resp.json().get('message')# 定义测试方法--手机号未注册def test_02_mpbile(self):# 准备登录测试数据data = {"mobile": "13800000001", "password": "929itheima.CN032@.20250528"}# 调用自己封装的api 获取响应结果resp = IhrmloginApi.login(data)# 打印查看print("手机号未注册:", resp.json())# 断言assert 200 == resp.status_codeassert False == resp.json().get('success')assert 20001 == resp.json().get("code")assert '用户名或密码错误' in resp.json().get('message')# 定义测试方法--手机号未注册def test_02_pwd_err(self):# 准备登录测试数据data = {"mobile": "13800000002", "password": "itheima.CN032@.20250528"}# 调用自己封装的api 获取响应结果resp = IhrmloginApi.login(data)# 打印查看print("密码错误:", resp.json())# 断言assert 200 == resp.status_codeassert False == resp.json().get('success')assert 20001 == resp.json().get("code")assert '用户名或密码错误' in resp.json().get('message')
  • 封装断言方法:
  1. 在项目目录下, 创建common目录,添加文件assert_tools.py
  2. 在文件assert_tools.py添加通用的断言函数common_assert()
  3. 实现此函数
  4. 在测试脚本层,需要断言的位置,使用这个断言函数来实现
    在这里插入图片描述
"""定义通用的工具函数 实现断言"""# 谁调用就给一个resp
def common_assert(resp,status_code,success,code,message):assert status_code == resp.status_codeassert success is resp.json().get("success")assert code == resp.json().get("code")assert message in resp.json().get("message")
  • 加入断言优化后
from api.ihrm_loginapi import IhrmloginApi
from common.assert_tools import common_assertclass TestIhrmLogin(object):# 定义测试方法--登录成功def test_01_login_success(self):# 准备登录测试数据data = {"mobile": "13800000002","password": "929itheima.CN032@.20250529"}# 调用自己封装的api 获取响应结果resp=IhrmloginApi.login(data)# 打印查看print("登陆成功:",resp.json())# 断言common_assert(resp, 200, True, 10000, "操作成功")# 定义测试方法--手机号未注册def test_02_mobile(self):# 准备登录测试数据data = {"mobile": "13800000001", "password": "929itheima.CN032@.20250529"}# 调用自己封装的api 获取响应结果resp = IhrmloginApi.login(data)# 打印查看print("手机号未注册:", resp.json())# 断言common_assert(resp, 200, False, 20001, "用户名或密码错误")# 定义测试方法--密码错误def test_03_pwd_err(self):# 准备登录测试数据data = {"mobile": "13800000002", "password": "itheima.CN032@.20250529"}# 调用自己封装的api 获取响应结果resp = IhrmloginApi.login(data)# 打印查看print("密码错误:", resp.json())# 断言common_assert(resp, 200, False, 20001, "用户名或密码错误")# 电话号码为空def test_04_mobile_is_none(self):# 准备登录测试数据data = {"mobile": "", "password": "929itheima.CN032@.20250529"}# 调用自己封装的api 获取响应结果resp = IhrmloginApi.login(data)# 打印查看print("手机号为空:", resp.json())# 断言common_assert(resp, 200, False, 20001, "用户名或密码错误")# 电话有特殊字符def test_05_mobile_have_special_char(self):# 准备登录测试数据data = {"mobile": "1213#######", "password": "929itheima.CN032@.20250529"}# 调用自己封装的api 获取响应结果resp = IhrmloginApi.login(data)# 打印查看print("手机号有特殊字符:", resp.json())# 断言common_assert(resp, 200, False, 20001, "用户名或密码错误")# 10位电话号码def test_06_mobile10(self):# 准备登录测试数据data = {"mobile": "", "password": "929itheima.CN032@.20250529"}# 调用自己封装的api 获取响应结果resp = IhrmloginApi.login(data)# 打印查看print("10位手机号", resp.json())# 断言common_assert(resp, 200, False, 20001, "用户名或密码错误")# 12位电话号码def test_07_mobile12(self):# 准备登录测试数据data = {"mobile": "111111111111", "password": "929itheima.CN032@.20250529"}# 调用自己封装的api 获取响应结果resp = IhrmloginApi.login(data)# 打印查看print("12位手机号:", resp.json())# 断言common_assert(resp, 200, False, 20001, "用户名或密码错误")# 密码为空def test_08_pwd_empty(self):# 准备登录测试数据data = {"mobile": "13800000002", "password": None}# 调用自己封装的api 获取响应结果resp = IhrmloginApi.login(data)# 打印查看print("12位手机号:", resp.json())# 断言common_assert(resp, 200, False, 20001, "用户名或密码错误")# 密码有特殊字符def test_09_pwd_have_special_char(self):# 准备登录测试数据data = {"mobile": "13800000002", "password": "929@@@@@"}# 调用自己封装的api 获取响应结果resp = IhrmloginApi.login(data)# 打印查看print("密码有特殊字符:", resp.json())# 断言common_assert(resp, 200, False, 20001, "用户名或密码错误")# 密码1位def test_10_pwd01(self):def test_09_pwd_have_special_char(self):# 准备登录测试数据data = {"mobile": "13800000002", "password": "9"}# 调用自己封装的api 获取响应结果resp = IhrmloginApi.login(data)# 打印查看print("密码1位:", resp.json())# 断言common_assert(resp, 200, False, 20001, "用户名或密码错误")# 密码100位def test_11_pwd100(self):def test_09_pwd_have_special_char(self):# 准备登录测试数据data = {"mobile": "13800000002","password": "929@@@@@@@929@@@@@@@929@@@@@@@929@@@@@@@929@@@@@@@929@@@@@@@929@@@@@@@929@@@@@@@929@@@@@@@"}# 调用自己封装的api 获取响应结果resp = IhrmloginApi.login(data)# 打印查看print("100位密码:", resp.json())# 断言common_assert(resp, 200, False, 20001, "用户名或密码错误")# 多参数def test_12_more_params(self):# 准备登录测试数据data = {"mobile": "13800000002", "password": "929itheima.CN032@.20250529","ara":"demo"}# 调用自己封装的api 获取响应结果resp = IhrmloginApi.login(data)# 打印查看print("登陆成功:", resp.json())# 断言common_assert(resp, 200, True, 10000, "操作成功")# 少参def test_13_less_params(self):# 准备登录测试数据data = {"mobile": "13800000002"}# 调用自己封装的api 获取响应结果resp = IhrmloginApi.login(data)# 打印查看print("少参password:", resp.json())# 断言common_assert(resp, 200, False, 20001, "用户名或密码错误")# 无参def test_14_no_params(self):# 准备登录测试数据data = None# 调用自己封装的api 获取响应结果resp = IhrmloginApi.login(None)# 打印查看print("少参password:", resp.json())# 断言common_assert(resp, 200, False, 99999, "抱歉")# 错误参数def test_15_err_params(self):# 准备登录测试数据data = {"abc": "13800000002","password": "929itheima.CN032@.20250529"}# 调用自己封装的api 获取响应结果resp = IhrmloginApi.login(data)# 打印查看print("无参:", resp.json())# 断言common_assert(resp, 200, False, 20001, "用户名或密码错误")
  • data有相似的值,把登录接口参数化写入json文件
[{"desc": "登录成功","req_date":{"mobile": "13800000002","password": "929itheima.CN032@.20250529"},"status_code":200,"success": true,"code": 10000,"message": "操作成功!"},{"desc": "手机号不存在","req_date":{"mobile": "13800000001","password": "929itheima.CN032@.20250529"},"status_code":200,"success": false,"code": 20001,"message": "用户名或密码错误"},{"desc":"密码错误","req_date":{"mobile": "13800000002","password": "itheima.CN032@.20250529"},"status_code":200,"success": false,"code": 20001,"message": "用户名或密码错误"},{"desc": "手机号为空","req_date":{"mobile": null,"password": "929itheima.CN032@.20250529"},"status_code": 200,"success": false,"code": 20001,"message": "用户名或密码错误"},{"desc": "手机号特殊字符","req_data": {"mobile": "13@######","password": "929itheima.CN032@.20250529"},"status_code": 200,"success": false,"code": 20001,"message": "用户名或密码错误"},{"desc": "10位手机号","req_data": {"mobile": "1323333333","password": "929itheima.CN032@.20250529"},"status_code": 200,"success": false,"code": 20001,"message": "用户名或密码错误"},{"desc": "12位手机号","req_data": {"mobile": "132333333322","password": "929itheima.CN032@.20250529"},"status_code": 200,"success": false,"code": 20001,"message": "用户名或密码错误"},{"desc": "密码为空","req_data": {"mobile": "1323333333","password": null},"status_code": 200,"success": false,"code": 20001,"message": "用户名或密码错误"},{"desc": "密码有特殊字符","req_data": {"mobile": "13800000002", "password": "929@@@@@"},"status_code": 200,"success": false,"code": 20001,"message": "用户名或密码错误"},{"desc": "密码1位","req_data": {"mobile": "13800000002", "password": "9"},"status_code": 200,"success": false,"code": 20001,"message": "用户名或密码错误"},{"desc": "密码100位","req_data": {"mobile": "13800000002", "password": "929@@@@@@@929@@@@@@@929@@@@@@@929@@@@@@@929@@@@@@@929@@@@@@@929@@@@@@@929@@@@@@@929@@@@@@@"},"status_code": 200,"success": false,"code": 20001,"message": "用户名或密码错误"},{"desc": "多参","req_data": {"mobile": "13800000002", "password": "929itheima.CN032@.20250529","ara":"demo"},"status_code": 200,"success": false,"code": 20001,"message": "用户名或密码错误"},{"desc": "少参","req_data": {"mobile": "13800000002"},"status_code": 200,"success": false,"code": 20001,"message": "用户名或密码错误"},{"desc": "无参","req_data":null,"status_code": 200,"success": false,"code": 20001,"message": "抱歉"},{"desc": "错误参数","req_data": {"mobile": "13800000002", "pass": "929itheima.CN032@.20250529"},"status_code": 200,"success": false,"code": 20001,"message": "用户名或密码错误"}
]

参数化

参数化的核心 : 数据驱动(用数据驱动测试用例执行)

  • 数据驱动 : 针对一个接口,只写一个测试方法,用一份测试数据文件,管理各个测试用例的测试数据.
  • 如登录接口:15条测试用例,只需要一个测试方法,对应一个含有15组数据的json文件,进行测试

回顾PyTest

  • 加法函数

http://www.hkcw.cn/article/PwHGnKKotc.shtml

相关文章

25 字符数组与字符串及多维数组详解:定义与初始化、访问与遍历、%s 格式符、内存剖析、编程实战

1 字符数组与字符串 1.1 字符数组 字符数组是 C 语言中用于存储一系列字符的基本数据结构。其定义方式与其他类型的数组类似,使用 char 类型来指定数组的元素类型。例如: char arr[10]; // 定义一个可存储 10 个字符的数组 此数组 arr 能够存储 10 个字…

IEEE旗下2区所有SCI汇总!

本期小编统计了【IEEE旗下】2区所有期刊的最新影响因子,分区、年发文量以及投稿经验,供大家参考! 1 IEEE Journal of Selected Topics in Applied Earth Observations and Remote Sensing 【影响因子】4.7 【期刊分区】JCR1区,中…

论文略读: STREAMLINING REDUNDANT LAYERS TO COMPRESS LARGE LANGUAGE MODELS

2025 ICLR 判断模型层的重要性->剪去不重要的层(用轻量网络代替) 这种方法只减少了层数量,所以可以用常用的方法加载模型 层剪枝阶段 通过输入与输出的余弦相似度来判断各个层的重要性 具有高余弦相似度的层倾向于聚集在一起&#xff0c…

Geoserver修行记--点击geoserver服务的WMTS能力(GetCapabilities)文档显示400 null

项目场景 在进行geoserver的项目部署过程中再次遇到的问题,这里再记录一下: 我在前端调用WMTS服务的时候,我需要查看图层的能力文档,点击下图出现如下所示的情况: 点击以后出现 400:null,如下…

Codeforces Round 1027 (Div. 3)(A-E)

题面链接&#xff1a;Dashboard - Codeforces Round 1027 (Div. 3) - Codeforces A. Square Year 思路 先看数字能否被开方&#xff0c;如果能输出0 即可 代码 #include<bits/stdc.h> using namespace std;#define vcoistnt ios_base::sync_with_stdio(false); ci…

在MDK中自动部署LVGL,在stm32f407ZGT6移植LVGL-8.3,运行demo,显示label

在MDK中自动部署LVGL&#xff0c;在stm32f407ZGT6移植LVGL-8.3 一、硬件平台二、实现功能三、移植步骤1、下载LVGL-8.42、MDK中安装LVGL-8.43、配置RTE4、配置头文件 lv_conf_cmsis.h5、配置lv_port_disp_template 四、添加心跳相关文件1、在STM32CubeMX中配置TIM7的参数2、使能…

数据结构数组总结

给定一个含有 n 个正整数的数组和一个正整数 target 。 找出该数组中满足其和 ≥ target 的长度最小的 连续子数组 [numsl, numsl1, ..., numsr-1, numsr] &#xff0c;并返回其长度。如果不存在符合条件的子数组&#xff0c;返回 0 。 示例 1&#xff1a; 输入&#xff1a;…

集成均衡功能电池保护芯片在大功率移动电源的应用,创芯微CM1341-DAT、杰华特JW3312、赛微微电CW1244、中颖SH366006

一文了解集成均衡功能电池保护IC在大功率移动电源的应用 创芯微CM1341-DAT 创芯微CM1341-DAT是一款专用于4串锂离子/磷酸铁锂电池的保护芯片&#xff0c;内置有高精度电压检测电路和电流检测电路。通过检测各节电池的电压、充放电电流及温度等信息&#xff0c;实现电池过充电…

2025年机电一体化与自动化系统国际会议(ICMAS 2025)

2025年机电一体化与自动化系统国际会议&#xff08;ICMAS 2025&#xff09; 2025 International Conference on Mechatronics and Automation Systems 一、大会信息 会议简称&#xff1a;ICMAS 2025 大会地点&#xff1a;中国南京 审稿通知&#xff1a;投稿后2-3日内通知 投稿…

Linux多线程(四)之Linux线程控制2【线程等待和线程终止】

文章目录 线程等待线程终止return退出线程pthread_exit退出线程pthread_cancel取消线程 线程等待 哪个线程先运行&#xff1f;由调度器说的算。 但是主线程一定要是最后退出&#xff01; 为什么需要线程等待&#xff1f; 已经退出的线程&#xff0c;其空间没有被释放&#…

Python自动化之selenium语句——浏览器设置显示尺寸、截图、刷新网页

目录 一、浏览器设置最大化、最小化 1.浏览器最大化 2.浏览器最小化 二、浏览器打开的位置、尺寸 1.浏览器打开位置 2.浏览器打开尺寸 三、浏览器截图 1.截图语句 2.运行成功后查看 四、刷新网页 上一节实现了打开浏览器、打开指定网址、关闭浏览器的操作&#xff0c…

Selenium的第三天打卡——Selenium应用(selenium元素选取以及浏览器操作)

Selenium 4 应用示例解析 目录 Selenium 4 应用示例解析 一、基本配置 二、浏览器设置&#xff08;这之前没有html基础的朋友可以先去了解一下html哦&#xff09; 三、元素定位方法 1.Selenium 4 元素的选中 ①开发者模式 ②观察代码 2.Selenium 4 元素抓取的核心内容…

智能仓储落地:机器人如何通过自动化减少仓库操作失误?

仓库作业的速度和准确性至关重要&#xff0c;尤其是在当前对无差错、高效作业的要求达到前所未有的环境下。每一个错误&#xff0c;无论是物品放错位置还是库存差异&#xff0c;都会在供应链中产生连锁反应&#xff0c;造成延误、增加成本&#xff0c;并最终影响客户满意度。 …

【Linux系统移植】Cortex-A8 Linux系统移植(超详细)

目录 前言 一、ARM开发板ARM简介RISC和CISCARM产品分布核心板S5pv210 SOC嵌入式系统开发方式 二、嵌入式系统组成为什么要系统移植内核移植框图 三、嵌入式开发环境搭建搭建开发环境总流程设置ubuntu与windows共享目录修改用户为root用户安装NFS服务器安装tftp服务器安装交叉编…

如何通过数据分析优化项目决策

通过数据分析优化项目决策需从明确数据分析目标、选择适当的数据分析工具、确保数据质量、建立数据驱动文化等方面入手&#xff0c;其中&#xff0c;明确数据分析目标是优化决策过程的基础&#xff0c;只有清晰明确的数据分析目标才能指导有效的数据采集与分析&#xff0c;避免…

纯前端实现图片伪3D视差效果

作者&#xff1a;vivo 互联网前端团队- Su Ning 本文通过depth-anything获取图片的深度图&#xff0c;同时基于pixi.js&#xff0c;通过着色器编程&#xff0c;实现了通过深度图驱动的伪3D效果。该方案支持鼠标/手势与手机陀螺仪双模式交互&#xff0c;在保证性能的同时&#x…

在ROS2(humble)+Gazebo+rqt下,实时显示仿真无人机的相机图像

文章目录 前言一、版本检查检查ROS2版本 二、步骤1.下载对应版本的PX4(1)检查PX4版本(2)修改文件名(3)下载正确的PX4版本 2.下载对应版本的Gazebo(1)检查Gazebo版本(2)卸载不正确的Gazebo版本(3)下载正确的Gazebo版本 3.安装bridge包4.启动 总结 前言 在ROS2的环境下&#xff…

Git的三种合并方式

在 Gitee&#xff08;码云&#xff09;中合并分支主要有三种方式&#xff1a;​普通合并&#xff08;Merge Commit&#xff09;、压缩合并&#xff08;Squash Merge&#xff09;​和变基合并&#xff08;Rebase Merge&#xff09;​。每种方式适用于不同的场景&#xff0c;各有…

微机系统-汇编语言入门

汇编语言及其程序设计 -1:汇编语言的语句格式 [名字] 操作符 操作数&#xff1b; [注释] 名字: 1. 是一种标识符。 2. 组成&#xff1a;A&#xff5e;Z&#xff0c;a&#xff5e;z&#xff1b; 0&#xff5e;9&#xff1b; 专用符号 &#xff1f;&#xff0e; _$ 3. 限制&…

互斥锁、自旋锁、读写锁、悲观锁、乐观锁的应用场景

一&#xff1a;并发 1.1MySQL并发事务访问相同记录 &#xff08;1&#xff09;读-读 不影响 &#xff08;2&#xff09;写-写 写的数据需要一个一个来&#xff0c;排队执行 &#xff08;3&#xff09;读-写 两次读…