
本文探讨了在java中,尤其是在web浏览器意外崩溃后,如何高效且鲁棒地动态重新初始化同类型`webdriver`实例的策略。文章重点介绍了如何利用java 8的`supplier`函数式接口和构造函数引用,替代复杂的`function`映射和条件判断,实现简洁、资源友好的对象创建,从而提升代码的可读性和可维护性。
在自动化测试框架中,当WebDriver实例因各种原因(如浏览器崩溃)失效时,重新初始化一个与之前类型相同的WebDriver实例是常见的需求。传统的做法可能涉及大量的if-else语句来判断当前WebDriver的类型并创建相应的实例。然而,Java 8引入的函数式编程特性为我们提供了更优雅、更具扩展性的解决方案。
问题场景与初始尝试
假设我们有一个RemoteWebDriver实例driver,它可能是ChromeDriver、EdgeDriver或FirefoxDriver等。当driver意外崩溃后,我们需要根据其原始类型创建一个新的实例。最初的尝试可能类似于以下结构,试图使用Map来避免冗长的条件判断,并使用Function来延迟实例的创建:
// 假设 originalDriver 是已经崩溃的 WebDriver 实例
// RemoteWebDriver originalDriver = ...;
Map.of(
ChromeDriver.class, getFunction(ChromeDriver.class),
EdgeDriver.class, getFunction(EdgeDriver.class),
FirefoxDriver.class, getFunction(FirefoxDriver.class),
OperaDriver.class, getFunction(OperaDriver.class)
)
.entrySet().stream()
.filter((e) -> e.getKey().isInstance(originalDriver))
.map((e)->e.getValue().identity()) // 这里的 identity() 调用是错误的
.findFirst()
.orElseThrow(() -> new RuntimeException("WebDriver type not detected"));
// 辅助方法,用于返回创建实例的 Function
@SneakyThrows // 假设使用了 Lombok 简化异常处理
static Function, RemoteWebDriver> getFunction(Class extends RemoteWebDriver> driverClass){
return c -> {
try {
return c.getConstructor().newInstance();
} catch (IllegalAccessException | InstantiationException e) {
throw new RuntimeException(e);
}
};
} 上述代码中,e.getValue().identity()的调用是错误的,因为getFunction返回的是一个Function,它需要一个参数来执行其逻辑(即apply方法),而不是identity方法。此外,对于无参数构造器创建对象的需求,Function接口可能不是最合适的选择,因为它通常用于接受一个输入并产生一个输出的场景。
优化方案一:使用 Supplier 接口
针对上述问题,Java 8的Supplier函数式接口更为契合。Supplier接口代表一个“供应者”,它不接受任何参数,但会返回一个结果。这完美符合我们延迟创建对象的需求。
立即学习“Java免费学习笔记(深入)”;
我们可以将getFunction方法修改为返回Supplier
import java.util.function.Supplier; import org.openqa.selenium.remote.RemoteWebDriver; import lombok.SneakyThrows; // 假设使用了 Lombok 简化异常处理 /** * 获取一个 Supplier,用于创建指定类型的 RemoteWebDriver 实例。 * Supplier 延迟执行,只有在调用 get() 时才创建实例。 * @param driverClass 要创建的 WebDriver 类的 Class 对象 * @return 一个 Supplier 实例,调用其 get() 方法将返回一个新的 RemoteWebDriver 实例 */ @SneakyThrows static SuppliergetSupplier(Class extends RemoteWebDriver> driverClass){ return () -> { try { // 使用 driverClass 绑定到 lambda 表达式,在 Supplier.get() 调用时创建实例 return driverClass.getConstructor().newInstance(); } catch (ReflectiveOperationException e) { // 更通用的反射异常捕获 throw new RuntimeException("Failed to create WebDriver instance for class: " + driverClass.getName(), e); } }; }
关键改变点:
-
返回类型: 从Function
, RemoteWebDriver>变为Supplier 。 - Lambda表达式: Function的lambda是c -> { ... },需要一个输入参数c。而Supplier的lambda是() -> { ... },不接受任何参数。
- 参数绑定: 由于Supplier的lambda不接受参数,原先的c.getConstructor()需要改为使用外部“effectively final”的driverClass变量,即driverClass.getConstructor()。
有了getSupplier方法,我们就可以这样使用它来重新初始化WebDriver:
本书全面介绍PHP脚本语言和MySOL数据库这两种目前最流行的开源软件,主要包括PHP和MySQL基本概念、PHP扩展与应用库、日期和时间功能、PHP数据对象扩展、PHP的mysqli扩展、MySQL 5的存储例程、解发器和视图等。本书帮助读者学习PHP编程语言和MySQL数据库服务器的最佳实践,了解如何创建数据库驱动的动态Web应用程序。
// 假设 originalDriver 是已经崩溃的 WebDriver 实例
// RemoteWebDriver originalDriver = new ChromeDriver(); // 示例
// ... (originalDriver 崩溃)
RemoteWebDriver newDriver = Map.of(
ChromeDriver.class, getSupplier(ChromeDriver.class),
EdgeDriver.class, getSupplier(EdgeDriver.class),
FirefoxDriver.class, getSupplier(FirefoxDriver.class),
OperaDriver.class, getSupplier(OperaDriver.class)
)
.entrySet().stream()
.filter((e) -> e.getKey().isInstance(originalDriver)) // 查找与崩溃实例类型匹配的条目
.map((e)->e.getValue().get()) // 调用 Supplier 的 get() 方法来创建新实例
.findFirst()
.orElseThrow(() -> new RuntimeException("WebDriver type not supported or detected"));
// 现在 newDriver 就是一个与 originalDriver 同类型的新实例
// System.out.println("New driver instance created: " + newDriver.getClass().getSimpleName());优化方案二:直接使用构造函数引用
更进一步,Java 8允许我们直接使用构造函数引用(Constructor References)作为Supplier。这意味着我们可以省去getSupplier辅助方法,直接将构造函数引用放入Map中,使代码更加简洁。
ChromeDriver::new就是ChromeDriver类的无参构造函数的引用,它本质上是一个Supplier
import java.util.Map; import java.util.function.Supplier; import org.openqa.selenium.chrome.ChromeDriver; import org.openqa.selenium.edge.EdgeDriver; import org.openqa.selenium.firefox.FirefoxDriver; import org.openqa.selenium.opera.OperaDriver; import org.openqa.selenium.remote.RemoteWebDriver; // 假设 originalDriver 是已经崩溃的 WebDriver 实例 // RemoteWebDriver originalDriver = new ChromeDriver(); // 示例 // ... (originalDriver 崩溃) // 明确指定 Map 的泛型类型,帮助编译器进行类型推断 RemoteWebDriver newDriver = Map., Supplier extends RemoteWebDriver>>of( ChromeDriver.class, ChromeDriver::new, EdgeDriver.class, EdgeDriver::new, FirefoxDriver.class, FirefoxDriver::new, OperaDriver.class, OperaDriver::new ) .entrySet().stream() .filter((e) -> e.getKey().isInstance(originalDriver)) .map((e)->e.getValue().get()) .findFirst() .orElseThrow(() -> new RuntimeException("WebDriver type not supported or detected"));
这种方法最为推荐,因为它:
- 简洁明了: 直接表达了“创建这个类的实例”的意图。
- 类型安全: 编译器可以更好地推断类型。
- 资源高效: 同样是延迟创建,只有在get()被调用时才执行构造函数。
关于泛型类型推断的提示:
在某些情况下,直接使用Map.of()时,编译器可能难以推断出正确的泛型类型,特别是当值类型是Supplier extends RemoteWebDriver>时。此时,可以通过在Map.of()前显式声明泛型参数来帮助编译器,例如:Map.
如果觉得显式声明泛型过于冗长,可以创建一个简单的辅助方法来“烘焙”类型信息:
// 辅助方法,用于类型推断
static Supplier extends RemoteWebDriver> supply(Supplier extends RemoteWebDriver> s) {
return s;
}
// 使用辅助方法后的 Map 创建
RemoteWebDriver newDriver = Map.of(
ChromeDriver.class, supply(ChromeDriver::new),
EdgeDriver.class, supply(EdgeDriver::new),
FirefoxDriver.class, supply(FirefoxDriver::new),
OperaDriver.class, supply(OperaDriver::new)
)
.entrySet().stream()
.filter((e) -> e.getKey().isInstance(originalDriver))
.map((e)->e.getValue().get())
.findFirst()
.orElseThrow(() -> new RuntimeException("WebDriver type not supported or detected"));这个supply方法实际上没有做任何事情,它的唯一作用是提供一个类型明确的上下文,帮助编译器在Map.of调用时正确推断泛型。
注意事项与最佳实践
- 异常处理:









