Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动

上几篇利用数据驱动配合POM搭建了一个自动化测试框架雏形,其实目前已经满足我们的需要了。但除了数据驱动之外,测试界还有一种广泛利用的驱动方式,叫做关键字驱动。这篇我们就讨论一下它。

先说什么是关键字。在写测试程序时,我们免不了要对页面上的元素进行操作,比如输入、点击、刷新等等。之前我们都是以代码的形式表现这些操作的,输入是调用sendKeys()方法,点击是click(),刷新是refresh()。通常执行一个test case经常需要调用这几个方法。比如我们还是用示例网站演示一下登录步骤,测试步骤如下:

1. 打开http://cslm-test.com/hrsystem/index.php;
2. 输入用户名"1001";
3. 输入密码"123";
3. 点击登录按钮(断言点:home按钮出现在home.php页);
4. 退出(断言点:登录按钮出现在index.php页)。

程序如下:

Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动_第1张图片
Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动_第2张图片

sendKeys()和click()各调用两次。这还是一个很简单的测试程序,试想如果测试步骤足够多,那我们会一直不停地来回使用它们。有人就觉得烦了,他们想如果每种操作只写一遍就好了,这样只要根据不同情况传入不同的参数就行。比如上面例子中的两次输入操作,它们都用sendKeys(),只不过参数不同,一个是员工代号,一个是密码。这两个参数的元素位置不同,使用的测试数据也不同:

Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动_第3张图片

注意划红线的部分,咱们介绍POM那篇说过,我们用object来表示网页元素的定位过程,所以我就也把object标上了。除了object和测试数据不同,定位器也有可能不一样,只不过这个例子中恰好都用id定位器。看出点端倪来了吧,只要根据不同情况替换这三个地方的东西就可以了。总结一下,一个相同点:它们都是输入操作;三个不同点: 定位器,object,和测试数据。

如果把这一个相同点三个不同点提取出来放在一个文件里并写成程序,是不是可以用下面这个格式表达呢?

Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动_第4张图片

我们的做法是逐行扫描文件,如果第一列是“输入操作”,则把后三列的值拿出来拼到下面的driver.findElement()语句中,如果不是则跳过。很明显,这两行都满足条件,第一行读完后输入了员工代号,第二行读完后输入了密码。整个过程中我们把重复调用sendKeys()的过程替换成了读文件的过程,然后根据情况传参数而已,这样driver.findElement(...).sendKeys()只用写一遍。是不是我们的目的达到了?文件中第一列的条件,也可以说指令,就是所谓的关键字。

这两行数据顺序不能乱,颠倒了就变成先输密码再输员工代码了,就不对了。所以,设计一个关键字驱动的文件一定要遵循test case的执行顺序。我现在把例子中所有的操作都变成关键字写在文件里:

Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动_第5张图片

从第二行打开浏览器,一直到最后一行的验证,每一行都是一个操作,而且它们都按照顺序写下来。注意,不是每个关键字都需要后面三列,有些操作比如像点击,就不需要最后一列测试数据,因为不需要输入任何东西;打开最大化浏览器窗口这些操作甚至什么参数都不用。设计文件时要具体步骤具体分析。按照这种思路,我们可以把它变成代码:

Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动_第6张图片

看这段伪代码。在一行一行读文件时,如果当前行的第一列关键字是“Input Text”,那就调用enter方法,该方法传递四个参数,后三个正好是文件后三列的数据,也就是定位器、object、测试数据。因为定位器可能是8种中的任意一种,所以需要用条件语句再进行判断。同理,如果当前行的第一列是别的关键字,那就调用别的相关方法。看到了吧,这里面最关键的就是关键字与元素操作的匹配:

Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动_第7张图片

所以,最终程序如下:

Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动_第8张图片
Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动_第9张图片
Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动_第10张图片
Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动_第11张图片
Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动_第12张图片

第50行开始对文件进行逐行扫描,每次取一行查看第一列,我用变量keyword表示。如果keyword匹配下面任意一个关键字,则调用相关操作的方法。所有操作方法在第117行到第165行。注意第168行,由于每个关键字触发的操作都需要判断定位器,所以我把定位器判断的过程单独提出来放到getObject()方法中,返回元素对应的object。执行一下:

Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动_第13张图片

这就是一个简单的关键字驱动示例,你可以用把自动化测试中任何一个操作写成一个关键字。看完这个例子,想必大家也明白了关键字驱动,原理无非是把元素操作与关键字匹配,然后根据test case的步骤在文件中罗列需要的关键字,最后根据不同的操作分别处理。牢记两个关键点:第一,用关键字驱动时文件中装的是test case,关键字排列顺序要与测试步骤一致;第二,关键字与元素操作匹配。

讨论数据驱动时还介绍了page object model,关键字驱动也可以用POM设计模式。我们给它改一下。首先项目结构如下:

Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动_第14张图片

先把元素的object提取出来写在LoginPage.properties中:

Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动_第15张图片

然后把读取object、定位器判断、操作方法都挪到LoginPage.java中:

Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动_第16张图片
Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动_第17张图片
Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动_第18张图片

主类Test.java中只剩下读取关键字和关键字匹配过程:

Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动_第19张图片
Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动_第20张图片
Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动_第21张图片

通过实例化LoginPage得到的对象调用相关方法完成test case。再次执行一下:

Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动_第22张图片

以上就是关键字驱动的介绍,原理就是把元素操作与关键字匹配,然后根据test case的步骤在文件中罗列需要的关键字,最后根据不同的操作分别处理。数据驱动和关键字驱动各有各的特点,数据驱动通过把测试数据从test case中分离避免了test case的重写,而关键字驱动则通过把元素操作与关键字匹配从而减少了测试步骤了重写,代码量大大下降。市面上用得比较多的测试框架就是基于关键字驱动设计的,有些甚至不用你写代码就可以使用,因为每一个关键字背后的代码人家都替你写好了。比如有一个叫Robot Framework的框架就能实现这点,而且还很受欢迎。如果感兴趣请看我的Robot Framework + Selenium2Library文集继续。

这篇文章的源代码是SeleniumExcelKeywordDrivenPOM项目。

本篇知识点及注意事项:
1. 关键字驱动测试的原理就是把元素操作与关键字匹配,然后根据test case的步骤在文件中罗列需要的关键字,最后根据不同的操作分别处理。
2. 用关键字驱动时文件中装的是test case,关键字排列顺序要与测试步骤一致。

你可能感兴趣的:(Selenium Web Driver自动化测试(java版)系列下半部分(35) - 关键字驱动)