当前位置: 首页 > news >正文

app稳定性测试-iOS篇

稳定性测试:测试应用程序在长时间运行过程中是否存在内存泄漏、崩溃等问题,以确保应用程序具有较高的稳定性和可靠性。

对于安卓端,官方提供了很好的稳定性测试工具:monkey。相比较而言,iOS则没有,而且当前网络上似乎也没有很好的第三方工具可以使用,因此只能自己写了。

我们要开发的iOS稳定性测试程序,应该至少包含以下内容:

  • 持续随机触发UI事件

  • 崩溃重启,测试不中断

  • 日志记录

首先我们确定以上设想的可行性,然后再制定实施方案。在iOS原生开发语言swift和object-C中提供了可进行单元测试和UI测试的XCTest框架,而同样可进行移动端UI测试的第三方框架还有Appium等,但相比较第三方的开源框架,原生的XCTest框架性能更好且更稳定,因此这里我们选择基于swift语言和XCTest框架来开发。

XCTest框架提供了非常全面的启动App和UI操作相关的API接口, 因此1、2两点完全可以实现,当然第三点的日志记录的实现也同样不会有什么问题。接下来就是具体实施了。

首先,我们创建一个用来执行测试的主类:StabilityTestRunner,然后再编写代码去实现以上三点。

持续随机触发UI事件

让我们拆分一下,随机触发UI事件,实际上包含两部分:随机UI元素和随机的UI操作。

那么:随机生成UI元素:

func randomElement(of types: [ElementType]) -> XCUIElement? { var allElement:[XCUIElement] = [] for type in types { if !self.exists{ break } var elements: [XCUIElement] if self.alerts.count > 0 { elements = self.alerts.descendants(matching: type).allElementsBoundByIndex }else { elements = self.descendants(matching: type).allElementsBoundByIndex } let filteredElements = elements.filter { element in if !element.exists { return false } if !element.isHittable || !element.isEnabled { return false // Filter out non clickable and blocked elements. } return true } allElement.append(contentsOf: filteredElements) } return allElement.randomElement() }

随机生成UI操作:

/** Random execution of the given UI operation. - parameter element: Page Elements. - parameter actions: Dictionary objects containing different UI operations. */ private func performRandomAction(on element: XCUIElement, actions: [String: (XCUIElement) -> ()]) { let keys = Array(actions.keys) let randomIndex = Int.random(in: 0..<keys.count) let randomKey = keys[randomIndex] let action = actions[randomKey] if action == nil { return } if !element.exists { return } if !element.isHittable { return } Utils.log("step\(currentStep): \(randomKey) \(element.description)") action!(element) }

持续测试和崩溃重启

while !isTestingComplete{ // Randomly select page elements. let element = app.randomElement(of: elementType) if element != nil { currentStep += 1 takeScreenshot(element: element!) performRandomAction(on: element!, actions: actions) // Perform random UI operations. XCTWaiter().wait(for: [XCTNSPredicateExpectation(predicate: NSPredicate(format: "self == %d", XCUIApplication.State.runningForeground.rawValue), object: app)], timeout: stepInterval) if app.state != .runningForeground { if app.state == .notRunning || app.state == .unknown { Utils.saveImagesToFiles(images: screenshotData) Utils.saveImagesToFiles(images: screenshotOfElementData, name: "screenshot_element") Utils.log("The app crashed. The screenshot before the crash has been saved in the screenshot folder.") } app.activate() } } }

日志记录

记录截图并标记UI元素:

private func takeScreenshot(element: XCUIElement) { let screenshot = app.windows.firstMatch.screenshot().image if screenshotData.count == 3 { let minKey = screenshotData.keys.sorted().first! screenshotData.removeValue(forKey: minKey) } let screenshotWithRect = Utils.drawRectOnImage(image: screenshot, rect: element.frame) screenshotData[currentStep] = screenshotWithRect.pngData() let screenshotOfElement = element.screenshot().pngRepresentation if screenshotOfElementData.count == 3 { let minKey = screenshotOfElementData.keys.sorted().first! screenshotOfElementData.removeValue(forKey: minKey) } screenshotOfElementData[currentStep] = screenshotOfElement }

通过文本日志记录测试执行过程:

static func log(_ message: String) { print(message) let dateFormatter = DateFormatter() dateFormatter.dateFormat = "yyyy-MM-dd HH:mm:ss" let dateString = dateFormatter.string(from: Date()) let fileManager = FileManager.default do { try fileManager.createDirectory(atPath: logSavingPath, withIntermediateDirectories: true, attributes: nil) } catch { print("Error creating images directory: \(error)") } var fileURL: URL if #available(iOS 16.0, *) { fileURL = URL.init(filePath: logSavingPath).appendingPathComponent("log.txt") } else { fileURL = URL.init(fileURLWithPath: logSavingPath).appendingPathComponent("log.txt") } do { try "\(dateString) \(message)".appendLineToURL(fileURL: fileURL) } catch { print("Error writing to log file: \(error)") }

日志导出:

// To add the log files to the test results file, you can view it on your Mac. The test results file path: /User/Library/Developer/Xcode/DerivedData/AppStability-*/Logs. let zipFile = "\(NSSearchPathForDirectoriesInDomains(.documentDirectory, .userDomainMask, true)[0])/Logs.zip" let attachment = XCTAttachment(contentsOfFile: URL(fileURLWithPath: zipFile)) attachment.name = "Logs" attachment.lifetime = .keepAlways // Add the "Logs.zip" file to the end of test result file. add(attachment) Utils.log("The logs for test steps has been added to the end of test result file at /User/Library/Developer/Xcode/DerivedData/AppStability-*/Logs")

注:以上代码只是主体实现,了解相关细节可通过GitHub或Gitee查阅完整代码。

总结

总的来说实现起来并不是很困难,当然从程序使用角度而言,用户可自定义随机UI事件的UI元素范围和UI操作的范围以及测试执行的时长和时间间隔,因此需要对ios应用程序和Xcode的使用以及iOS UI事件有一定的了解,具体使用可查看完整工程中的示例。

感谢每一个认真阅读我文章的人,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走:

这些资料,对于【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴上万个测试工程师们走过最艰难的路程,希望也能帮助到你!有需要的小伙伴可以点击下方小卡片领取

http://www.jsqmd.com/news/304011/

相关文章:

  • 导师严选10个AI论文平台,助你轻松搞定本科论文!
  • 工业大数据如何定义及其在制造业中的核心价值
  • NJet MCP网关:让AI 从“能聊”到“能干”
  • 图片批量压缩神器!一秒搞定,可对图片格式转换,压缩后画质依旧清晰~
  • 1秒搞定文件/文件夹批量重命名,效率狂飙200%!支持14种命名方法,使用完全免费,批量修改处理~
  • C盘瞬间多出10GB空间!电脑垃圾清理神器,支持重复文件清理、系统瘦身等,免安装无广告绿色版! 软媒清理大师
  • 电脑系统优化工具,免费且功能多Glary Utilities
  • 你用Audiobookshelf 打造的专属有声书库,cpolar让它通勤、居家都超实用。
  • 抠图新手必看~AI抠图免费使用,可替换背景,提高效率,替代ps抠图!
  • Spring Boot 核心接口与扩展点详细指南
  • C# 封装、继承、多态 通俗解释
  • Docker搭建Mysql8的主从复制
  • 无限制使用!Ai配音工具来了,免费文字转语音神器,支持多角色配音及一键生成视频字幕生成,祈风TTS
  • 贵金属涨势愈发迅猛 背后核心动因深度解析
  • 地下管网的“水质哨兵”如何实现72小时隐患预警?
  • 效果惊人的好!无需连网,本地离线跨平台一键AI抠图神器,开源免费使用,实用值得收藏 小宾AI抠图
  • Pagehelper触发 JVM 类校验失败,Idea 却因 -noverify 藏了雷
  • Web测试和App测试的区别
  • 2026成都壁挂炉维修优质服务推荐榜
  • 超纯水哪家口碑好?技术、服务与用户反馈全解析
  • 2026成都专业律师事务所推荐榜:成都遗产继承律师事务所、成都刑事律师事务所、成都交通事故罪辩护律师事务所、成都借款纠纷律师事务所选择指南
  • 微美全息(NASDAQ:WIMI)BlockEdge框架:为工业4.0开辟区块链与边缘计算创新之路
  • 使用postgresql、postgis数据库作为存储仓库,发布geoserver矢量切片服务(pbf切片)、矢量切片图层组服务
  • NOKOV 度量动作捕捉助力多飞行器集联平台(IAP)高精度定位研究|北理工俞玉树团队成果解读
  • 2026最新抗菌316厨具/家用食堂厨具/食品304厨具/纯钛厨具/厨具定制批发推荐:五星厨具专家,20年品质护航
  • Windows 下隐藏命令行子进程窗口的技术方案
  • 右键菜单管理
  • 【大模型开源教程】Llama3-Chinese-8B:专为中文打造的大语言模型,领航智能对话
  • 基于SpringBoot的闪电队篮球俱乐部管理系统的设计与开发(源码+论文+部署+安装)
  • 大型语言模型(LLMs)是怎样“学习”的?一封给网络工程师的大模型指南