使用 uiautomator2+pytest+allure 进行 Android 的 UI 自动化测试
目录
前言:
介绍
pytest
uiautomator2
allure
环境搭建
pytest
uiautomator2
allure
pytest 插件
实例
初始化 driver
fixture 机制
数据共享
测试类
参数化
指定顺序
运行指定级别
重试
hook 函数
断言
运行
运行某个文件夹下的用例
运行某个方法
运行某个类
运行 P0 级
运行非 P0 级
main 方式
报告
失败详情
失败截图
uiautomator2 基本操作
启动服务
事件
点击
滑动
监听
查看元素
安装
启动工具
无线运行
前言:
uiautomator2是基于Android平台的UI自动化测试框架,pytest是一种功能丰富的Python测试框架,而allure是一种用于生成漂亮测试报告的工具。结合使用这三个工具,您可以进行Android的UI自动化测试,并生成直观且可视化的测试报告。
本文主要讲解使用 uiautomator2+pytest+allure 进行 Android 的 UI 自动化测试,其实主要目的是写一些实战的脚本来更深入学习 pytest 框架.
另外也顺便介绍一下 uiautomator2 这款自动化框架,在使用上也是非常的顺畅.
之前我已经使用 appium+testng 写了一套自动化脚本了并且在公司实际使用了.这次就不用公司的 app 测试了,使用上家公司 58 同城的 app 进行自动化测试.
介绍
做 UI 自动化肯定需要选择一种适合的测试框架,比如 java 的 testng、python 的 unittest,主要目的是让代码的层级明确、简洁、复用性强,本次介绍下 python 的 pytest 框架.
pytest
pytest 官方:pytest: helps you write better programs — pytest documentation
The pytest framework makes it easy to write small tests, yet scales to support complex functional testing for applications and libraries.
官方的一段介绍,简单来说就是让写测试代码更容易并且没有那么多约束.当然这块不重点介绍 pytest 为什么好、怎么好,只需要记住 pytest 就是一个测试框架就够了.
uiautomator2
uiautomator2 是一个 Android UI 自动化框架,支持 Python 编写测试脚本对设备进行自动.底层基于 Google uiautomator,隶属于最近比较火热的 openatx 项目中.
下图是运行示意图:
设备中需要安装 atx-agent 作为 uiautomator2 的服务端,解析收到的请求,并转化成 uiautomator2 的代码.总体看来交互过程没有那么繁琐,在实际使用上的确比 appium 快不少.
allure
allure 是一款测试报告,炫酷的页面加上多种数据统计,比 HTMLTestRunner 报告强百倍,当然也支持多语言.
官方地址:http://allure.qatools.ru
环境搭建
使用 mac 电脑搭建环境
pytest
最新版本出到 4.0 了,但是实际使用 4.0 和 allure 有些不兼容.
所以推荐使用 3.7 版本的 pytest
pip install pytest==3.7
uiautomator2
uiautomator2 也是 python 的一个类库,用 pip 安装即可.
pip install uiautomator2
allure
brew install allure
pip install pytest-allure-adaptor
有了测试框架、自动化框架、测试报告,基本上就能 coding 了.
pytest 插件
pytest 插件可以实现失败重试、打印进度、指定顺序
pip install pytest-sugar # 打印进度pip install pytest-rerunfailures # 失败重试pip install pytest-ordering # 执行顺序
当然插件还有很多,这里就不一一介绍了.
实例
初始化 driver
做 UI 自动化都需要初始化一个 driver 对象,这个 driver 对象可以点击事件、滑动、双击等操作
uiautomator2 的初始化 driver 方式
相比 appium 配置很少,同时可以设置全局隐式等待元素时间
import uiautomator2 as ut2
def init_driver(self,device_name):'''初始化driver:return:driver'''try:logger.info(device_name)d = ut2.connect(device_name)#logger.info("设备信息:{}".format(d.info))# 设置全局寻找元素超时时间d.wait_timeout = wait_timeout # default 20.0# 设置点击元素延迟时间d.click_post_delay = click_post_delay#d.service("uiautomator").stop()# 停止uiautomator 可能和atx agent冲突logger.info("连接设备:{}".format(device_name))return dexcept Exception as e:logger.info("初始化driver异常!{}".format(e))
fixture 机制
unittest 框架有 setup 和 teardown 方法,用来做初始化和结束测试操作.pytest 是用@pytest.fixture方法来实现 setup 和 teardown.
下面这段代码就是定义一个 driver_setup 方法,来初始化和结束.
# 当设置autouse为True时,
# 在一个session内的所有的test都会自动调用这个fixture
@pytest.fixture()
def driver_setup(request):logger.info("自动化测试开始!")request.instance.driver = Driver().init_driver(device_name)logger.info("driver初始化")request.instance.driver.app_start(pck_name, lanuch_activity, stop=True)time.sleep(lanuch_time)allow(request.instance.driver)def driver_teardown():logger.info("自动化测试结束!")request.instance.driver.app_stop(pck_name)request.addfinalizer(driver_teardown)
另外还有一种方式实现,可以理解为 setup 和 teardown 在一个方法内,通过 yield 关键字停顿.
@pytest.fixture()
def init(self,scope="class"):self.home = Home(self.driver)self.home.news_tab()self.news = News(self.driver)logger.info("初始化消息模块")yield self.newslogger.info("结束消息模块")
yield 关键字是在 python 语法生成器和迭代器中使用,用来节省内存.
比如 for 循环一个大列表,一次性都循环出来非常浪费性能.
所以通过 yield 关键字来控制循环.
下面演示下 yield:
#!/usr/bin/env python
# -*- coding: utf-8 -*-def yt():print "第一次打印"yield 0print("第二次打印")if __name__ == '__main__':a = yt()print next(a)print next(a)
如果直接调用 yt 函数会发现啥也打印不出来,因为此时只是声明了 yt 函数并没有真正的使用.
使用 next 方法调用第一次,输入结果如下:
yield 在此时相当于 return 0,此时不会输出"第二次打印",会在这块停住.
第一次打印
0
使用 next 方法调用第二次,输入结果如下:
第二次打印
再来回顾下上面那个例子:
在 yield 之前完成了 setup 操作并且返回 self.news 对象
在 yied 之后完成了 teardown 操作
@pytest.fixture()
def init(self,scope="class"):self.home = Home(self.driver)self.home.news_tab()self.news = News(self.driver)logger.info("初始化消息模块")yield self.newslogger.info("结束消息模块")
数据共享
在 pytest 中只需要写 conftest.py 类,可以实现数据共享,不需要 import 就能自动找到一些配置.
刚才讲到的初始化 driver_setup 函数,就可以定在 conftest.py 类中,此时这个函数是全局可以函数,在测试类中使用如下:
使用@pytest.mark.usefixtures 装饰器就能引用到 driver_setup 函数
@allure.feature("测试发布")
@pytest.mark.usefixtures('driver_setup')
class TestNews:@pytest.fixture(params=item)def item(self, request):return request.param
测试类
pytest 检测如果是 test 开头或者 test 结尾的类,都认为是可以执行测试类.
在测试类中写 test 开头的测试方法
@allure.story('测试首页搜索')
def test_home_search(self,init):init.home_search()
参数化
假设场景是首页搜索多个词,需要借助参数化来完成
使用@pytest.mark.parametrize
@pytest.mark.parametrize(('kewords'), [(u"司机"), (u"老师"), (u"公寓")])
def test_home_moresearch(self, init,kewords):init.home_more_search(kewords)
指定顺序
假设发布用例,需要先登录才可以.可以通过用例排序的方式先登录,再发布
使用@pytest.mark.run,odrer 从小到大优先执行
@pytest.mark.usefixtures('driver_setup')
@pytest.mark.run(order=1)
# 指定login先执行
class TestLogin:
运行指定级别
假设已经写了很多用例,有些用例是冒烟用例,可以指定级别运行.
使用@pytest.mark.P0
@allure.story('测试首页更多')
@pytest.mark.P0
def test_home_more(self, init):init.home_more()
命令行执行: pytest -v -m "P0", 会执行所有 P0 级别的用例
重试
这个时候需要借助 pytest-rerunfailures 插件,用法如下:
@pytest.mark.flaky(reruns=5, reruns_delay=2)
@allure.story('测试精选活动')
def test_news_good(self,init):init.news_good()
当然这种方法是指定某个 case 失败重试
还可以全局设置用户如下:
pytest --reruns 2 --reruns_delay 2
reruns:重试次数
reruns_delay:重试的间隔时间
hook 函数
在 conftest.py 文件中定义@pytest.hookimpl函数,这个函数可以 hook 住 pytest 运行的状况
@pytest.hookimpl(tryfirst=True, hookwrapper=True)
def pytest_runtest_makereport(item, call):'''hook pytest失败:param item::param call::return:'''# execute all other hooks to obtain the report objectoutcome = yieldrep = outcome.get_result()# we only look at actual failing test calls, not setup/teardownif rep.when == "call" and rep.failed:mode = "a" if os.path.exists("failures") else "w"with open("failures", mode) as f:# let's also access a fixture for the fun of itif "tmpdir" in item.fixturenames:extra = " (%s)" % item.funcargs["tmpdir"]else:extra = ""f.write(rep.nodeid + extra + "\n")
从代码中可以看出可以获取失败情况的相关信息,当时有了失败信息就可以搞事情了,比如当用例失败的时候截图或者记录失败数量做数据统计.
断言
在跑用例的时候最后一步都会断言一下,比如断言元素是否存在等
def assert_exited(self, element):'''断言当前页面存在要查找的元素,存在则判断成功:param driver::return:'''if self.find_elements(element):logger.info("断言{}元素存在,成功!".format(element))assert Trueelse:logger.info("断言{}元素存在,失败!".format(element))assert False
还可以这样优化代码:
def assert_exited(self, element):'''断言当前页面存在要查找的元素,存在则判断成功:param driver::return:'''assert self.find_elements(element) == True,"断言{}元素存在,失败!".format(element)logger.info("断言{}元素存在,成功!".format(element))
assert 失败后会跑出 AssertionError 和定义的文案
AssertionError: 断言xxxxx元素存在,失败!
运行
介绍下几种常用命令行运行
运行某个文件夹下的用例
运行某个文件下的所有用例
pytest android/testcase
运行某个方法
类文件地址::方法名
pytest test_home.py::test_home_more
或者使用-k 参数 + 方法名
pytest -k test_home_more
运行某个类
有的时候需要调试正个测试类中所有测试方法
直接跟上类文件地址
pytest test_home.py
运行 P0 级
pytest -v -m "P0"
运行非 P0 级
pytest -v -m "not P0"
main 方式
在 run.py 中写如下代码,这种方式相当于把命令行参数封装到脚本中.
pytest.main(["-s","--reruns=2", "android/testcase","--alluredir=data"])
报告
测试代码写完了,还差一个非常好看的报告.以前我们一般都用 HTMLTestRunner 报告,但是 HTMLTestRunner 报告功能比较单一并且也不支持失败截图.
偶然在社区中看到了 allure 的帖子,看了展示报告简直是吊炸天,先附一张跑完用例的截图.
另外可以在代码中设置报告层次,用法如下:
@allure.feature("测试首页")
@pytest.mark.usefixtures('driver_setup')
class TestHome:@pytest.fixture()def init(self,scope="class"):self.home = Home(self.driver)logger.info("初始化首页模块")yield self.homelogger.info("结束首页模块")@allure.story('测试首页搜索')def test_home_search(self,init):init.home_search()
设置 allure.feature 和 allure.story,相当于上下级关系.
失败详情
点击失败用例就能看到失败的相关信息
失败截图
在跑自动化的过程已经遇到失败情况,需要一张截图描述当时的情况.
在上面提到@pytest.hookimpl函数中,最后调用截图方法,使用
allure.attach 把截图加上.
需要注意的是 attach 中的第二个参数是图片的二进制信息.
@pytest.hookimpl(tryfirst=True, hookwrapper=True)
def pytest_runtest_makereport(item, call):'''hook pytest失败:param item::param call::return:'''# execute all other hooks to obtain the report objectoutcome = yieldrep = outcome.get_result()# we only look at actual failing test calls, not setup/teardownif rep.when == "call" and rep.failed:mode = "a" if os.path.exists("failures") else "w"with open("failures", mode) as f:# let's also access a fixture for the fun of itif "tmpdir" in item.fixturenames:extra = " (%s)" % item.funcargs["tmpdir"]else:extra = ""f.write(rep.nodeid + extra + "\n")pic_info = adb_screen_shot()with allure.step('添加失败截图...'):allure.attach("失败截图", pic_info, allure.attach_type.JPG)
uiautomator2 基本操作
启动服务
执行如下命令:
python -m uiautomator2 init
会在手机上安装 atx-agent.apk 并且会在手机上启动服务
2018-12-14 18:03:50,691 - __main__.py:327 - INFO - Detect pluged devices: [u'a3f8ca3a']
2018-12-14 18:03:50,693 - __main__.py:343 - INFO - Device(a3f8ca3a) initialing ...
2018-12-14 18:03:51,154 - __main__.py:133 - INFO - install minicap
2018-12-14 18:03:51,314 - __main__.py:140 - INFO - install minitouch
2018-12-14 18:03:51,743 - __main__.py:168 - INFO - apk(1.1.7) already installed, skip
2018-12-14 18:03:51,744 - __main__.py:350 - INFO - atx-agent is already running, force stop
2018-12-14 18:03:52,308 - __main__.py:213 - INFO - atx-agent(0.5.0) already installed, skip
2018-12-14 18:03:52,490 - __main__.py:254 - INFO - launch atx-agent daemon
2018-12-14 18:03:54,568 - __main__.py:273 - INFO - atx-agent version: 0.5.0
atx-agent output: 2018/12/14 18:03:52 [INFO][github.com/openatx/atx-agent] main.go:508: atx-agent listening on 192.168.129.93:7912
监听的是手机上的 ip+ 默认 7921.
事件
事件类型比如点击和滑动等,介绍几个常用的.
点击
根据 id、xpath、text 定位元素,和 appium 使用上差别不大.
self.d(resourceId=element).click()
self.d.xpath(element).click()
self.d(text=element).click()
滑动
前 4 个参数是坐标,time 是控制滑动时间
self.d.drag(self.width / 2, self.height * 3 / 4, self.width / 2, self.height / 4, time)
监听
这个用于首次启动 app 点击权限或者开屏幕广告
when 方法就相当于 if 判断,满足条件才会点击,可以生去一大堆逻辑代码.
driver.watcher("允许").when(text="允许").click(text="允许")
driver.watcher("跳过 >").when(text="跳过 >").click(text="跳过 >")
driver.watcher("不要啦").when(text="不要啦").click(text="不要啦")
查看元素
安装
需要安装 weditor 库
pip install weditor
启动工具
python -m weditor
会在自动打开浏览器并且展示元素,相当于 web 版本的 uiautomatorviewer,使用起来比较方便.
无线运行
上边提到的手机 ip,有个这个手机 ip 就可以进行无线运行脚本
把 connect 中的方法替换成手机 ip 就可以了
# d = ut2.connect(device_name)
d = ut2.connect("192.168.129.93")
作为一位过来人也是希望大家少走一些弯路
在这里我给大家分享一些自动化测试前进之路的必须品,希望能对你带来帮助。
(软件测试相关资料,自动化测试相关资料,技术问题答疑等等)
相信能使你更好的进步!
点击下方小卡片
相关文章:

使用 uiautomator2+pytest+allure 进行 Android 的 UI 自动化测试
目录 前言: 介绍 pytest uiautomator2 allure 环境搭建 pytest uiautomator2 allure pytest 插件 实例 初始化 driver fixture 机制 数据共享 测试类 参数化 指定顺序 运行指定级别 重试 hook 函数 断言 运行 运行某个文件夹下的用例 运行某…...

Android APP性能及专项测试
Android篇 1. 性能测试 Android性能测试分为两类: 1、一类为rom版本(系统)的性能测试 2、一类为应用app的性能测试Android的app性能测试包括的测试项比如: 1、资源消耗 2、内存泄露 3、电量功耗 4、耗时 5、网络流量消耗 6、移动…...

人工智能自然语言处理:N-gram和TF-IDF模型详解
人工智能自然语言处理:N-gram和TF-IDF模型详解 1.N-gram 模型 N-Gram 是一种基于统计语言模型的算法。它的基本思想是将文本里面的内容按照字节进行大小为 N 的滑动窗口操作,形成了长度是 N 的字节片段序列。 每一个字节片段称为 gram,对所…...

linux内核调试工具记录
Linux性能测试使用的工具在github网站可见,网址如下: slides: http://www.slideshare.net/brendangregg/linux-performance-analysis-new-tools-and-old-secrets video: https://www.usenix.org/conference/lisa14/conference-program/presentation/greg…...

XSS 攻击的检测和修复方法
XSS 攻击的检测和修复方法 XSS(Cross-Site Scripting)攻击是一种最为常见和危险的 Web 攻击,即攻击者通过在 Web 页面中注入恶意代码,使得用户在访问该页面时,恶意代码被执行,从而导致用户信息泄露、账户被…...

Spring后置处理器BeanFactoryPostProcessor与BeanPostProcessor源码解析
文章目录 一、简介1、BeanFactoryPostProcessor2、BeanPostProcessor 二、BeanFactoryPostProcessor 源码解析1、BeanDefinitionRegistryPostProcessor 接口实现类的处理流程2、BeanFactoryPostProcessor 接口实现类的处理流程3、总结 三、BeanPostProcessor 源码解析 一、简介…...

NXP i.MX 6ULL工业开发板硬件说明书( ARM Cortex-A7,主频792MHz)
前 言 本文档主要介绍TLIMX6U-EVM评估板硬件接口资源以及设计注意事项等内容。 创龙科技TLIMX6U-EVM是一款基于NXP i.MX 6ULL的ARM Cortex-A7高性能低功耗处理器设计的评估板,由核心板和评估底板组成。核心板经过专业的PCB Layout和高低温测试验证,稳…...

Ubuntu 放弃了战斗向微软投降
导读这几天看到 Ubuntu 放弃 Unity 和 Mir 开发,转向 Gnome 作为默认桌面环境的新闻,作为一个Linux十几年的老兵和Linux桌面的开发者,内心颇感良多。Ubuntu 做为全世界Linux界的桌面先驱者和创新者,突然宣布放弃自己多年开发的Uni…...

高并发的哲学原理(六)-- 拆分网络单点(下):SDN 如何替代百万人民币的负载均衡硬件
上一篇文章的末尾,我们利用负载均衡器打造了一个五万 QPS 的系统,本篇文章我们就来了解一下负载均衡技术的发展历程,并一起用 SDN(软件定义网络)技术打造出一个能够扛住 200Gbps 的负载均衡集群。 负载均衡发展史 F5 …...

用OpenCV进行图像分割--进阶篇
1. 引言 大家好,我的图像处理爱好者们! 在上一篇幅中,我们简单介绍了图像分割领域中的基础知识,包含基于固定阈值的分割和基于OSTU的分割算法。这一次,我们将通过介绍基于色度的分割来进一步巩固大家的基础知识。 闲…...

Spring框架概述及核心设计思想
文章目录 一. Spring框架概述1. 什么是Spring框架2. 为什么要学习框架?3. Spring框架学习的难点 二. Spring核心设计思想1. 容器是什么?2. IoC是什么?3. Spring是IoC容器4. DI(依赖注入)5. DL(依赖查找&…...

Unity自定义后处理——Vignette暗角
大家好,我是阿赵。 继续说一下屏幕后处理的做法,这一期讲的是Vignette暗角效果。 一、Vignette效果介绍 Vignette暗角的效果可以给画面提供一个氛围,或者模拟一些特殊的效果。 还是拿这个角色作为底图 添加了Vignette效果后࿰…...

Java读取Excel 单元格包含换行问题
Java读取Excel 单元格包含换行问题 需求解决方案 需求 针对用户上传的Excel数据,或者本地读取的Excel数据。单元格中包含了换行,导致读取的数据被进行了切片。 正常读取如下图所示。 解决方案 目前是把数据读取出来的cell转成字符串后,…...

Django实现接口自动化平台(十)自定义action names【持续更新中】
相关文章: Django实现接口自动化平台(九)环境envs序列化器及视图【持续更新中】_做测试的喵酱的博客-CSDN博客 深入理解DRF中的Mixin类_做测试的喵酱的博客-CSDN博客 python中Mixin类的使用_做测试的喵酱的博客-CSDN博客 本章是项目的一…...

[爬虫]解决机票网站文本混淆问题-实战讲解
前言 最近有遇到很多小伙伴私信向我求助,遇到的问题基本上都是关于文本混淆或者是字体反爬的问题。今天给大家带来其中一个小伙伴的实际案例给大家讲讲解决方法 📝个人主页→数据挖掘博主ZTLJQ的主页 个人推荐python学习系列: ☄️爬虫J…...
【已解决】Flask项目报错AttributeError: ‘Request‘ object has no attribute ‘is_xhr‘
文章目录 报错及分析报错代码分析 解决方案必要的解决方法可能有用的解决方法 报错及分析 报错代码 File "/www/kuaidi/6f47274023d4ad9b608f078c76a900e5_venv/lib/python3.6/site-packages/flask/json.py", line 251, in jsonifyif current_app.config[JSONIFY_PR…...

【Java基础教程】Java学习路线攻略导图——史诗级别的细粒度归纳,持续更新中 ~
Java学习路线攻略导图 上篇 前言1、入门介绍篇2、程序基础概念篇3、包及访问权限篇4、异常处理篇5、特别篇6、面向对象篇7、新特性篇8、常用类库篇 前言 🍺🍺 各位读者朋友大家好!得益于各位朋友的支持和关注,我的专栏《Java基础…...
IntelliJ IDEA 2023.1 更新内容总结
IntelliJ IDEA 2023.1 更新内容总结 * 主要更新内容 * UI 大改版 * 性能改进项 * 其它更新内容IntelliJ IDEA 2023.1 更新内容总结 主要更新内容 IntelliJ IDEA 2023.1 针对新的用户界面进行了大量重构,这些改进都是基于收到的宝贵反馈而实现的。官方还实施了性能增强措施, …...

什么是计算机蠕虫?
计算机蠕虫诞生的背景 计算机蠕虫的诞生与计算机网络的发展密切相关。20世纪60年代末和70年代初,互联网还处于早期阶段,存在着相对较少的计算机和网络连接。然而,随着计算机技术的进步和互联网的普及,计算机网络得以迅速扩张&…...

【机器学习】吴恩达课程1-Introduction
一、机器学习 1. 定义 计算机程序从经验E中学习,解决某一任务T,进行某一性能P,通过P测定在T上的表现因经验E而提高。 2. 例子 跳棋程序 E:程序自身下的上万盘棋局 T:下跳棋 P:与新对手下跳棋时赢的概…...
利用DeepSeek编写能在DuckDB中读PostgreSQL表的表函数
前文实现了UDF和UDAF,还有一类函数是表函数,它放在From 子句中,返回一个集合。DuckDB中已有PostgreSQL插件,但我们可以用pqxx库实现一个简易的只读read_pg()表函数。 提示词如下: 请将libpqxx库集成到我们的程序&#…...

设计模式——责任链设计模式(行为型)
摘要 责任链设计模式是一种行为型设计模式,旨在将请求的发送者与接收者解耦,通过多个处理器对象按链式结构依次处理请求,直到某个处理器处理为止。它包含抽象处理者、具体处理者和客户端等核心角色。该模式适用于多个对象可能处理请求的场景…...

Unity + HybirdCLR热更新 入门篇
官方文档 HybridCLR | HybridCLRhttps://hybridclr.doc.code-philosophy.com/docs/intro 什么是HybirdCLR? HybridCLR(原名 huatuo)是一个专为 Unity 项目设计的C#热更新解决方案,它通过扩展 IL2CPP 运行时,使其支持动态加载和…...
MATLAB实现井字棋
一、智能决策系统与博弈游戏概述 (一)智能决策系统核心概念 智能决策系统(Intelligent Decision System, IDS)是通过数据驱动和算法模型模拟人类决策过程的计算机系统,核心目标是在复杂环境中自动生成最优策略&#…...
css3 新增属性/滤镜效果/裁剪元素/图片适应盒子/定义和使用变量/恢复默认initial
从 CSS3 发布至今,CSS 标准引入了大量新特性,极大地丰富了前端开发的能力。以下是 CSS3 之后的重要新增属性、模块与特性总结,涵盖布局、动画、交互、视觉、选择器、单位等多个领域。 🎨 视觉与效果增强 属性/功能作用示例filte…...

Appium+python自动化(七)- 认识Appium- 上
简介 经过前边的各项准备工作,终于才把appium搞定。 一、appium自我介绍 appium是一款开源的自动化测试工具,可以支持iOS和安卓平台上的原生的,基于移动浏览器的,混合的应用(APP)。 1、 使用appium进…...

告别复杂操作!电脑极简风格计时使用
无论是工作、学习还是日常生活,这款小巧实用的计时工具都能成为你掌控时间的好帮手。特别适合需要频繁切换正计时、倒计时和查看当前时间的场景。界面简洁,操作便捷,助你高效管理每一刻。 这是一款免安装的工具,下载后可直接打开…...

Compose原理 - 整体架构与主流程
一、整体架构 在官方文档中(Jetpack Compose 架构层 | Android Developers),对Compose的分层有所阐述: 其中 Runtime:提供Compose的基础运行能力,包括State、Side-effects、CompositionLocal、Compositio…...

《Python语言程序设计》2018 第4章第9题3重量和价钱的对比,利用第7章的概念来解答你
利用类来解答这个问题。 pack1, price1 50, 24.59 pack2, price2 25, 11.99class result:def __init__(self,pack,price):self.pack packself.price pricedef set_pack(self):return self.packdef set_price(self):return self.pricedef get_result(self):return self.pric…...

从监控到告警:Prometheus+Grafana+Alertmanager+告警通知服务全链路落地实践
文章目录 一、引言1.1 监控告警的必要性1.2 监控告警的基本原理1.2.1 指标采集与存储1.2.2 告警规则与触发机制1.2.3 多渠道通知与闭环 二、技术选型与架构设计2.1 为什么选择 Prometheus 及其生态2.1.1 Prometheus 优势分析2.1.2 Grafana 可视化能力2.1.3 Alertmanager 灵活告…...