位置: 编程技术 - 正文

Android 进行单元测试难在哪-part1(android单选)

编辑:rootadmin
原文链接 : Against Android Unit Tests 原文作者 : Matthew Dupree 译文出自 : 开发技术前线 www.devtf.cn 译者 : chaossss 校对者: tiiime 状态 : 完成

推荐整理分享Android 进行单元测试难在哪-part1(android单选),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:android单activity,android单选,android单选dialog,android 单元测试教程,android单选按钮代码,android 单元测试教程,android 单元测试教程,安卓实现单选题,内容如对您有帮助,希望把文章链接给更多的朋友!

正如我在序中所说,在 Android 中难于进行测试是众多 Android 开发者的共识。上一篇博文发出后,有许多同行回复了我,并对我的观点表示支持:

— Andy Dyer (@dammitandy) April ,

@philosohacker大大发话了!他写了一篇有关如何能更好地测试 Android 应用的博文,地址戳我 — Andy Dyer (@dammitandy) April ,

卧槽,我终于知道在 Android 上测试应用的正确姿势了!感谢:地址戳我

非常赞同“Android SDK 排斥 Android 测试单元”的看法。你应该让你的代码和 SDK 的的联系尽可能地小。

— Pascal Hartig (@passy) April ,

所以在 Android 中进行单元测试困难的问题,我提到的这些已经说得足够清楚了。但事实上,问题的根源比这更显而易见。难于进行单元测试的一部分原因是:为了启动 Roboletric 使你的测试单元能够以一种合理的速度运作,你往往需要做许多繁复且没有意义的事。不过在我看来,这并不是关键因素,关键在于:Google 设计的 Android SDK 和 Google 官方提倡的应用架构模式让测试变得困难,在某些情况下,进行测试甚至是不可能的。

这样说可能让人感觉是泛泛而谈,为了证明我的结论,我今天将会用一整个系列的博文来论证我的观点。在接下来的博文里,我会尽可能解释为什么官方鼓励的应用开发方式会使进行单元测试困难,甚至不可能。除此以外,我还会在这些博文里通过重构应用架构探索增强 Android 可测试性的办法,使得应用的代码不需要直接依赖于 Android SDK。通过介绍 Android 怎么让测试变得困难,我会延伸出我想到的重构应用的几个建议,我会在系列博文的第四篇里把它们列出来。

一个(看似)简单的测试示例

要分析这个问题,我们不妨先从一个简单的测试范例开始研究。示例里面用的是 Google IOsched 应用的源码,因为我在第一篇博文里就说道:IOsched 就是 Android 开发者的参考模板。

IOsched 应用里有一个有关 IO 大会细节的页面:

页面里有一个“+”按钮,我们能通过这个按钮将 IO 大会添加日历的对应日期里,如果 IO 大会已经被添加到日历中,“+”按钮则会变成“check”按钮;如果用户再次点击“check”按钮,IO 大会则会从日历中被移除。打开相应的源码我们会发现:这个按钮的状态储存在一个叫作 mStarred 的实例变量里,Activity 会根据按钮的状态启动一个 Service,Service 将在它的 onStop() 方法中将对应的 IO 大会添加到用户的日历中/将对应的 IO 大会从用户的日历中移除。

在这样的代码里写一个测试单元看起来是个明智的选择,但问题就来了:你无法为这样的代码写一个测试单元。你可能就不懂了,为什么不能写啊?别急,为了找到问题所在,我们不妨回到 Android 系统里,并想一想通常情况下,是什么让代码可以被测试。

我们都知道,一个测试单元包含了三个步骤:准备,测试,断言。为了顺利地展示我们测试中的断言步骤,我们需要获得程序在执行测试步骤中的代码后的状态。我们不妨将这个状态称为“测试后状态”。对一个测试单元来说,只有三种办法能获得测试后状态。

第一种办法就是检查测试步骤中执行的方法的返回值,写下这样的测试代码倒不难:

Android 进行单元测试难在哪-part1(android单选)

第二种办法就是获得被测试对象中那些可访问的属性的引用,并断言你取得的这些属性就是你想要的:

第三种办法是检查对象中被注入的依赖的状态。这些依赖或许是不可访问的,但因为你在测试函数中对它们进行注入,那么你肯定会持有它们的引用

那么我们现在已经把能够在一个测试单元中完成断言步骤的方法都列出来了,接下来就该解释为什么刚刚的 Android 代码无法被测试了:答案很简单,我们根本没有办法在 onStop() 方法里完成测试单元的断言步骤——因为 onStop() 方法根本就没有返回值。SessionDetailActivity 里也不存在能帮助我们验证 SessionCalendarService 是否通过正确的命令启动的可访问属性。最后,启动 SessionCalendarService 的代码也不属于注入到 SessionDetailActivity 中的依赖。

你以为这样就完了吗?很不幸,还有比这更糟糕的。在 onStop() 里完成测试单元的准备步骤也是不可能的。为了在测试单元中完成准备步骤,你必须能够改变影响程序测试后状态的所有因素,不妨把它们叫作“预测试状态”。而我们没有任何办法能够在 onStop() 方法里改变预测试状态,所以我们不能对应用进行单元测试。

onStop() 方法里,预测试状态是一个叫作 mInitStarred 的布尔值,你看看写在 onStop() 里的代码估计就懂我的意思了:

这个布尔值会在 ContentProvider 的查询操作完成后通过回调方法被设置:

让 mInitStarred 在 Loader 回调方法里被初始化的原因是:如果用户向他们的日历添加了在 SessionDetailActivity 中选择的 IO 大会事件,那么用于标识该 IO 大会是否被添加到用户日历中的变量将会被存储到数据库中包含所有 IO 大会的表中。我们都知道,Google 希望我们使用 Loader 去获得数据库中的数据,所以 mInitStarred 在 Loader 回调方法里被初始化。

注:但是 Google 自己用 Loader 取数据都非常困难,详见视频:Click me!

mInitStarred 用于判断应用是否必须启动 SessinCalendarService 更新用户的日历,添加/移除某个 IO 大会的信息。如果从数据库中查询获得的信息表示用户的日历已经添加了该大会,而且添加按钮已经转换为“check”,我们就不需要进行任何操作。

所以,如果我们要在 onStop() 方法里进行测试,我们想要确定 onStop() 方法是否真正启动了一个根据“+”按钮的状态去更新用户日历的 Service。但很不幸,mInitStarred 是一个在 Loader 回调中被初始化的私有实例变量,如果要在测试单元里改变它的值,一定会非常麻烦。那我们现在不妨再来想想改变测试单元预测试状态的办法。

第一个办法是,改变影响对象预测试状态的公有属性:

注:改变测试状态的公有属性显然有很多办法。但如果,我举个例子,预测试状态由被测试对象的依赖决定,那我们当然也可以通过改变对象依赖的公有属性改变被测试对象的预测试状态。

第二个办法是改变传入被测试方法的参数:

就像改变测试后状态一样,这两个办法都不能解决我们的问题,所以我们根本不可能在 onStop() 方法里面完成测试单元的准备步骤,问题的根源在于:SessionDetailActivity 里根本不存在能用于改变 mInitStarred 的公有属性。除此以外,onStop() 方法也不会把 mInitStarred 当作一个参数,所以我们无法通过改变 onStop() 的参数值来改变预测试状态。

结论

经过上面一系列源码的分析,每一个希冀进行单元测试的开发者都站在了悬崖的边缘上:即便是 Google 官方所谓的开发模板应用 - IOsched 中一个看起来非常简单的模块,要进行单元测试都困难重重。如果你曾经为了在 Activity 里为你的业务逻辑添加相应的测试单元绞尽脑汁,我现在可以告诉你为什么会这么蛋疼:大部分情况下,要在 Activity 里添加相应的测试单元只是不能实现而已。某些情况下,因为你没有办法改变被测试对象的预测试状态,使得你无法完成单元测试的准备步骤。在另一些情况下,因为你无法访问相关的测试后状态,使得你无法完成对象的断言步骤。而 SessionDetailActivity 的 onStop() 方法就是完美的示例,因为上面的两种情况在它身上体现地淋漓尽致。

Android 里还有许多类似的代码也是无法测试的,这让我不禁怀疑:让我们写下无法测试的代码的罪魁祸首,会不会就是带有 Android 平台特性的应用架构方式。所以在下一篇博文里,我将会深入探索这个假设。

Android应用开发SharedPreferences存储数据的使用方法 1.概述。SharePreferences是用来存储一些简单配置信息的一种机制,使用Map数据结构来存储数据,以键对的方式存储,采用了XML式将数据存储到设备中。例

将android客户端的错误日志压缩上传到服务器 原文:将android客户端的错误日志压缩上传到服务器源代码下载地址:

Unity3D项目文件目录发布后对应的ios/android应用目录 unity3d项目文件目录发布后,对应的ios/android应用目录分类:unity3d--:人阅读评论(0)收藏举报Unity3d的Resource、AssetBundle与手游动态更新的报告,

标签: android单选

本文链接地址:https://www.jiuchutong.com/biancheng/384304.html 转载请保留说明!

上一篇:android(6) 扇形菜单实现(扇形的菜)

下一篇:Android应用开发SharedPreferences存储数据的使用方法(Android应用开发实训)

  • 受票方与付款方不一致
  • 失控发票一定要补税吗
  • 小规模纳税人给个人怎么开发票
  • 财务上用替票合法么
  • 人工成本比例
  • 收到退税如何记账
  • 城市维护建设税是什么意思
  • 房产税计税依据含税还是不含税
  • 持有待售固定资产为什么不计提折旧
  • 企业所得税季报资产总额怎么填
  • 一般纳税人交增值税会计科目
  • 收到高新企业补助款分录
  • 企业计提固定资产折旧
  • 农产品发票有抵税吗
  • 企业哪些可以抵扣增值税
  • 行政事业单位2014年前已交社保费
  • 金税盘提示离线开票时间超限深圳
  • 税收分类编码格式
  • 开餐费发票要开具数量和单价吗
  • 企业税收风险有哪些方面
  • 会计做账中记账表怎么做
  • 未开票的增值税发票能验旧吗
  • 固定资产处置流程
  • 汽车租赁公司产品服务
  • 公司注销要交分红税吗
  • 想买黄金投资怎么做
  • php strlen函数和mb_strlen
  • 职业病治疗费用由谁来承担
  • uefiu盘安装系统步骤win10
  • win7系统中怎样
  • php判断两个给定值相同
  • 废品损失是什么类账户
  • 布拉索夫地图
  • 存放中央银行款项科目按其资金性质
  • 招行网银专业版怎么登陆
  • 事件循环(Event Loop)
  • 未实际发生的费用 可否申请赔偿
  • 世界上最早的计算机是
  • 小狐狸803050
  • 新企业会计准则是哪一年颁布的
  • php操作mysql数据库
  • Css中的color属性用于设置html元素的背景颜色
  • 预算会计的核算对象是什么
  • 小汽车的消费税税率按照气缸容量实行差别
  • 居间协议合同范本完整版
  • 费用化支出期末一般转入哪个账户?
  • 物业管理公司的主管部门是哪个单位
  • 充电口有烧焦味怎么简单解决
  • 合并报表存货的计税基础
  • 金税盘抵扣税款分录怎么做
  • 无形资产比如
  • 会计一般什么时候忙
  • 上市公司股票增发条件
  • 成本法核算的投资收益缴纳企业所得税
  • 经营利润所得税费用
  • 公司员工机票可以抵扣多少呢
  • 公司配股对股价的影响
  • 存货计提存货跌价准备
  • 12月计提的工资没有发完怎么做账
  • 固定资产更换配件 资本化标准
  • 保险赔偿收入如何减税额
  • 无票收入怎么写分录
  • 房地产开发成本和房地产开发产品
  • ubuntu从u盘安装软件
  • 金山卫士电脑版
  • window10怎么获取ip地址
  • win10无法进入休眠状态的原因
  • win7网络连接不见了
  • windows找不到硬盘驱动器
  • win8 设置
  • 电脑出现内部版本7601怎么办
  • Win10 mobile 10586.107升级后会出现哪些bug?
  • js实现上下左右跳动
  • js的文件操作
  • 发票查验结果不一致的原因有哪些
  • 动物大联盟是国内品牌吗
  • 国家税务总局辽宁省税务局
  • 小船载重多少
  • 国家电子税务登录入口
  • 养猪场环保税
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设