一、本章需要学会什么?
1、单元测试
- 开发好的系统中存在很多的方法。
- 如何对这些方法的正确性进行测试?
2、反射
- 如何在程序运行时去得到Class对象?
- 然后去获取Class中的每个成分。
3、注解
- 注解是什么?
- 具体是如何在Java程序中解决问题的?
4、动态代理[重难点]
- 框架技术的底层会用到的一些原理知识。
二、单元测试
1、概述
- 单元测试就是针对最小的功能单元编写测试代码,Java程序最小的功能单元是方法。
- 因此,单元测试就是针对Java方法的测试,进而检查方法的正确性。
(1)目前测试方法是如何进行的?存在什么问题?
- 目前我们做过的方法测试就是写一个main方法,调用自己写的方法,验证是否会出现异常。
- 存在问题:
- 只有一个main方法,如果一个方法的测试失败了,其他方法测试会受到影响。
- 无法得到测试的结果报告,需要程序员自己去观察测试是否成功。
- 无法实现自动化测试。
(2)Junit单元测试框架
- JUnit是使用Java语言实现的单元测试框架,它是开源的,Java开发者都应当学习并使用JUnit编写单元测试。
- 此外,几乎所有的IDE工具都集成了JUnit,这样我们就可以直接在IDE中编写并运行JUnit测试,JUnit目前最新版本是5。
(3)JUnit优点
-
JUnit可以灵活的选择执行哪些测试方法,可以一键执行全部测试方法。
-
JUnit可以生成全部方法的测试报告。
-
单元测试中的某个方法测试失败了,不会影响到其他测试方法的测试。
总结
1、Junit单元测试是做什么的?
- 测试类中方法的正确性的。
2、Junit单元测试的优点是什么?
- JUnit可以选择执行哪些测试方法,可以一键执行全部测试方法的测试。
- JUnit可以生成测试报告,如果测试良好则是绿色;如果测试失败,则是红色。
- 单元测试中的某个方法测试失败了,不会影响到其他测试方法的测试。
2、快速入门
(1)需求
- 使用单元测试进行业务方法预期结果、正确性测试的快速入门
(2)分析实现
-
将JUnit的jar包导入到项目中:
-
IDEA工具通常整合好了Junit框架,一般不需要导入,但电脑一定要联网。
-
如果IDEA没有整合好,需要自己手动导入如下2个JUnit的jar包到模块。
-
-
编写测试方法:
-
该测试方法必须是公共的、无参数、无返回值的非静态方法。
-
-
在测试方法上使用@Test注解:
-
标注该方法是一个测试方法。
-
-
在测试方法中完成被测试方法的预期正确性测试。
-
选中测试方法,选择 “JUnit运行”:
-
如果测试良好则是绿色;
-
如果测试失败则是红色。
-
(3)一键全部执行测试
-
一键执行测试类下的所有测试方法
-
一键
(4)源码
package com.app.d1_junit;
/**
模拟业务类
*/
public class UserService {
/**
模拟用户登录业务功能的方法
* @param loginName 接收传入的用户名
* @param passWord 接收传入的密码
* @return 用户名与密码都正确,则返回登录成功,否则返回您的用户名或密码有误
*/
public String loginName(String loginName, String passWord) {
if ("admin".equals(loginName) && "abc123".equals(passWord)) {
return "登录成功";
}else {
return "您的用户名或密码有误";
}
}
/**
模拟查询全部用户名的业务功能的方法
*/
public void selectNames() {
System.out.println(10/0); // 制造一个bug!
System.out.println("查询全部用户名成功~~");
}
}
package com.app.d1_junit;
import org.junit.Assert;
import org.junit.Test;
/**
测试类
*/
public class TestUserService {
/**
测试方法:
1、必须是公共的、无参数无返回值的非静态方法。
2、必须在方法上使用@Test注解: 标注该方法是一个测试方法。
*/
@Test
public void testLoginName() {
UserService userService = new UserService();
String rs = userService.loginName("admin", "abc123");
// 在测试方法中完成被测试方法的预期正确性测试:断言。
/**
参数一:断言失败提示的消息
参数二:期待值: 登录成功(一定要与业务功能代码中的值一样)
参数三:实际值: 测试出来的值
*/
Assert.assertEquals("您的登录业务功能可能出现问题",
"登录成功", rs);
}
@Test
public void testSelectNames() {
UserService userService = new UserService();
userService.selectNames();
}
}
总结
- JUnit单元测试的实现过程是什么样的?
- 必须导入Junit框架的jar包。
- 定义的测试方法必须是公共的、无参数、无返回值的方法。
- 测试方法必须使用@Test注解标记。
- JUnit测试某个方法,测试全部方法怎么处理?成功的标志是什么?
- 测试某个方法可直接右键该方法启动测试。
- 测试全部方法,可选择类或者模块启动测试。
- 红色则是未通过,绿色则是通过,黄色则是无异常但测试结果与你的预期结果不一样。
3、常用注解(JUnit 4.xxxx版本)
注解 | 说明 |
---|---|
@Test | 用来修饰方法为测试方法 |
@Before | 用来修饰实例方法,该方法会在每一个测试方法执行之前执行一次 |
@After | 用来修饰实例方法,该方法会在每一个测试方法执行之后执行一次 |
@BeforeClass | 用来修饰静态方法,该方法会在所有测试方法执行之前只执行一次 |
@AfterClass | 用来修饰静态方法,该方法会在所有测试方法执行之后只执行一次 |
- 开始执行的方法:初始化资源
- 执行完之后的方法:释放资源
package com.app.d1_junit;
/**
模拟业务类
*/
public class UserService {
/**
模拟用户登录业务功能的方法
* @param loginName 接收传入的用户名
* @param passWord 接收传入的密码
* @return 用户名与密码都正确,则返回登录成功,否则返回您的用户名或密码有误
*/
public String loginName(String loginName, String passWord) {
if ("admin".equals(loginName) && "abc123".equals(passWord)) {
return "登录成功";
}else {
return "您的用户名或密码有误";
}
}
/**
模拟查询全部用户名的业务功能的方法
*/
public void selectNames() {
// System.out.println(10/0); // 制造一个bug!
System.out.println("查询全部用户名成功~~");
}
}
package com.app.d1_junit;
import org.junit.*;
/**
测试类
*/
public class TestUserService {
/**
@Before注解:
用来修饰实例方法,该方法会在每一个测试方法执行之前执行一次
*/
@Before
public void before() {
System.out.println("---我是@Before注解修饰的实例方法before,我会在每个测试方法之前执行一次---");
}
/**
@After注解:
用来修饰实例方法,该方法会在每一个测试方法执行之后执行一次
*/
@After
public void after() {
System.out.println("---我是@After注解修饰的实例方法after,我会在每个测试方法之后执行一次---");
}
/**
@BeforeClass注解:
用来修饰静态方法,该方法会在所有测试方法执行之前只执行一次
*/
@BeforeClass
public static void beforeClass() {
System.out.println("---我是@BeforeClass注解修饰的静态方法beforeClass,我会在所有测试方法执行之前执行一次---");
}
/**
@AfterClass注解:
用来修饰静态方法,该方法会在所有测试方法执行之后只执行一次
*/
@AfterClass
public static void afterClass() {
System.out.println("---我是@AfterClass注解修饰的静态方法afterClass,我会在所有测试方法执行之后执行一次---");
}
/**
测试方法:
1、必须是公共的、无参数无返回值的非静态方法。
2、必须在方法上使用@Test注解: 标注该方法是一个测试方法。
*/
@Test
public void testLoginName() {
UserService userService = new UserService();
String rs = userService.loginName("admin", "abc123");
// 在测试方法中完成被测试方法的预期正确性测试:断言。
/**
参数一:断言失败提示的消息
参数二:期待值: 登录成功(一定要与业务功能代码中的值一样)
参数三:实际值: 测试出来的值
*/
Assert.assertEquals("您的登录业务功能可能出现问题",
"登录成功", rs);
}
@Test
public void testSelectNames() {
UserService userService = new UserService();
userService.selectNames();
}
}
4、常用注解(JUnit 5.xxxx版本)
- 作用和JUnit 4.xxxx版本一样的,只是改了方法名称。
注解 | 说明 |
---|---|
@Test | 用来修饰方法为测试方法 |
@BeforeEach | 用来修饰实例方法,该方法会在每一个测试方法执行之前执行一次 |
@AfterEach | 用来修饰实例方法,该方法会在每一个测试方法执行之后执行一次 |
@BeforeAll | 用来修饰静态方法,该方法会在所有测试方法之前只执行一次 |
@AfterAll | 用来修饰静态方法,该方法会在所有测试方法之后只执行一次 |
- 开始执行的方法:初始化资源
- 执行完之后的方法:释放资源