移动端自动化测试Appium-java
一、Appium的简介
移动端的自动化测试框架
模拟人的操作进行功能自动化常用于功能测试、兼容性测试
跨平台的自动化测试
二、Appium的原理
核心是web服务器,接受客户端的连接,接收客户端的命令,在手机设备上执行命令,收集命令执行的结果。
Session,客户端初始化一个 session 会话,发送 POST/session 请求到服务器端,这些请求里面都会带有一个对象:desiredCapabilities,这个时候服务器端会启动自动化 session 然后返回一个 session ID,以后的命令都会用这个 seesion ID 去匹配。
desired capabilities 对象其实是一个 key-value 的集合,里面包含了各种各样的信息,发送到服务器端后,服务器解析这些信息就知道了客户端对哪种 session 感兴趣,然后就会启动相应的session,这里面的信息会影响着服务器端启动 session 的类型。
Appium Server,Appium 是一个用 Node.js 编写的 HTTP server,它创建、并管理多个 WebDriver sessions 来和不同平台交互。
Apium ClientsAppium 开始一个测试后,就会在被测设备(手机)上启动一个 server ,监听来
自 Appium server 的指令,每种平台(如 iOS 和 Android)都有不同的运行和交互方式,所以 Appium 会用某个桩程序“侵入”该平台,并接受指令,来完成测试用例的运行。
三、Appium的使用
3.1导入依赖
<dependency><groupId>org.testng</groupId><artifactId>testng</artifactId><version>7.0.0-beta1</version></dependency><dependency><groupId>io.appium</groupId><artifactId>java-client</artifactId><version>8.5.1</version><!-- <scope>test</scope>--></dependency><dependency><groupId>org.seleniumhq.selenium</groupId><artifactId>selenium-java</artifactId><version>4.9.1</version></dependency>
3.2 创建测试类
使用@test注解,类的名字要符合标识符命名规则即可
import io.appium.java_client.AppiumDriver;
import org.openqa.selenium.remote.DesiredCapabilities;
import org.testng.annotations.Test;public class NewTest {//定义 AppiumDriver 对象 AppiumDriver driver;@Testpublic void f(){}}
3.3 @BeforeClass 注解
@BeforeClass 注解 用于设定进入测试类后,在所有测试之前首先要执行的代码,指定测试设备平台。
@BeforeClasspublic void beforeClass() throws MalformedURLException {//指定测试设备信息DesiredCapabilities device=new DesiredCapabilities();//使用移动设备或模拟器种类device.setCapability("deviceName","Android Emulator");//指定哪个移动操作平台device.setCapability("platformName", "Android");//指定移动操作系统版本device.setCapability("platformVersion", "4.4.2");//指定 app 程序包名,即被测程序名device.setCapability("appPackage","com.android.calculator2");//指定 app 启动页名称device.setCapability("appActivity",".Calculator");//启动appdriver = new AndroidDriver(new URL("http://localhost:4723/wd/hub"),device);}
3.4 @AfterClass 注解
在所有测试之后再执行的代码,可在该注解写入app退出代码
@AfterClasspublic void afterClass() {//退出appdriver.quit()}
3.5 @Test(属性) 注解
指定要测试的方法,方法名只要符合标识符命名规则。
属性
description="测试描述"
priority=优先级,从 0 开始
timeout=?ms,超时时间
dataProvider="Dataprovider 的名称或方法名"
dataProviderClass=产生测试数据的类
@Test(description="123",priority=0)public void f() {System.out.println("hello");}
3.6 识别操作元素
通过\Android\android-sdk\tools\ uiautomatorviewer 去探操作元素的id
driver.findElementById(resource-id 属性)
driver.findElementByClassName(class 属性)
@Test(description="123",priority=0)public void f() {//点击driver.findElementById("com.android.calculator2:id/digit3").click();driver.findElementById("com.android.calculator2:id/plus").click();driver.findElementById("com.android.calculator2:id/digit2").click();driver.findElementById("com.android.calculator2:id/equal").click();//获取文本String sum=driver.findElementByClassName("android.widget.EditText").getText();//输入driver.findElementById("属性").sendKeys("数据");//清空driver.findElementById("属性").clear();System.out.print(sum);}
3.7 断言
断言主要用于判断预期结果与实际结果是否一致
//用于测试期望结果的断言,即测试两个对象是否相等Assert.assertEquals("实际值", "期望值");//布尔值断言Boolean rs=sum.contains("");Assert.assertTrue(rs);
3.8 模拟按键
driver.pressKeyCode(keycode),必须使用 AmdroidDriver
AndroidDriver driver;@Test
public void f() {driver.pressKeyCode("按键");
}
3.9 参数化
使用数组返回参数
@DataProvider
public Object[][] getData(){Object data[][]={ {"234","+","56","290"},{"256","-","40","216"}, {"3","*","6","18"} };return data;
}@Test(dataProvider="getData")
public void f(String one,String two,String exact) {}
读参数文件
@DataProviderpublic Object[][] getParam() throws Exception{List<String[]> rows=new ArrayList<String[]>();//导入文件File file=new File("文件路径");//读取字节文件FileReader reader=new FileReader(file);//字节转字符BufferedReader buffer=new BufferedReader(reader);String row=null;//读取每一行数据while((row=buffer.readLine())!=null){String columns[]=row.split("\t");rows.add(columns);}//文件关闭reader.close();//转换ObjectObject[][] data=new Object[rows.size()][];for(int i=0;i<rows.size();i++)data[i]=rows.get(i);return data;}
定义获得参数的单独类和方法
使用@DataProvider(name="测试数据集名")修饰获取参数的方法名。
@Test(dataProvider="getParam",dataProviderClass=参数类名.class)
public class Parm {@DataProvider()public Object[][] getParam() throws Exception{Object data[][]={ {"234","+","56","290"},{"256","-","40","216"}, {"3","*","6","18"} };return data;}}
3.10 测试报告
java项目加入TestReport
package app; //修改为正确的包名
import java.io.File;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.nio.file.StandardOpenOption;
import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.Date;
import org.testng.ITestContext;
import org.testng.ITestResult;
import org.testng.TestListenerAdapter;public class TestReport extends TestListenerAdapter{private String reportPath;@Overridepublic void onStart(ITestContext context) {File htmlReportDir = new File("test-output");if (!htmlReportDir.exists()) {htmlReportDir.mkdirs();}String reportName = formateDate()+"_result.html";reportPath = htmlReportDir+"/"+reportName;File report = new File(htmlReportDir,reportName);if(report.exists()){try {report.createNewFile();} catch (IOException e) {e.printStackTrace();}}StringBuilder sb = new StringBuilder("<html><head><meta http-equiv=\"Content-Type\" content=\"text/html; charset=utf-8\" />"+ "<title>自动化测试报告</title></head><body style=\"background-color:#99FFCC;\">"+ "<div id=\"top\" align=\"center\"><p style=\"font-weight:bold;\">测试用例运行结果列表</p>" + "<table width=\"90%\" height=\"80\" border=\"1\" align=\"center\" cellspacing=\"0\" rules=\"all\" style=\"table-layout:relative;\">"+ "<thead>"+ "<tr>"+ "<th>测试用例名</th>"+ "<th>测试用例结果</th>"+ "</tr>"+ "</thead>"+ "<tbody style=\"word-wrap:break-word;font-weight:bold;\" align=\"center\">");String res = sb.toString();try { Files.write((Paths.get(reportPath)),res.getBytes("utf-8")); } catch (IOException e) { e.printStackTrace(); } }@Overridepublic void onTestSuccess(ITestResult result) {StringBuilder sb = new StringBuilder("<tr><td>");sb.append(result.getMethod().getRealClass()+"."+result.getMethod().getMethodName());sb.append("</td><td><font color=\"green\">Passed</font></td></tr>");String res = sb.toString();try {Files.write((Paths.get(reportPath)),res.getBytes("utf-8"),StandardOpenOption.APPEND);} catch (IOException e) {e.printStackTrace();}}@Overridepublic void onTestSkipped(ITestResult result) {StringBuilder sb = new StringBuilder("<tr><td>");sb.append(result.getMethod().getRealClass()+"."+result.getMethod().getMethodName());sb.append("</td><td><font color=\"yellow\">Skipped</font>");sb.append("<p align=\"left\">测试用例<font color=\"red\">跳过</font>,原因:<br>");sb.append("<br><a style=\"background-color:#CCCCCC;\">");Throwable throwable = result.getThrowable(); sb.append(throwable.getMessage()); sb.append("</a></p></td></tr>");String res = sb.toString();try {Files.write((Paths.get(reportPath)),res.getBytes("utf-8"),StandardOpenOption.APPEND);} catch (IOException e) {e.printStackTrace();}}@Overridepublic void onTestFailure(ITestResult result) {StringBuilder sb = new StringBuilder("<tr><td>");sb.append(result.getMethod().getRealClass()+"."+result.getMethod().getMethodName());sb.append("</td><td><font color=\"red\">Failed</font><br>");sb.append("<p align=\"left\">测试用例执行<font color=\"red\">失败</font>,原因:<br>");sb.append("<br><a style=\"background-color:#CCCCCC;\">");Throwable throwable = result.getThrowable();sb.append(throwable.getMessage());sb.append("</a></p></td></tr>");String res = sb.toString();try {Files.write((Paths.get(reportPath)),res.getBytes("utf-8"),StandardOpenOption.APPEND);} catch (IOException e) {e.printStackTrace();}}@Overridepublic void onFinish(ITestContext testContext) {StringBuilder sb = new StringBuilder("</tbody></table><a href=\"#top\">返回顶部</a></div></body>");sb.append("</html>");String msg = sb.toString();try {Files.write((Paths.get(reportPath)),msg.getBytes("utf-8"),StandardOpenOption.APPEND);} catch (IOException e) {e.printStackTrace();}}public static String formateDate(){SimpleDateFormat sf = new SimpleDateFormat("yyyyMMdd HHmmss");Calendar cal = Calendar.getInstance();Date date = cal.getTime();return sf.format(date);}
}
测试类中添加监听器@Listeners({TestReport.class})即可,class 是固定关键字,放到测试类名的上一行
@Listeners({TestReport.class})
public class NewTest {}
测试报告默认存储位置Java 项目名\test-output
四、adb命令使用
android debug bridge 即 android 调试桥,在cmd输入
查看版本 adb version
显示所有的设备 adb devices
安装程序 adb install .apk 文件名,apk 文件名即程序包名,程序名中不要使用汉字
卸载 App adb uninstall 完整包名,包名不是 apk 文件名,可以使用uiautomatorviewer寻找package
查看手机操作系统的版本号 adb shell getprop ro.build.version.release
显示设备中的包以及包的启动项
所有包 adb shell getprop ro.build.version.release
查找某个包 adb shell pm list packages | findstr 查找关键字
查找启动项 adb shell dumpsys window w | findstr \/ | findstr name=
需要提前启用app
在指定设备中运行命令 adb -s 设备名 shell 命令
adb 服务器的开关 adb start-server adb kill-server
五、
相关文章:

移动端自动化测试Appium-java
一、Appium的简介 移动端的自动化测试框架 模拟人的操作进行功能自动化常用于功能测试、兼容性测试 跨平台的自动化测试 二、Appium的原理 核心是web服务器,接受客户端的连接,接收客户端的命令,在手机设备上执行命令,收集命令…...

IO: 作业:Day1
思维导图 main.c #include"student.h" int main(int argc, const char *argv[]) { stuPtr hcreat(); int n0; add_node(h); add_node(h); add_node(h); show(h); save(h,"student.txt"); stuPtr ptrc…...

ue5 替换角色的骨骼网格体和动画蓝图
一开始动画蓝图,骨骼网格体都是用的女性角色 现在把它换成男性 编译 保存 运行 把动画类换成ABP_Manny 进入ABP_Manny中 进入到idle 找到这个拖进来 编译 就变成站着端枪 运行一下,没有问题...
el-cascader 树状选择-点击父级禁用子级
背景:项目上需要实现树状选择,点击父级禁用子级的功能,element组件本身没有该配置项说明:需要实现几个功能点:点击父级禁用子级;再次点击取消禁用;仅回填所选级;上下级不关联实现代码…...

AWS re:Invent 的创新技术
本月早些时候,Amazon 于 12 月 1 日至 5 日在内华达州拉斯维加斯举行了为期 5 天的 re:Invent 大会。如果您从未参加过 re:Invent 会议,那么最能描述它的词是“巨大”——不仅从与会者人数(60,000 人)来看&…...

PHP7和PHP8的最佳实践
php 7 和 php 8 的最佳实践包括:使用类型提示以避免运行时错误;利用命名空间组织代码并避免命名冲突;采用命名参数、联合类型等新特性增强可读性;用错误处理优雅地处理异常;关注性能优化,如避免全局变量和选…...
Debian、Ubuntu 22.04和ubuntu 24.04国内镜像源(包括 docker 源)
Debian 更换国内清华源 1、备份原文件mv /etc/apt/sources.list /etc/apt/sources.list.old 2、写入新源,以下是 Debian 11 的: cat > /etc/apt/sources.list << EOF deb https://mirrors.tuna.tsinghua.edu.cn/debian/ bullseye main contrib…...

点亮一个esp32 的led
最近入了一个ESP32 兄弟们,这玩意还可以,买来肯定是给它点亮啊对吧 我就是点灯侠🎇 😭千万不要不接天线啊,不然你会一直找不到你的wifi 1.点灯第一步你得有IDE Arduino 就是这个绿东西 可是怎么下载安装呢ÿ…...
C++ shared_ptr进一步认知,为什么引用计数>2退出作用域都可以调用析构
1.使用智能指针需要#include <memeroy> 2.上代码: #include <memory> #include <iostream> using namespace std; struct lifePeriod {lifePeriod():a(1){cout << "无参构造!" << endl;}virtual ~lifePeriod(…...
JavaScript代码片段二
见过不少人、经过不少事、也吃过不少苦,感悟世事无常、人心多变,靠着回忆将往事串珠成链,聊聊感情、谈谈发展,我慢慢写、你一点一点看...... JavaScript统计文字个数、特殊字符转义、动态插入js代码、身份证验证 统计文字个数 f…...

【计算机视觉】单目深度估计模型-Depth Anything-V2
概述 本篇将简单介绍Depth Anything V2单目深度估计模型,该模型旨在解决现有的深度估计模型在处理复杂场景、透明或反射物体时的性能限制。与前一代模型相比,V2版本通过采用合成图像训练、增加教师模型容量,并利用大规模伪标签现实数据进行学…...
Servlet 和 Spring MVC:区别与联系
前言 在 Java Web 开发中,Servlet 和 Spring MVC 是两个重要的技术。Servlet 是 Java Web 的基础组件,而 Spring MVC 是一个高级 Web 框架,建立在 Servlet 的基础之上,提供了强大的功能和易用性。这篇文章将从定义、原理、功能对…...

【期末复习】三、内存管理
1.物理内存管理 空闲内存管理方式主要分为:等长划分和不等长划分。 内存管理方式 单一连续分区 基本思想:一段时间内只有一个进程在内存。 特点:简单,内存利用率低, 有三种不同的布局: 固定分区 把内存空间分割成若干区域, 称为分区。 每个分区的大小可以相同也可…...

Microsoft Azure Cosmos DB:全球分布式、多模型数据库服务
目录 前言1. Azure Cosmos DB 简介1.1 什么是 Azure Cosmos DB?1.2 核心技术特点 2. 数据模型与 API 支持2.1 文档存储(Document Store)2.2 图数据库(Graph DBMS)2.3 键值存储(Key-Value Store)…...

【Docker】安装registry本地镜像库,开启Https功能
下载镜像 docker pull registry:2 需要启动https功能,就要生成服务端的自签名的证书和私钥,以及在docker客户端安装这个经过签名的证书。 第一步:生成公私钥信息,第二步,制作证书签名申请文件, 第三步&…...

JUC--线程池
线程池 七、线程池7.1线程池的概述7.2线程池的构建与参数ThreadPoolExecutor 的构造方法核心参数线程池的工作原理 Executors构造方法newFixedThreadPoolnewCachedThreadPoolnewSingleThreadExecutornewScheduledThreadPool(int corePoolSize) 为什么不推荐使用内置线程池&…...
后端Java开发:第十一天
第十一天:方法重载 - 理解与应用 今天我们继续深入 Java 的世界,讨论 Java 中的 方法重载(Method Overloading)。你可能会想,什么是方法重载?简单来说,方法重载允许你在一个类中定义多个同名方…...

基于 GEE 的长时间序列 Landsat 5 影像下载
目录 1 完整代码 2 运行结果 1 完整代码 var LT5 ee.ImageCollection("LANDSAT/LT05/C01/T1"),imageVisParam {"opacity":1,"bands":["B4","B3","B2"],"gamma":1},roi ee.FeatureCollection(&quo…...

Unity-Mirror网络框架从入门到精通之Attributes属性介绍
前言 在现代游戏开发中,网络功能日益成为提升游戏体验的关键组成部分。Mirror是一个用于Unity的开源网络框架,专为多人游戏开发设计。它使得开发者能够轻松实现网络连接、数据同步和游戏状态管理。本文将深入介绍Mirror的基本概念、如何与其他网络框架进…...

软考证书邮寄步骤
一、点击网址 https://www.ruankao.org.cn/ 复制上述网址,粘贴至浏览器中。点击 “报名入口” 。 二、点击入口 选择考试批次。点击你所在考试地点的入口并进入。 三、登录 输入手机号和密码。进行验证。 四、点击基本信息 点击右上角。进入 “基本信息” 。 五、…...

【HarmonyOS 5.0】DevEco Testing:鸿蒙应用质量保障的终极武器
——全方位测试解决方案与代码实战 一、工具定位与核心能力 DevEco Testing是HarmonyOS官方推出的一体化测试平台,覆盖应用全生命周期测试需求,主要提供五大核心能力: 测试类型检测目标关键指标功能体验基…...

iPhone密码忘记了办?iPhoneUnlocker,iPhone解锁工具Aiseesoft iPhone Unlocker 高级注册版分享
平时用 iPhone 的时候,难免会碰到解锁的麻烦事。比如密码忘了、人脸识别 / 指纹识别突然不灵,或者买了二手 iPhone 却被原来的 iCloud 账号锁住,这时候就需要靠谱的解锁工具来帮忙了。Aiseesoft iPhone Unlocker 就是专门解决这些问题的软件&…...

《通信之道——从微积分到 5G》读书总结
第1章 绪 论 1.1 这是一本什么样的书 通信技术,说到底就是数学。 那些最基础、最本质的部分。 1.2 什么是通信 通信 发送方 接收方 承载信息的信号 解调出其中承载的信息 信息在发送方那里被加工成信号(调制) 把信息从信号中抽取出来&am…...

Cloudflare 从 Nginx 到 Pingora:性能、效率与安全的全面升级
在互联网的快速发展中,高性能、高效率和高安全性的网络服务成为了各大互联网基础设施提供商的核心追求。Cloudflare 作为全球领先的互联网安全和基础设施公司,近期做出了一个重大技术决策:弃用长期使用的 Nginx,转而采用其内部开发…...
JVM暂停(Stop-The-World,STW)的原因分类及对应排查方案
JVM暂停(Stop-The-World,STW)的完整原因分类及对应排查方案,结合JVM运行机制和常见故障场景整理而成: 一、GC相关暂停 1. 安全点(Safepoint)阻塞 现象:JVM暂停但无GC日志,日志显示No GCs detected。原因:JVM等待所有线程进入安全点(如…...
Python ROS2【机器人中间件框架】 简介
销量过万TEEIS德国护膝夏天用薄款 优惠券冠生园 百花蜂蜜428g 挤压瓶纯蜂蜜巨奇严选 鞋子除臭剂360ml 多芬身体磨砂膏280g健70%-75%酒精消毒棉片湿巾1418cm 80片/袋3袋大包清洁食品用消毒 优惠券AIMORNY52朵红玫瑰永生香皂花同城配送非鲜花七夕情人节生日礼物送女友 热卖妙洁棉…...
【SSH疑难排查】轻松解决新版OpenSSH连接旧服务器的“no matching...“系列算法协商失败问题
【SSH疑难排查】轻松解决新版OpenSSH连接旧服务器的"no matching..."系列算法协商失败问题 摘要: 近期,在使用较新版本的OpenSSH客户端连接老旧SSH服务器时,会遇到 "no matching key exchange method found", "n…...
libfmt: 现代C++的格式化工具库介绍与酷炫功能
libfmt: 现代C的格式化工具库介绍与酷炫功能 libfmt 是一个开源的C格式化库,提供了高效、安全的文本格式化功能,是C20中引入的std::format的基础实现。它比传统的printf和iostream更安全、更灵活、性能更好。 基本介绍 主要特点 类型安全:…...
es6+和css3新增的特性有哪些
一:ECMAScript 新特性(ES6) ES6 (2015) - 革命性更新 1,记住的方法,从一个方法里面用到了哪些技术 1,let /const块级作用域声明2,**默认参数**:函数参数可以设置默认值。3&#x…...

企业大模型服务合规指南:深度解析备案与登记制度
伴随AI技术的爆炸式发展,尤其是大模型(LLM)在各行各业的深度应用和整合,企业利用AI技术提升效率、创新服务的步伐不断加快。无论是像DeepSeek这样的前沿技术提供者,还是积极拥抱AI转型的传统企业,在面向公众…...