UI自动化测试示例:python+pytest+selenium+allure
重点应用是封装、参数化:
比如在lib文件夹下,要存储封装好的方法和必要的环境变量(指网址等)
1.cfg.py:封装网址和对应的页面
SMP_ADDRESS = 'http://127.0.0.1:8234'SMP_URL_LOGIN = f'{SMP_ADDRESS}/login.html'
SMP_URL_DEVICE_MODEL = f'{SMP_ADDRESS}/index.html#/devicemodel'
SMP_URL_SERVICE_RULE = f'{SMP_ADDRESS}/index.html#/svcrule'
二、登录部分
2.1 lib.py-登录模块的封装
主要就是输入账号名和密码进行测试。
from selenium import webdriver
from selenium.webdriver.common.by import By
from selenium.webdriver.support.ui import Select
import time
from lib.cfg import *class SMP_UI:def __init__(self):options = webdriver.ChromeOptions()#排除一些不需要的日志输出。options.add_experimental_option( 'excludeSwitches', ['enable-logging'])self.wd = webdriver.Chrome(options=options)self.wd.implicitly_wait(3) #隐式等待时间为5秒,等待元素加载完成#登录模块测试def login(self, username, password):self.wd.get(SMP_URL_LOGIN) #打开登录网址#time.sleep(1)#sleep方便观察if username is not None:self.wd.find_element(By.ID, 'username').send_keys(username)if password is not None:self.wd.find_element(By.ID, 'password').send_keys(password)self.wd.find_element(By.ID, 'loginBtn').click()#点击登录按钮smpUI = SMP_UI()
2.2自动化测试脚本代码-登录
from selenium.webdriver.common.by import By
import time,pytest
from lib.webUI_smp import smpUIdef test_SMP_login_001():#正确账号和密码smpUI.login('byhy','sdfsdf' )nav = smpUI.wd.find_elements(By.TAG_NAME, 'nav') #登陆后就可以查看nav了,有信息代表登陆成功assert nav != []@pytest.fixture
def clearAlert():yieldtry:smpUI.wd.switch_to.alert.accept()except Exception as e:print(e)@pytest.mark.parametrize('username, password, expectedalert', [(None, 'sdfsdf', '请输入用户名'),#空用户名('byhy', None, '请输入密码'),#空密码('byhy', 'sdfsd', '登录失败: 用户名或者密码错误'),#少输入密码('byhy', 'sdfsdff', '登录失败: 用户名或者密码错误'),#多输入密码('byh', 'sdfsdf', '登录失败: 用户名不存在'),#少输入用户名('byhyy', 'sdfsdf', '登录失败: 用户名不存在'),#多输入用户名])
def test_SMP_login_002(username, password, expectedalert, clearAlert):smpUI.login(username, password)smpUI.wd.implicitly_wait(5)#隐式等待alert = smpUI.wd.switch_to.alert.textassert alert == expectedalert #对弹出框进行断言对比
三、商品部分
3.1 lib.py-商品服务模块的封装
主要就是添加商品、删除商品等,输出第一页的第一条的值(检查返回值)
#添加设备模块检测def add_device_model(self, devType, name, desc):# 实例化选择框做为对象select = Select(smpUI.wd.find_element(By.ID, "device-type"))# 选择对应列表中的种类select.select_by_visible_text(devType)#清除填充框后填-设备型号ele = smpUI.wd.find_element(By.ID, 'device-model')ele.clear()ele.send_keys(name)#清除填充框后填-型号描述ele = smpUI.wd.find_element(By.ID, 'device-model-desc')ele.clear()ele.send_keys(desc)#点提交smpUI.wd.find_element(By.CSS_SELECTOR, '.add-one-submit-btn-div .btn').click()# time.sleep(1)#得到第一页的设备信息(注:一页五条),是显示第一页的第一个数据def get_first_page_device_models(self):time.sleep(1)self.wd.implicitly_wait(0)values = self.wd.find_elements(By.CSS_SELECTOR,'.field-value')self.wd.implicitly_wait(10)deviceModels = []for idx, value in enumerate(values):if (idx+1) % 3 == 0:deviceModels.append([values[idx-2].text, values[idx-1].text, values[idx].text])return deviceModels#删除第一个项目def del_first_item(self) -> bool:self.wd.implicitly_wait(0)delBtn = self.wd.find_elements(By.CSS_SELECTOR,'.result-list-item:first-child .result-list-item-btn-bar span:first-child')self.wd.implicitly_wait(10)if not delBtn:return FalsedelBtn[0].click()self.wd.switch_to.alert.accept()return True
3.2自动化测试脚本代码-商品服务
之间登录商品增删改查的界面。注意编完数据后要删除
from selenium.webdriver.common.by import By
import time,pytest
from lib.webUI_smp import smpUI
from selenium.webdriver.support.ui import Select
from selenium.webdriver.support.ui import WebDriverWait
from selenium.webdriver.support import expected_conditions as EC
from lib.cfg import *@pytest.fixture(scope='module')
def inDeviceModelMgr():print('inDeviceModelMgr setup')smpUI.login('byhy','sdfsdf' ) #直接登录smpUI.wd.get(SMP_URL_DEVICE_MODEL)#登录后转到SMP_URL_DEVICE_MODEL这个主界面yield@pytest.fixture()
def delAddedDeviceModel():yieldprint('删除添加的设备型号')smpUI.del_first_item()@pytest.mark.parametrize('type, device, desc', [("存储柜", 'hebut-cn-box-01', '河北工业大学菜鸟柜子01柜-10大20中40小'),#添加设备-存储柜测试("存储柜", '河'*100, '河北工业大学菜鸟柜子02柜-10大20中40小'),#测试名称最大长度
("电瓶车充电站", 'hebut-dpccdz-es-01', '河北工业大学北辰充电站01站-220v电瓶车充电站'),#添加设备-电瓶车充电站
("洗车站", 'hebut-xcz-washcar-01', '河北工业大学洗车站东站'),#添加设备-洗车站测试
("汽车充电站", 'hebut-powercar-01', '河北工业大学7千瓦汽车充电站01'),#添加设备-汽车充电站])
#添加设备-存储柜测试
def test_SMP_device_model_001(type,device,desc,inDeviceModelMgr, delAddedDeviceModel):# 点击添加按钮topBtn = smpUI.wd.find_element(By.CSS_SELECTOR, '.add-one-area > span')if topBtn.text == '添加':topBtn.click()smpUI.add_device_model(type,device,desc)dms = smpUI.get_first_page_device_models()assert dms == [[type,device,desc ]]smpUI.wd.implicitly_wait(3)def test_SMP_device_model_501(inDeviceModelMgr, delAddedDeviceModel):# 点击添加按钮topBtn = smpUI.wd.find_element(By.CSS_SELECTOR, '.add-one-area > span')if topBtn.text == '添加':topBtn.click()smpUI.add_device_model("存储柜", 'daixiugai-01', '待修改测试数据-01')smpUI.wd.implicitly_wait(3)# 点击修改按钮changeBtn = smpUI.wd.find_element(By.XPATH,'/html/body/main/div[3]/div/div[2]/span[2]').click()# if changeBtn.text == '修改':# changeBtn.click()# 实例化选择框做为对象select = Select(smpUI.wd.find_element(By.XPATH,'/html/body/main/div[3]/div/div[1]/div[1]/select'))# 选择对应列表中的种类select.select_by_visible_text("洗车站")# 清除填充框后填-设备型号ele = smpUI.wd.find_element(By.XPATH,'/html/body/main/div[3]/div/div[1]/div[2]/input')ele.clear()ele.send_keys('修改后的型号1')# 清除填充框后填-型号描述ele = smpUI.wd.find_element(By.XPATH,'/html/body/main/div[3]/div/div[1]/div[3]/input')ele.clear()ele.send_keys('修改后的型号描述')# 点提交smpUI.wd.find_element(By.XPATH,'/html/body/main/div[3]/div/div[2]/span[1]').click()dms = smpUI.get_first_page_device_models()assert dms == [["洗车站",'修改后的型号1','修改后的型号描述' ]]smpUI.wd.implicitly_wait(3)def test_SMP_device_model_601(inDeviceModelMgr, delAddedDeviceModel):# 点击添加按钮topBtn = smpUI.wd.find_element(By.CSS_SELECTOR, '.add-one-area > span')if topBtn.text == '添加':topBtn.click()smpUI.add_device_model("存储柜", 'daixshanchu-01', '待删除测试数据-01')time.sleep(1)smpUI.del_first_item()dms = smpUI.get_first_page_device_models()assert ['存储柜', 'daixshanchu-01', '待删除测试数据-01'] not in dmssmpUI.wd.implicitly_wait(3)
四、业务规则部分
4.1 lib.py-业务规则模块的封装
def add_svc_rule(self, ruleName:str, ruleType:str,minFee:str,estFee:str, feeRate=None, desc:str=None):"""添加业务规则:param ruleName: 规则名称:param ruleType: 规则类型,只能是:后付费-上报业务量、预付费-下发费用、预付费-下发业务量:param minFee: 最小费用,不需要时,填写空字符串:param estFee: 预计费用,不需要时,填写空字符串:param desc: 描述:param feeRate: 费率, 如果ruleType是预付费-下发费用: 不用填写预付费-下发业务量: 格式为 ['千瓦时', '1'], 元素分别是 单位、单价后付费-上报业务量: 格式为 [['10L', '小时','1'],['20L', '小时','2'],], 每个元素里面分别是 : 业务码、单位、单价:return:"""#名称输入ele = self.wd.find_element(By.CSS_SELECTOR, '.add-one-form > .field:nth-child(1) >input')ele.clear()ele.send_keys(ruleName)select = Select(self.wd.find_element(By.CSS_SELECTOR, ".add-one-form select"))select.select_by_visible_text(ruleType)if ruleType != '后付费-上报业务量': #预付费-下发业务量操作ele = self.wd.find_element(By.CSS_SELECTOR, '.add-one-form > .field:nth-child(3) >input')ele.clear()if minFee:ele.send_keys(minFee)#输入最小消费ele = self.wd.find_element(By.CSS_SELECTOR, '.add-one-form > .field:nth-child(4) >input')ele.clear()if estFee:ele.send_keys(estFee)#输入预估消费# 三者都有描述, 用xpath而不用 .field:nth-child 因为后付费-上报业务量 次序会变if desc:ele = self.wd.find_element(By.XPATH,"//*[@class='add-one-submit-btn-div']/preceding-sibling::*[1]/input")ele.clear()ele.send_keys(desc)#输入对应描述# 费率填写if ruleType == '预付费-下发费用':# 没有费率设置passelif ruleType == '后付费-上报业务量':# 先删除上次添加遗留的费率self.wd.implicitly_wait(3) # 先修改短等待时间while True:eles = self.wd.find_elements(By.CSS_SELECTOR, '.fee-rate span:last-child')if eles:eles[0].click() #删除之前的费率time.sleep(0.5)else:breakself.wd.implicitly_wait(3) # 再改回来for one in feeRate:#遍历每个费率块self.wd.find_element(By.CSS_SELECTOR,'.fee-rate-list button').click()#锚定整个框entry = self.wd.find_element(By.CSS_SELECTOR,'div.fee-rate:nth-last-child(2)')# 输入业务码entry.find_element(By.CSS_SELECTOR, 'input:nth-of-type(1)').send_keys(one[0])# 输入计费单位entry.find_element(By.CSS_SELECTOR, 'input:nth-of-type(2)').send_keys(one[1])# 输入单位价格entry.find_element(By.CSS_SELECTOR, 'input:nth-of-type(3)').send_keys(one[2])elif ruleType == '预付费-下发业务量':#锚定整个框entry = self.wd.find_element(By.CSS_SELECTOR, 'div.fee-rate')# 输入计费单位entry.find_element(By.CSS_SELECTOR, 'input:nth-of-type(1)').send_keys(feeRate[0])# 输入单位价格entry.find_element(By.CSS_SELECTOR, 'input:nth-of-type(2)').send_keys(feeRate[1])else:raise Exception('ruleType 参数值错误')# 确定提交self.wd.find_element(By.CSS_SELECTOR, '.add-one-submit-btn-div .btn').click()self.wd.implicitly_wait(3)#业务部分查询首页def get_first_page_svc_rules(self):time.sleep(1)self.wd.implicitly_wait(0)items = self.wd.find_elements(By.CSS_SELECTOR,'.result-list-item-info')self.wd.implicitly_wait(10)rules = []for item in items:nameValueList = item.find_elements(By.CSS_SELECTOR, '.field>.field-name, .field>.field-value')itemInfo = []for idx, _ in enumerate(nameValueList):if (idx+1) % 2 == 0:nameEle,valueEle = nameValueList[idx-1], nameValueList[idx]if nameEle.text == '规则内容':ruleContent = {}sfns = valueEle.find_elements(By.CSS_SELECTOR,'.sub-field-name')for sfnEle in sfns:sfn = sfnEle.textsfvEle = sfnEle.find_element(By.XPATH, "following-sibling::*[1]")if sfn == '费率':ruleContent[sfn] = sfvEle.textelse:ruleContent[sfn] = sfvEle.textitemInfo.append(ruleContent)else:itemInfo.append(valueEle.text)rules.append(itemInfo)return rules#删除第一个项目def del_first_item(self) -> bool:self.wd.implicitly_wait(0)delBtn = self.wd.find_elements(By.CSS_SELECTOR,'.result-list-item:first-child .result-list-item-btn-bar span:first-child')self.wd.implicitly_wait(10)if not delBtn:return FalsedelBtn[0].click()self.wd.switch_to.alert.accept()return True
4.2自动化测试脚本代码-业务规则
注意事项跟3.2的商品的自动化测试脚本类似
from selenium.webdriver.common.by import By
import time,pytest
from lib.webUI_smp import smpUI
from lib.cfg import *@pytest.fixture(scope='module')
def inServiceRuleMgr():print('** inServiceRuleMgr setup **')smpUI.login('byhy','sdfsdf' )smpUI.wd.get(SMP_URL_SERVICE_RULE)yield@pytest.fixture()
def delAddedServiceRule():yieldprint('** 删除添加的业务规则')smpUI.del_first_item()@pytest.mark.parametrize('svcname, svctype,paymin,pay, svcdesc,jifeidanwei,feilv', [("全国-电瓶车充电费率1", "预付费-下发业务量", "0.1","2","全国-电瓶车充电费率1的描述",['千瓦时', '1'],None),#添加预付费-下发业务量])
def test_SMP_service_rule_001(svcname, svctype,paymin,pay, svcdesc,jifeidanwei,feilv,inServiceRuleMgr, delAddedServiceRule):# 点击添加按钮topBtn = smpUI.wd.find_element(By.CSS_SELECTOR,'.add-one-area > span')if topBtn.text == '添加':topBtn.click()smpUI.add_svc_rule(svcname, svctype,paymin,pay, jifeidanwei,svcdesc)dms = smpUI.get_first_page_svc_rules()assert dms[0][:3] == [svcname, svctype, {'最小消费': paymin, '预估消费': pay, '费率': f'单位:{jifeidanwei[0]} \n单价:{jifeidanwei[1]}'}]@pytest.mark.parametrize('svcname, svctype,paymin,pay, svcdesc,jifeidanwei,feilv', [("南京-洗车机费率1", "预付费-下发费用", "2","10","南京-洗车机费率1的描述",None,None),#预付费-下发费用])
def test_SMP_service_rule_101(svcname, svctype,paymin,pay, svcdesc,jifeidanwei,feilv,inServiceRuleMgr, delAddedServiceRule):# 点击添加按钮topBtn = smpUI.wd.find_element(By.CSS_SELECTOR,'.add-one-area > span')if topBtn.text == '添加':topBtn.click()smpUI.add_svc_rule(svcname, svctype,paymin,pay,svcdesc)dms = smpUI.get_first_page_svc_rules()assert dms[0][:3] == [svcname, svctype, {'最小消费': paymin, '预估消费': pay}]@pytest.mark.parametrize('svcname, svctype,paymin,pay, svcdesc,jifeidanwei,feilv', [("南京-存储柜费率1", "预付费-下发业务量", None,None,None,None,[['100L','小时','2']]),#添加预付费-下发业务量,['50L','小时','1'],['10L','小时','0.5']])
def test_SMP_service_rule_201(svcname, svctype,paymin,pay, svcdesc,jifeidanwei,feilv,inServiceRuleMgr, delAddedServiceRule):# 点击添加按钮topBtn = smpUI.wd.find_element(By.CSS_SELECTOR,'.add-one-area > span')if topBtn.text == '添加':topBtn.click()smpUI.add_svc_rule(svcname, svctype, paymin, pay, svcdesc, feilv)dms = smpUI.get_first_page_svc_rules()assert dms[0][:3] == [svcname, svctype, {'业务码': f'单位:{feilv[0]} \n计费单位:{feilv[1]} \n单位价格:{feilv[2]}'}]
还有部分没粘上去,万变不离其宗,还要接口自动化测试,这里先不写,下次再写。
相关文章:
UI自动化测试示例:python+pytest+selenium+allure
重点应用是封装、参数化: 比如在lib文件夹下,要存储封装好的方法和必要的环境变量(指网址等) 1.cfg.py:封装网址和对应的页面 SMP_ADDRESS http://127.0.0.1:8234SMP_URL_LOGIN f{SMP_ADDRESS}/login.html SMP_URL_DE…...
C/C++ 编程小工具
编写了 tools.h 和 tools.cpp,用于 Debug、性能测试、打印日志。 tools.h #ifndef TOOLS_H #define TOOLS_H#include <time.h> #include <fstream> #include <iostream> #include <random> #include <chrono> #include <vector&…...
第四十二章 使用 WS-ReliableMessaging
文章目录 第四十二章 使用 WS-ReliableMessaging从 Web 客户端发送一系列消息 第四十二章 使用 WS-ReliableMessaging IRIS 支持 WS-ReliableMessaging 规范的部分内容,如简介中所述。此规范提供了一种按顺序可靠地传递一系列消息的机制。本页介绍如何手动使用可靠…...
利士策分享,婚姻为何被称为大事?
利士策分享,婚姻为何被称为大事? 在历史的长河中,婚姻一直被视为人生中的头等大事,这一观念跨越时空,深深植根于各种文化和社会结构中。 古人为何将婚姻称为“大事”,这背后蕴含着丰富的社会、文化和心理寓…...
malloc源码分析之 ----- 你想要啥chunk
文章目录 malloc源码分析之 ----- 你想要啥chunktcachefastbinsmall binunsorted binbin处理top malloc源码分析之 ----- 你想要啥chunk tcache malloc源码,这里以glibc-2.29为例: void * __libc_malloc (size_t bytes) {mstate ar_ptr;void *victim;vo…...
软考系统分析师知识点五:数据通信与计算机网络
前言 今年报考了11月份的软考高级:系统分析师。 考试时间为:11月9日。 倒计时:32天。 目标:优先应试,其次学习,再次实践。 复习计划第一阶段:扫平基础知识点,仅抽取有用信息&am…...
windows客户端SSH连接ubuntu/linux服务器,三种网络连接:局域网,内网穿透(sakuraftp),虚拟局域网(zerotier)
windows客户端SSH连接ubuntu/linux服务器,三种网络连接:局域网,内网穿透(sakuraftp),虚拟局域网(zerotier) 目录 SSH简述、三种网络连接特点SSH简述局域网内连接内网穿透(…...
Python 工具库每日推荐【openpyxl 】
文章目录 引言Python Excel 处理库的重要性今日推荐:openpyxl 工具库主要功能:使用场景:安装与配置快速上手示例代码代码解释实际应用案例案例:自动生成月度销售报告案例分析高级特性条件格式数据验证扩展阅读与资源优缺点分析优点:缺点:总结【 已更新完 TypeScript 设计…...
本地生活服务项目入局方案解析!本地生活服务商系统能实现怎样的作业效果?
当前,各大平台的本地生活服务业务日渐兴盛,提高创业者入局意向的同时,也让本地生活服务项目有哪些等问题也成为了多个创业者社群中的热议对象。而从目前的讨论情况来看,在创业者们所询问的众多本地生活服务项目中,通过…...
ML 系列:【13 】— Logistic 回归(第 2 部分)
文章目录 一、说明二、挤压方法三、Logistic 回归中的损失函数四、后记 一、说明 在这篇文章中,我们将深入研究 squashing 方法,这是有符号距离方法(第 12节)的一种很有前途的替代方案。squashing 方法通过提供增强的对异常值…...
45岁被裁员的程序员,何去何从?
在当今快速变化的技术行业,职业生涯的稳定性受到挑战。在45岁被裁员,对很多程序员来说,可能是一种惊慌失措的体验。然而,这个阶段也可以被视为一个重新审视和调整方向的机会。本文将对可能的出路进行全方位的分析,并提…...
云计算Openstack Neutron
OpenStack Neutron是OpenStack云计算平台中的网络服务组件,它为OpenStack提供了强大的网络连接功能。 一、基本概念 Neutron是一个网络服务项目,旨在为OpenStack提供网络连接。它允许用户创建和管理虚拟网络,包括子网、路由、安全组等&…...
PointNet++网络详解
数据集转换 数据集转换的意义在于将原本的 txt 点云文件转换为更方便运算的npy点云文件,同时,将原本的xyzrgb这 6 个维度转换为xyzrgbc,最后一个c维度代表该点云所属的类别。 for anno_path in anno_paths:print(anno_path)try:elements a…...
Java | Leetcode Java题解之第459题重复的子字符串
题目: 题解: class Solution {public boolean repeatedSubstringPattern(String s) {return kmp(s s, s);}public boolean kmp(String query, String pattern) {int n query.length();int m pattern.length();int[] fail new int[m];Arrays.fill(fa…...
【动态规划-最长公共子序列(LCS)】【hard】力扣1092. 最短公共超序列
给你两个字符串 str1 和 str2,返回同时以 str1 和 str2 作为 子序列 的最短字符串。如果答案不止一个,则可以返回满足条件的 任意一个 答案。 如果从字符串 t 中删除一些字符(也可能不删除),可以得到字符串 s &#x…...
图片编辑为底片,智能工具助力,创作精彩视觉作品
在当今数字化时代,图像编辑已成为表达创意和美化视觉作品的重要手段。借助智能工具,即使是初学者也能轻松驾驭图片编辑。接下为大家展示图片编辑为底片图片的效果。 1.打开“首助编辑高手”,选择这里“图片批量处理”版块页面上 2.导入保存有…...
机器学习/数据分析--用通俗语言讲解时间序列自回归(AR)模型,并用其预测天气,拟合度98%+
时间序列在回归预测的领域的重要性,不言而喻,在数学建模中使用及其频繁,但是你真的了解ARIMA、AR、MA么?ACF图你会看么?? 时间序列数据如何构造???,我打过不少…...
回溯算法之值子集和问题详细解读(附带Java代码解读)
子集和问题(Subset Sum Problem) 是一个经典的组合优化问题。问题可以这样描述: 给定一个整数集合和一个目标整数 target,我们需要从集合中选出若干个整数,使它们的和等于 target。如果这样的子集存在,返回…...
mysql游标的使用
说明: 虽然我们也可以通过筛选条件 WHERE 和 HAVING,或者是限定返回记录的关键字 LIMIT 返回一条记录,但是,却无法在结果集中像指针一样,向前定位一条记录、向后定位一条记录,或者是 随意定位到某一条记录 …...
linux udev详解
1.概念介绍 1.1sysfs文件系统 Linux 2.6以后的内核引入了sysfs文件系统,sysfs被看成是与proc、devfs和devpty同类别的文件系统,该文件系统是一个虚拟的文件系统,它可以产生一个包括所有系统硬件的层级视图,与提供进程和状态信息…...
多云管理“拦路虎”:深入解析网络互联、身份同步与成本可视化的技术复杂度
一、引言:多云环境的技术复杂性本质 企业采用多云策略已从技术选型升维至生存刚需。当业务系统分散部署在多个云平台时,基础设施的技术债呈现指数级积累。网络连接、身份认证、成本管理这三大核心挑战相互嵌套:跨云网络构建数据…...
Java多线程实现之Callable接口深度解析
Java多线程实现之Callable接口深度解析 一、Callable接口概述1.1 接口定义1.2 与Runnable接口的对比1.3 Future接口与FutureTask类 二、Callable接口的基本使用方法2.1 传统方式实现Callable接口2.2 使用Lambda表达式简化Callable实现2.3 使用FutureTask类执行Callable任务 三、…...
Matlab | matlab常用命令总结
常用命令 一、 基础操作与环境二、 矩阵与数组操作(核心)三、 绘图与可视化四、 编程与控制流五、 符号计算 (Symbolic Math Toolbox)六、 文件与数据 I/O七、 常用函数类别重要提示这是一份 MATLAB 常用命令和功能的总结,涵盖了基础操作、矩阵运算、绘图、编程和文件处理等…...
什么是Ansible Jinja2
理解 Ansible Jinja2 模板 Ansible 是一款功能强大的开源自动化工具,可让您无缝地管理和配置系统。Ansible 的一大亮点是它使用 Jinja2 模板,允许您根据变量数据动态生成文件、配置设置和脚本。本文将向您介绍 Ansible 中的 Jinja2 模板,并通…...
MySQL账号权限管理指南:安全创建账户与精细授权技巧
在MySQL数据库管理中,合理创建用户账号并分配精确权限是保障数据安全的核心环节。直接使用root账号进行所有操作不仅危险且难以审计操作行为。今天我们来全面解析MySQL账号创建与权限分配的专业方法。 一、为何需要创建独立账号? 最小权限原则…...
Go 语言并发编程基础:无缓冲与有缓冲通道
在上一章节中,我们了解了 Channel 的基本用法。本章将重点分析 Go 中通道的两种类型 —— 无缓冲通道与有缓冲通道,它们在并发编程中各具特点和应用场景。 一、通道的基本分类 类型定义形式特点无缓冲通道make(chan T)发送和接收都必须准备好࿰…...
七、数据库的完整性
七、数据库的完整性 主要内容 7.1 数据库的完整性概述 7.2 实体完整性 7.3 参照完整性 7.4 用户定义的完整性 7.5 触发器 7.6 SQL Server中数据库完整性的实现 7.7 小结 7.1 数据库的完整性概述 数据库完整性的含义 正确性 指数据的合法性 有效性 指数据是否属于所定…...
WPF八大法则:告别模态窗口卡顿
⚙️ 核心问题:阻塞式模态窗口的缺陷 原始代码中ShowDialog()会阻塞UI线程,导致后续逻辑无法执行: var result modalWindow.ShowDialog(); // 线程阻塞 ProcessResult(result); // 必须等待窗口关闭根本问题:…...
Leetcode33( 搜索旋转排序数组)
题目表述 整数数组 nums 按升序排列,数组中的值 互不相同 。 在传递给函数之前,nums 在预先未知的某个下标 k(0 < k < nums.length)上进行了 旋转,使数组变为 [nums[k], nums[k1], …, nums[n-1], nums[0], nu…...
算法打卡第18天
从中序与后序遍历序列构造二叉树 (力扣106题) 给定两个整数数组 inorder 和 postorder ,其中 inorder 是二叉树的中序遍历, postorder 是同一棵树的后序遍历,请你构造并返回这颗 二叉树 。 示例 1: 输入:inorder [9,3,15,20,7…...
