Java 8 开始引入了许多很酷的新特性。实际上,Java 每6个月就会发布一个新版本!这么多新功能,很难跟上更新的脚步。
这些新特性旨在改进 Java 代码冗长的问题,具体方法就是减少样板代码。
下面是自动化测试中用到的 Java 8 至 Java 12 一些新特性,通过Todo示例应用展示。
1. 创建集合
Java 9 为List、Set和Map这样的集合类加入了of()静态工厂方法,让创建列表变得更容易。
在 Java 9 之前,要创建任务列表tasks可以这样做:
List tasks = Arrays.asList("pay mortgage bill", "make dentist appointment", "get car washed");
这段代码看起来并不可怕,比起创建List对象然后循环调用add()方法已经改进了不少。
然而,Java 9 可以用List.of()。不仅可读性更强,依赖的import更少,而且更容易记住:
List tasks = List.of("pay mortgage bill", "make dentist appointment", "get car washed");
使用of()创建集合的一个缺点,创建的集合不可变。这意味着集合创建后不能添加、删除或者排序。
2. 局部变量类型推断
Java 代码冗长其中一个原因是需要显式声明变量或者对象类型。当类型包含范型操作符<>或者类名很长时,代码看起来很枯燥。
Java 10 引入了局部变量类型推断,用var声明变量。使用var必须在声明的时候初始化变量,因为 Java 需要根据赋值语句进行类型推断。
例如,可以用var替代List声明tasks。由于使用List.of()建立字符串列表并赋值,Java 知道tasks的类型是List。
var tasks = List.of( "pay mortgage bill", "make dentist appointment", "get car washed");
我非常钟爱var,这是我最喜欢的 Java 新特性。写 Java 代码从此不再枯燥。尽管如此,还是会提醒自己不要得意忘形,代码的可读性依然很重要。如果程序员自己都不能根据赋值语句判断出类型,那么最好还是直接写上变量类型,让代码变得更清晰易懂。
3. 函数式编程新特性
Java 8 加入了许多函数式编程新特性,比如 lambda 表达式。在集合框架中新加入的forEach()方法中可以使用。
在此之前,可以使用传统的for循环在 Todo 应用中添加task:
var tasks = List.of("pay mortgage bill", "make dentist appointment", "get car washed");
for(int i=0; i
todosPage.addTask(tasks.get(i));
}
也可以使用改进版for循环,看起来简洁了一点:
var tasks = List.of("pay mortgage bill", "make dentist appointment", "get car washed");
for(String task: tasks){
todosPage.addTask(task);
}
Java 8 中的forEach()支持函数式编程,遍历集合更简洁。可以像下面这样遍历list:
tasks.forEach(
forEach()方法内部可以写一个 lambda 表达式。
lambda 表达式以当前集合元素名task开始,变量可以使用1个字母的短名。由于是局部final变量,因此不能在方法外使用。实际开发中会使用t,这里使用了全名。
tasks.forEach(task
接着是->操作符与在元素上执行的操作:
tasks.forEach(task -> todosPage.addTask(task));
实际上,这里只有1个变量与1个待执行的操作。可以用object::method方式来缩短表达式,不用为列表的每个元素声明局部变量,同样的代码可以更简洁:
tasks.forEach(todosPage::addTask);
然而,如果需要对集合中的项目执行多个操作,可以直接在->后面使用大括号
tasks.forEach(task -> {
todosPage.addTask(task);
//TODO: 在 task 上执行更多操作
});
所以,就上面的例子来说,最终代码如下:
var tasks = List.of("pay mortgage bill","make dentist appointment","get car washed");
tasks.forEach(todosPage::addTask);
4. Stream
添加任务后,接下要进行验证。看看如何获取所有task。
Java 8 之前,需要编写代码查找所有WebElement对象,然后循环遍历获取每个task的文本。迭代过程中,需要将String添加到一个新列表然后返回:
public List getAllTasks(){
List elementList = driver.findElements(tasksLocator);
List taskList = new ArrayList();
for(WebElement element : elementList){
taskList.add(element.getText());
}
return taskList;
}
这段代码相当冗长了。幸运的是 Java 8 引入了Stream,处理集合数据变得更简单。
为 WebDriver 的findElements()返回的 Web 元素调用stream()方法:
driver.findElements(tasksLocator).stream()
Stream提供了许多方法可供选择,这里使用map()对集合中的每个元素执行函数,即从元素获取文本,例如WebElement.getText():
driver.findElements(tasksLocator).stream().map(WebElement::getText)
上面的代码会返回应用函数后的结果Stream,其中包含了针对每个元素调用WebElement.getText()后的String值。
这里希望返回String列表,因此调用collect()。
public List getAllTasks(){
return driver.findElements(tasksLocator).stream().map(WebElement::getText).collect(Collectors.toList());
}
可以看到,比起之前的方法代码精简了许多。
5. switch 表达式
Java 12 引入了一种switch新用法。
比如,在代码中进行 API 调用。Java 12 之前,我们会使用switch语句决定调用哪个API:
public static Response execute(String endpoint, HttpMethod method){
Response response = null;
switch(method){
case GET:
response = request.get(endpoint);
break;
case POST:
response = request.post(endpoint);
break;
}
response.then().log().all();
return response;
}
每种情况下,都需要执行一个赋值操作,同时带上一个break语句。
有了 Java 12 带来的switch新用法,可以大大简化代码,可以根据switch处理的结果直接为变量赋值。
以上面的代码为例,可以将每个表达式赋值给response。在switch内部,把:换成->不需要原来的break与default语句:
public static Response execute(String endpoint, HttpMethod method){
Response response = switch(method){
case GET -> request.get(endpoint);
case POST -> request.post(endpoint);
}
response.then().log().all();
return response;
}
完整源代码在 GitHub 上
github.com/angiejones/new-java-features-test-automation