SoftwareTest7 - JUnit 是个啥 ? 他跟 Selenium 有什么关系 ?

JUnit 是个啥 ? 他跟 Selenium 有什么关系 ?

  • 一 . 相关注解
    • 1.1 @Test
    • 1.2 @BeforeEach
    • 1.3 @BeforeAll
    • 1.4 @AfterEach
    • 1.5 @AfterAll
  • 二 . 断言
    • 2.1 assertEquals
    • 2.2 assertNotEquals
    • 2.3 assertTrue / assertFalse : 断言结果为真/假
    • 2.4 assertNull / assertNotNull : 断言结果为空 / 不为空
  • 三 . 用例的执行顺序
    • 3.1 通过 Order 注解来排序
  • 四 . 参数化
    • 4.1 单参数注解手动编写数据源
    • 4.2 多参数从 csv 文件获取数据来源
    • 4.3 动态参数方法
  • 五 . 测试套件
    • 5.1 指定类来运行包下所有用例
    • 5.2 指定包来运行包下所有用例

Hello , 大家好 , 又给大家带来新的专栏喽 ~

这个专栏是专门为零基础小白从 0 到 1 了解软件测试基础理论设计的 , 虽然还不足以让你成为软件测试行业的佼佼者 , 但是可以让你了解一下软件测试行业的相关知识 , 具有一定的竞争实力 .

那这篇文章 , 就开始 Selenium 实战了 , 我们可以通过代码来操控浏览器来去做一些操作了 , 这个过程非常有趣 , 敬请期待

那也欢迎大家订阅此专栏 : https://blog.csdn.net/m0_53117341/category_12427509.html

希望大家都能够拿到好的 Offer
在这里插入图片描述

我们之前讲过 , 自动化不就是通过 selenium 去实现的吗 , 怎么又引入了个 JUnit 呢 ?
JUnit 是 Java 中的单元测试工具 , 但是我们可以借用 JUnit 库中一些非常好的方法让我们的自动化锦上添花
我们给大家介绍的是 JUnit 5 , 需要我们至少是 JDK 1.8 版本才可以使用
先把 JUnit 的依赖贴在这里

<dependency><groupId>org.junit.jupiter</groupId><artifactId>junit-jupiter</artifactId><version>5.8.2</version><scope>test</scope>
</dependency><dependency><groupId>org.junit.platform</groupId><artifactId>junit-platform-suite</artifactId><version>1.8.2</version><scope>test</scope>
</dependency>

image.png
如果之前使用过 JUnit4 的话 , 那么这次下载 Junit5 可能就会失败
解决方法是找到之前下载好的 JUnit4 依赖 , 删掉之后再重新下载 JUnit5
那引入了 JUnit 之后 , 我们就不需要再创建 main 去调用各个方法了

一 . 相关注解

JUnit 中提供了非常强大的注解功能
我们常用的注解都有 : @Test、@BeforeEach、@BeforeAll、@AfterEach、@AfterAll

1.1 @Test

在方法上面加上 @Test 就代表该方法是测试方法
我们看看加上这个注解有什么用处 ?
image.png
image.png
@Test表示该方法是测试方法 , 执行当前这个类的时候 , 会自动执行该类下所有带 @Test注解的用例

在写自动化的时候 , 我们通常把每个方法都叫做每个自动化的测试用例

image.png

package com.ethan3;import org.junit.jupiter.api.Test;public class junitTest {@Testpublic void testAnnotation() {System.out.println("测试 @Test 注解");}
}

我们通过控制台也能发现好多信息
image.png

1.2 @BeforeEach

@BeforeEach 指的是当我们的方法被 @BeforeEach 修饰的时候 , 就意味着当前的方法需要在每个用例执行之前都执行一次
image.png

package com.ethan3;import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;public class junitTest {@BeforeEachpublic void testAnnotation() {System.out.println("测试 @Test 注解");}@Testpublic void test1() {System.out.println("test1");}@Testpublic void test2() {System.out.println("test2");}
}

1.3 @BeforeAll

@BeforeAll 的作用是当前的方法需要在当前类下所有用例之前执行一次
但是加到方法上面 , 运行还报错了
image.png
它提示我们要想使用 @BeforeAll 的话 , 这个方法需要是静态的

package com.ethan3;import org.junit.jupiter.api.BeforeAll;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;public class junitTest {@BeforeAllpublic static void testAnnotation() {System.out.println("测试 @Test 注解");}@Testpublic void test1() {System.out.println("test1");}@Testpublic void test2() {System.out.println("test2");}
}

image.png
我们也可以参考 类变量 , 整个类只有一个 , 所以只会打印一次

1.4 @AfterEach

@AfterEach 指的是 当前的方法需要在每个用例执行之后都执行一次
image.png

package com.ethan3;import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.BeforeAll;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;public class junitTest {@AfterEachpublic void testAnnotation() {System.out.println("测试 @Test 注解");}@Testpublic void test1() {System.out.println("test1");}@Testpublic void test2() {System.out.println("test2");}
}

1.5 @AfterAll

@AfterAll 指的是 当前的方法在当前类下所有的用例执行之后执行一次
注意 : 被该注解修饰的方法必须是静态的
image.png

二 . 断言

2.1 assertEquals

当我们获取到文本的时候 , 我们可以判断该文本对不对
比如这个案例 :
image.png
那我们就可以通过断言来站在机器的角度上看一下到底是否正确

package com.ethan3;import org.junit.jupiter.api.*;
import org.openqa.selenium.By;
import org.openqa.selenium.chrome.ChromeDriver;public class junitTest {@Testpublic void review() {ChromeDriver driver = new ChromeDriver();driver.get("https://www.baidu.com");String text = driver.findElement(By.cssSelector("#su")).getAttribute("value");// 得到"百度一下"// 假如我们获取到的不是百度一下,而是百度两下// 对于测试人员来说,就是 Bug// 但是对于程序来说,是正确的,因为他并未报错System.out.println(text);// 使用 Assertions.assertEquals 进行断言Assertions.assertEquals("期盼值", text);driver.quit();}
}

如果跟我们设定的预期值不符 , 就会报错 , 测试人员就会立马排查问题

2.2 assertNotEquals

assertNotEquals 的作用就是当实际值和预期值不符的时候 , 程序才不报错 , 运行成功

package com.ethan3;import org.junit.jupiter.api.*;
import org.openqa.selenium.By;
import org.openqa.selenium.chrome.ChromeDriver;public class junitTest {@Testpublic void review() {ChromeDriver driver = new ChromeDriver();driver.get("https://www.baidu.com");String text = driver.findElement(By.cssSelector("#su")).getAttribute("value");// 得到"百度一下"// 假如我们获取到的不是百度一下,而是百度两下// 对于测试人员来说,就是 Bug// 但是对于程序来说,是正确的,因为他并未报错System.out.println(text);// 使用 Assertions.assertEquals 进行断言Assertions.assertNotEquals("期盼值", text);driver.quit();}}

2.3 assertTrue / assertFalse : 断言结果为真/假

package com.ethan3;import org.junit.jupiter.api.*;
import org.openqa.selenium.By;
import org.openqa.selenium.chrome.ChromeDriver;public class junitTest {@Testpublic void assertDemo() {Assertions.assertTrue(1 == 1);}
}

2.4 assertNull / assertNotNull : 断言结果为空 / 不为空

package com.ethan3;import org.junit.jupiter.api.*;
import org.openqa.selenium.By;
import org.openqa.selenium.chrome.ChromeDriver;public class junitTest {@Testpublic void assertDemo2() {Assertions.assertNull("","你好");// 断言失败}@Testpublic void assertDemo3() {Assertions.assertNotNull("","你好");// 断言成功}
}

image.png

三 . 用例的执行顺序

官方网站没有明确说明默认的执行顺序的规则 , 测试用例的执行并不会按照我们编写用例的顺序来执行
image.png
那就会存在一定问题 :
比如我们的博客系统 , 我们想要对登录功能进行测试 , 需要编写测试用例

  1. 检查登录页面展现是否正确
  2. 检查正常登录
  3. 检查异常登陆

如果先执行了第二个测试用例 , 页面就跳转到了博客列表页 . 那接下来再去执行第一条测试用例 , 就肯定会报错
我们只有先检查登录页面展现是否正确之后 , 再去检查第三条

3.1 通过 Order 注解来排序

我们的当前类中 , 有非常多的用例 , 那么在这个类中 , 按照什么顺序来执行呢 ?
我们首先需要在类上加上一个注解 : @TestMethodOrder(MethodOrderer.OrderAnnotation.class)
这个注解的作用是强调(声明)用 order 来进行排序
接下来在每个用例上面加上 @Order(顺序) 来决定每个用例的执行顺序

package com.ethan4;import org.junit.jupiter.api.MethodOrderer;
import org.junit.jupiter.api.Order;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.TestMethodOrder;@TestMethodOrder(MethodOrderer.OrderAnnotation.class)
public class junitTest {@Test@Order(1)public void loginTest() {System.out.println("1::loginTest");}@Test@Order(2)public void editTest() {System.out.println("2::editTest");}@Test@Order(3)public void indexTest() {System.out.println("3::indexTest");}}

image.png
注意 : 用例上面的 @Test 注解是不能删掉的

四 . 参数化

针对于博客系统的登录功能 , 每个用户的用户名和密码是各不相同的 . 这样的话就会导致参数各不相同
这样的话我们就需要写出无穷无尽的方法去解决他
image.png
我们就可以使用参数化的方式来处理各式各样的参数问题 , 尽可能通过一个用例 , 多组参数来模拟用户的行为

4.1 单参数注解手动编写数据源

在使用参数化注解之前 , 需要先声明该方法为参数化方法 , 使用注解 : @ParameterizedTest
那这个参数从哪里来 , 我们还需要声明一个注解 : @ValueSource(strings = {“甄嬛”,“沈眉庄”,“安陵容”})
image.png
这就是我们的参数化 , 通过参数化注解 , 我们就把参数传递给测试用例中 , 供测试用例使用

package com.ethan4;import org.junit.jupiter.api.MethodOrderer;
import org.junit.jupiter.api.Order;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.TestMethodOrder;
import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.ValueSource;
import org.openqa.selenium.By;
import org.openqa.selenium.chrome.ChromeDriver;@TestMethodOrder(MethodOrderer.OrderAnnotation.class)
public class junitTest {@ParameterizedTest@ValueSource(strings = {"甄嬛","沈眉庄","安陵容"})public void sparamTest(String name) {System.out.println("name:" + name);}
}

那这个 @ValueSource() 到底是怎么回事 ?
@ValueSource(数据类型方法 = {“参数1”,“参数2”,“参数3”…})
其中 , 数据类型支持 : short、byte、int、long、double、char…
image.png
这几个数据类型在 JUnit 中不单单是简单的数据类型 , JUnit 将这些数据类型封装成了一个方法
我们回到 IDEA 也可以看一下
image.png
但是 @ValueSource() 只支持单参数

4.2 多参数从 csv 文件获取数据来源

package com.ethan4;import org.junit.jupiter.api.MethodOrderer;
import org.junit.jupiter.api.Order;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.TestMethodOrder;
import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.CsvSource;
import org.junit.jupiter.params.provider.ValueSource;
import org.openqa.selenium.By;
import org.openqa.selenium.chrome.ChromeDriver;public class junitTest {@ParameterizedTest@CsvSource({"甄嬛,20", "沈眉庄,25", "安陵容,26"})public void moreParamsTest(String name,int age) {System.out.println("name:" + name + " age:" + age);}}

image.png


那要是我们有非常多的数据 , 我们难不成要写非常多的 “” , 这肯定不行
我们就可以在其他文件写好 , 让我们的注解直接调用到这个文件 , 执行里面的数据
我们创建一个 csv 文件 , 在里面写好数据
image.png
然后使用 @CsvFileSource(files = “路径名”) 导入文件

package com.ethan4;import org.junit.jupiter.api.MethodOrderer;
import org.junit.jupiter.api.Order;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.TestMethodOrder;
import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.CsvFileSource;
import org.junit.jupiter.params.provider.CsvSource;
import org.junit.jupiter.params.provider.ValueSource;
import org.openqa.selenium.By;
import org.openqa.selenium.chrome.ChromeDriver;@TestMethodOrder(MethodOrderer.OrderAnnotation.class)
public class junitTest {@ParameterizedTest // 表示该方法是参数化方法@CsvFileSource(files = "D:\\study\\JavaCode\\selenium_demo\\test.csv") // 导入第三方文件public void csvFileParamsTest(String name,int age) {System.out.println("name:" + name + " age:" + age);}}

image.png
如果发生乱码导致报错的话 , 手动创建一个 txt 文件 , 按照这种格式来编辑数据 , 然后再修改后缀名为 csv

zhangsan,18
lisi,20
wangwu,26

4.3 动态参数方法

通过动态方法提供数据源 , 构造出动态的数据

package com.ethan4;import org.junit.jupiter.api.MethodOrderer;
import org.junit.jupiter.api.Order;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.TestMethodOrder;
import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.*;
import org.openqa.selenium.By;
import org.openqa.selenium.chrome.ChromeDriver;import java.util.stream.Stream;public class junitTest {@ParameterizedTest // 先声明该方法是参数化方法@MethodSource("methodParams") // 指定动态数据来源public void dynamicParamsTest(String name,int age) {System.out.println("name:" + name + " age:" + age);}/*** 1. 方法必须是静态的* 2. 返回值是 Stream<Arguments>* 3. 返回 Stream.of 对象* @return* @throws InterruptedException*/static Stream<Arguments> methodParams() throws InterruptedException {// 构造动态参数String[] arr = new String[5];for (int i = 0; i < arr.length; i++) {Thread.sleep(500);arr[i] = System.currentTimeMillis() + "";}return Stream.of(Arguments.arguments(arr[0],20),Arguments.arguments(arr[1],20),Arguments.arguments(arr[2],20),Arguments.arguments(arr[3],20),Arguments.arguments(arr[4],20));}}

image.png
我们运行一下
image.png


那我们把刚才写好的数据来源注释掉 , 然后编写一个新的数据来源 , 与我们打印的用例同名 , 但是上面的 @MethodSource() 注解 , 我们就不写括号里面的数据来源了 , 这样也是可以的
image.png

package com.ethan4;import org.junit.jupiter.api.MethodOrderer;
import org.junit.jupiter.api.Order;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.TestMethodOrder;
import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.*;
import org.openqa.selenium.By;
import org.openqa.selenium.chrome.ChromeDriver;import java.util.stream.Stream;@TestMethodOrder(MethodOrderer.OrderAnnotation.class)
public class junitTest {@ParameterizedTest // 先声明该方法是参数化方法@MethodSource // 指定动态数据来源public void dynamicParamsTest(String name,int age) {System.out.println("name:" + name + " age:" + age);}static Stream<Arguments> dynamicParamsTest() throws InterruptedException {// 构造动态参数String[] arr = new String[5];for (int i = 0; i < arr.length; i++) {Thread.sleep(500);arr[i] = System.currentTimeMillis() + "";}return Stream.of(Arguments.arguments(arr[0],20),Arguments.arguments(arr[1],20),Arguments.arguments(arr[2],20),Arguments.arguments(arr[3],20),Arguments.arguments(arr[4],20));}
}

也是可以运行的
image.png


最后注意一点 , 使用了参数化注解的方法不能再用 @Test 注解了 , 也就意味着 @Test 只能作用在非参数化的用例上
image.png

五 . 测试套件

为了模拟测试套件 , 我们新创建三个类 , 每个类中都写上一些最基础的测试方法
image.png
我们之前都是运行一个类中的所有方法 , 那我们要是想运行多个类中的多个方法 , 意思就是我们怎么一起运行 test1 test2 test3 呢 ?
我们就可以使用测试套件

5.1 指定类来运行包下所有用例

我们首先需要创建一个测试套件 , 实际上就是创建一个新的类 , 加上注解 : @Suite , 标识这是一个测试套件类而不是测试类
image.png
接下来 , 我们需要指定我们需要跑哪些类 , 使用注解 : @SelectClasses() , 里面使用 {} 把我们需要测试的类加进去
image.png
运行一下
image.png
那如果我们 @SelectClasses() 里面的某一个类里面的某个用例没加 @Test , 那么就不会执行这个方法
image.png
我们刚刚还介绍了参数化的用例 , 他并没有用 @Test 注解啊 ?
其实参数化用例 , 测试套件也是能扫描到的
image.png

5.2 指定包来运行包下所有用例

使用 : @SelectPackages("包的路径) 来指定要执行的方法集合
image.png
运行一下
image.png
他为什么只跑了 junitTest 了呢 ?
这是因为我们指定包来运行范围的话 , 文件名命名规则需要以 Test/Tests 结尾 (T 还得必须大写)
image.png

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/178302.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

在Spring Boot中使用进程内缓存和Cache注解

在Spring Boot中使用内缓存的时候需要预先知道什么是内缓存&#xff0c;使用内缓存的好处。 什么是内缓存 内缓存&#xff08;也称为进程内缓存或本地缓存&#xff09;是指将数据存储在应用程序的内存中&#xff0c;以便在需要时快速访问和检索数据&#xff0c;而无需每次都从…

如何用SaleSmartly集成WhatsApp账号(内含WhatsApp个人号、商业号、API号对比图)

用SaleSmartly集成WhatsApp账号 如果企业有多个WhatsApp账号&#xff0c;无论是个人账号还是工作账号&#xff0c;员工操作起来可能会觉得难以管理和切换。SaleSmartly就可以解决这个问题&#xff0c;让员工在一个平台上同时使用多个WhatsApp账号&#xff0c;不需要频繁地登录和…

sqlmap requires ‘python-pymysql‘ third-party library

使用sqlmap进行udf提权报错&#xff1a; [14:06:04] [CRITICAL] sqlmap requires python-pymysql third-party library in order to directly connect to the DBMS MySQL. You can download it from https://github.com/PyMySQL/PyMySQL. Alternative is to use a package pyt…

关于Java抽象类和接口的总结和一点个人的看法

꒰˃͈꒵˂͈꒱ write in front ꒰˃͈꒵˂͈꒱ ʕ̯•͡˔•̯᷅ʔ大家好&#xff0c;我是xiaoxie.希望你看完之后,有不足之处请多多谅解&#xff0c;让我们一起共同进步૮₍❀ᴗ͈ . ᴗ͈ ა 本文由xiaoxieʕ̯•͡˔•̯᷅ʔ 原创 CSDN 如需转载还请通知˶⍤⃝˶个人主页&am…

【Linux】Linux基础IO(上)

​ ​&#x1f4dd;个人主页&#xff1a;Sherry的成长之路 &#x1f3e0;学习社区&#xff1a;Sherry的成长之路&#xff08;个人社区&#xff09; &#x1f4d6;专栏链接&#xff1a;Linux &#x1f3af;长路漫漫浩浩&#xff0c;万事皆有期待 上一篇博客&#xff1a;【Linux】…

Linux-查询目录下包含的目录数或文件数

1. 前置 1&#xff09;ls Linux最常用的命令之一&#xff0c;列出该目录下的包含内容。 -l&#xff1a;use a long listing format-以列表的形式展现 -R&#xff1a;list subdirectories recursively-递归列出子目录 2&#xff09;| 管道符 将上一条命令的输出&#xff…

Java内存结构

1.对象的结构 一个Java对象在内存中包括3个部分&#xff1a;对象头、实例数据和对齐填充 2.虚拟机存储数据的方式 2.1小端存储 : 便于数据之间的类型转换&#xff0c;例如:long类型转换为int类型时&#xff0c;高地址部分的数据可以直接截掉。 2.2大端存储 : 便于数据类型…

LeetCode反转链表的五种Java实现方式

给你单链表的头节点 head &#xff0c;请你反转链表&#xff0c;并返回反转后的链表。 示例 1&#xff1a; 输入&#xff1a;head [1,2,3,4,5] 输出&#xff1a;[5,4,3,2,1]示例 2&#xff1a; 输入&#xff1a;head [1,2] 输出&#xff1a;[2,1]示例 3&#xff1a; 输入&a…

14——1

这句话的意思是&#xff0c;如图中月份12天数23时&#xff0c;就是1223&#xff1b;当月份9天数2时&#xff0c;就是0902. 可以看到在上面给出的数组元素中&#xff0c;并没有连续挨在一起的2023数字元素——就有人可能输出答案0。 所以这里要看一下—— ——子序列的含义&…

AM驱动架构—优质Mini-LED显示技术解决方案

MiniLED背光驱动方案作为一种新兴的显示技术方案&#xff0c;具有更高的亮度、更广的色域范围和更低的功耗等优势&#xff1b;以其出色的性能和广泛的应用前景备受关注是实现MiniLED显示效果的重要环节。 MiniLED背光驱动方案是指通过控制MiniLED芯片的电流和亮度&#xff0c;…

JavaScript语法、语句、数据类型

一、JavaScript语法&#xff1a; 1、JavaScript字面量&#xff1a; JavaScript中的固定值称为字面量。数字字面量可以是整数、小数或者科学计数&#xff08;e&#xff09;,如3.1415926、1008、123e5等&#xff1b;字符串字面量可以使用单引号或者双引号&#xff0c;如“corli…