
本文探讨了在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<Class, RemoteWebDriver> getFunction(Class 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 实例 */@SneakyThrowsstatic Supplier getSupplier(Class 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<Class, RemoteWebDriver>变为Supplier。Lambda表达式: Function的lambda是c -> { … },需要一个输入参数c。而Supplier的lambda是() -> { … },不接受任何参数。参数绑定: 由于Supplier的lambda不接受参数,原先的c.getConstructor()需要改为使用外部“effectively final”的driverClass变量,即driverClass.getConstructor()。
有了getSupplier方法,我们就可以这样使用它来重新初始化WebDriver:
PHP Apache和MySQL 网页开发初步
本书全面介绍PHP脚本语言和MySOL数据库这两种目前最流行的开源软件,主要包括PHP和MySQL基本概念、PHP扩展与应用库、日期和时间功能、PHP数据对象扩展、PHP的mysqli扩展、MySQL 5的存储例程、解发器和视图等。本书帮助读者学习PHP编程语言和MySQL数据库服务器的最佳实践,了解如何创建数据库驱动的动态Web应用程序。
385 查看详情
// 假设 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.<Class, Supplier>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时。此时,可以通过在Map.of()前显式声明泛型参数来帮助编译器,例如:Map.<Class, Supplier>of(…)。
如果觉得显式声明泛型过于冗长,可以创建一个简单的辅助方法来“烘焙”类型信息:
// 辅助方法,用于类型推断static Supplier supply(Supplier 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调用时正确推断泛型。
注意事项与最佳实践
异常处理:
以上就是Java中动态创建WebDriver实例的策略与实践的详细内容,更多请关注创想鸟其它相关文章!
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/982748.html
微信扫一扫
支付宝扫一扫