软件测试 | 参数化测试用例

1.参数化简介

参数化是自动化测试的一种常用技巧,测试人员可以将测试代码中的某些变量的输入使用参数来代替。我们以测试百度搜索功能为例,每次测试搜索功能,都要测试搜索框中输入的不同的搜索内容,在进行这个测试过程中,除了搜索框中的数据在变化,测试的步骤也是重复的,这时就可以使用参数化的方式来解决测试数据变化,测试步骤不变的问题。参数化就是把测试需要用到的参数写到数据集合里,让测试程序自动从这个集合里面取数据,同时每条数据都生成一个对应的测试用例。

2.参数化使用方法

我们使用Appium测试框架编写测试用例时,通常会结合单元测试框架一起使用。使用测试框架的参数化机制,可以减少代码重复。参数化的使用方法是,在测试代码前添加装饰器完成测试数据的传输。示例代码如下(Python版和Java版)。

Python版本

@pytest.mark.parametrize("argvnames",argvalues)

Java版本

  1. @ParameterizedTest

  2. @ValueSource(strings = argvalues)

不同编程语言提供的单元测试框架爱支持的参数传递方式也不一样,但都会在测试用例上添加一个装饰器用以帮助参数化的实现,以Python语言提供的单元测试框架pytest为例,pytest自带了参数化功能,在测试用例上添加参数化需要用到装饰器@pytest.mark.parametrize(),同时需要传入两个参数“argvnames” 与 “argvalues”,第一个参数需要一个或者多个变量来接收列表中的每组数据,第二个参数传递存储数据的列表。测试用例需要使用同名的字符串接收测试数据(与“argvnames” 里面的名字一致),且列表有多少个元素就会生成并执行多个测试用例。下面示例使用参数化定义3组数据,每组数据都存放在一个数据序列中。分别将数据序列传入(test_input,expected)参数中,示例代码如下(Python版和Java版)。

Python版本

 
  1. # content of test_expectation.py

  2. import pytest

  3. @pytest.mark.parametrize("test_input,expected",[("3+5",8),("2+4",6),("6+9",

  4. 42)])

  5. def test_eval(test_input,expected):

  6. assert eval(test_input == expected)

运行结果如下:

 
  1. ...

  2. test_expectation.py ..F

  3. test_input = '6+9',expected = 42

  4. @pytest.mark.parametrize("test_input,expected",

  5. [("3+5",8),("2+4",6),("6+9",42)])

  6. def test_eval(test_input,expected):

  7. > assert eval(test_input) == expected

  8. E AssertionError:assert 54 == 42

  9. E + where 54 = eval('6*9')

  10. test_expection.py:6:AssertionError

Java版本

 
  1. public class BookParamTest{

  2. @ParameterizedTest

  3. @MethodSource("intProvider")

  4. void testWithExplicitLocalMethodSource(int first,int second,int sum){

  5. assertEquals(first + second , sum);

  6. }

  7. static Stream<Arguments> intProvider(){

  8. return Stream.of(

  9. arguments(3,5,8),

  10. arguments(3,5,6),

  11. arguments(6,9,42)

  12. );

  13. }

  14. }

运行结果如下:

 
  1. ...

  2. org.opentest4j.AssertionFailedError:

  3. Expected : 8

  4. Actual : 6

  5. <Click to see difference>

  6. at org.junit.jupiter.api.AssertionUtils.fail(AssertionUtils.java:55)

  7. at org.junit.jupiter.api.AssertionUtils.failNotEqual(AssertionUtils.java:62)

  8. at org.junit.jupiter.api.AssertEquals.assertEquals(AssertEquals.java:150)

  9. at org.junit.jupiter.api.AssertEquals.assertEquals(AssertEquals.java:145)

  10. at org.junit.jupiter.api.Assertions.assertEquals(Assertions.java:527)

  11. ...

从上面的运行结果可以看出,执行的3条测试用例分别对应3组数据,但测试步骤完全相同,只是传入的测试数据发生了变化。

3.案例演示

本测试案例使用“雪球”应用,打开雪球App,点击App页面上的搜索输入框,且在搜索输入框中输入“alibaba”关键词,然后在关键词联想出来的列表里点击“阿里巴巴”项(见图5-19),选择股票分类,获取股票类型为“BABA”的股票价格,最后验证价格在预期的10%范围浮动。

这个案例使用了参数化机制和Hamcrest断言机制,核心代码如下(Python版和Java版。)

(1)Python演示代码

 
  1. from appium import webdriver

  2. import pytest

  3. from hamcrest import *

  4. class TestXueqiu:

  5. #省略

  6. #参数化

  7. @pytest.mark.parametrize("keyword,srock_type,expect_price",[

  8. ('alibaba','BABA',170)

  9. ('xaiomi','01810',8.5)

  10. ])

  11. def test_search self,keyword,stock_type,expect_price):

  12. # 点击搜索项

  13. self.driver.find_element_by_id("home_search").click()

  14. # 向搜索框中输入"keyword"

  15. self.driver.find_element_by_id(

  16. "com.xueqiu.android:id/search_input_text"

  17. ).send_keys(keyword)

  18. #点击搜索结果

  19. self.driver.find_element_by_id("name").click()

  20. #获取价格

  21. price = float(self.driver.find_element_by_xpath(

  22. "//*[contains(@resource-id,'stockCode')\

  23. and @text ='%s']/../../..\

  24. //*[contains(@resource-id,'current_price')]"

  25. % stock_type

  26. ).text )

  27. # 使用断言

  28. assert_that(price,close_to(expect_price,expect_price * 0.1))

  29. ...

(2)Java演示代码

 
  1. import org.junit.jupiter.params.ParameterizedTest;

  2. import org.junit.jupiter.params.provider.Arguments;

  3. import org.junit.jupiter.params.provider.MethodSource;

  4. import java.util.stream.Stream;

  5. import static org.hamcrest.MathcherAssert.assertThat;

  6. import static org.hamcrest.Matchers.closeTo;

  7. import static org.junit.jupiter.params.provider.Arguments.arguments;

  8. public class XueqiuTest{

  9. //省略

  10. @ParameterizedTest

  11. @MethodSource

  12. void testSearch(String keyword,String stockType,float expectPrice){

  13. //点击搜索项

  14. driver.findElement(By.id("home_search")).click();

  15. //向搜索框输入"keyword"

  16. driver.findElement(By.id("com.xueqiu.android:id/search_input_text"\

  17. )).sendKeys(keyword);

  18. //点击搜索结果

  19. driver.findElement(By.id("name")).click();

  20. //获取价格

  21. String format = String.format("//*[contains(@resource-id,\

  22. 'stockCode' and @text='%s')]/../../..//*[contains(@resource-id,\

  23. 'current_price')]",stockType);

  24. String text = driver.findElement(By.xpath format)).getText();

  25. double price = Double.parseDouble(text);

  26. assertThat(price , closeTo(expectPrice,expectPrice * 0.1));

  27. }

  28. static Stream<Arguments> testSearch(){

  29. return Stream.of(

  30. arguments("alibaba","BABA",170),

  31. arguments("xiaomi","01810",8.5)

  32. );

  33. }

  34. }

上面的代码传入了两组测试数据,每组有3个数据,分被为搜索关键词、股票类型和股票价格。在执行测试用例时,分别将两组数据传入测试代码中,用以搜索不同的关键词,并使用Hamcrest实现股票价格的断言。

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

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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值