3

GlobalVariables 类包含在我的框架中使用的不同变量,其中一个是 WebDriver 实例:

  public class GlobalVariables
        {
        public static WebDriver driver;
    //Some other static global variables required across my framework
        public GlobalVariables(String propertiesFile)
        {
        initializeVariables(propertiesFile);
        }
        public void initializeVariables(String propertiesFile)
        {
        GlobalInitializer obj=new GlobalInitializer();
        obj.initialize(String propertiesFile);
        }
   }

GlobalInitializer 包含初始化所有 GlobalVariables 的方法:

public class GlobalInitializer extends GlobalVariables
{
public void initialize(String propertiesFile)
{
  //Some logic to read properties file and based on the properties set in it, call other initialization methods to set the global variables.
}
public void initializeDriverInstance(String Browser)
{
driver=new FireFoxDriver();
}

//其他一些初始化其他全局变量的方法。}

我有许多 GetElement 类,它们使用驱动程序实例来获取 UI 控制元素,例如:

public class GetLabelElement extends GlobaleVariables
{
public static WebElement getLabel(String someID)
{
return driver.findElement(By.id(someId));
}
//Similar methods to get other types of label elements.
}

public class GetTextBoxElement extends GlobaleVariables
{
public static WebElement getTextBox(String someXpath)
{
return driver.findElement(By.xpath(someXpath));
}
//Similar methods to get other types of text box elements.
}

我有其他类在 UI 控件上执行一些操作(这些类也使用全局变量)例如:

public class GetLabelProperties extends GlobalVariables
{
public static String getLabelText(WebElement element)
{
return element.getText();
}
}

public class PerformAction extends GlobalVariables
{
public static void setText(String textBoxName,String someText)
{
driver.findElement(someLocator(textBoxName)).setText("someText");
}
    //Some other methods which may or may not use the global variables to perform some action
}

我在 testng 中的测试类如下所示:

public class TestClass
{
GlobalVariables globalObj=new GlobalVariables(String propertiesFile);
@Test(priority=0)
{
GlobalVariables.driver.get(someURL);
//Some assertion.
}
@Test(priority=1)
{
WebElement element=GetLabelElement.getLabel(someID);
String labelName=GetLabelProperties.getLabelText(element);
//Some assertion.
}
@Test(priority=2)
{
WebElement element=GetTextBoxElement.getTextBox(someXpath);
PerformAction.setText(element.getText(),someText);
//Some assertion.
}
}

我有基于场景的类似的多个测试类。现在,如果我单独运行这些测试,它们运行良好。但是当我尝试并行运行它们时,这些测试会以某种奇怪的方式失败。在分析时,我发现它是由每个测试初始化​​的静态全局变量,从而使其他测试失败。现在我应该如何实现我的目标,即并行运行多个测试而对我的框架设计进行最小的更改?我已经尝试搜索选项,并且遇到了一些选项,即 1) 使用同步。2)创建ThreadLocal实例(注意:我已经尝试过这个解决方案但仍然是同样的问题。测试相互混淆导致失败。我已将 WebDriver 实例标记为 ThreadLocal 并覆盖了 ThreadLocal 的 initialValue 方法来初始化驱动程序实例。我仍然不确定我是否正确实施了它。)。现在我不确定如何在给定的场景中最好地实施任何一种解决方案。任何帮助表示赞赏。蒂亚!

4

4 回答 4

4

我找到了解决方案:使用 ThreadLocal 是在大型多线程环境中运行测试的最佳解决方案。在多线程环境中使用 WebDriver 的代码片段:

public static ThreadLocal<WebDriver> driver;
driver=new ThreadLocal<WebDriver>()
                {
                    @Override
                    protected WebDriver initialValue()
                    {
                        return new FirefoxDriver(); //You can use other driver based on your requirement.
                    }
                };

现在每次创建测试线程时都会打开一个新的浏览器。ThreadLocal 将确保每个线程只有一个静态 webdriver 实例副本。[注意:确保您的其他全局变量也是 ThreadLocals。在我的情况下,他们不是我遇到测试错误问题的原因]。我想分享一些额外的知识,以便其他人可能会发现它的信息。在 ThreadLocal 中,每当调用 ThreadLocal.get() 方法时,您必须确保有一个用于初始化线程本地的规定,如上面 initialValue() 方法中所示,否则您可能会遇到空指针异常。感谢大家。

于 2013-09-29T06:40:56.943 回答
2

如果您打算非并行运行,那么使用静态 webdriver 成员(或通过引用传递在测试类之间共享的实例)很好,因为它是不必在测试类之间关闭 webdriver 实例的好方法。但是,如果您想并行运行,则需要为每个线程拥有一个 webdriver 实例,因此在这种情况下,使用静态成员是错误的方法。相反,您需要在调用测试用例类时创建或传递 webdriver 实例。

此外,您正在为测试的每个步骤将测试分解为单独的测试。这是非常不寻常的,我看不出你这样做的原因。您可以像人们通常所做的那样,通过将所有测试步骤保留在一个单一的测试用例中来真正简化您的测试。

于 2013-09-29T03:18:36.537 回答
1

你得到这个是因为 JVM 如何处理静态成员和方法。

如果要并行运行,则不能拥有静态 webdriver 对象。

资料来源:我在我工作的地方实施的自动回归系统 - 我们遇到了这个问题。

于 2013-09-29T03:10:24.237 回答
1

你可以试试这样的

公共类 DriverManager {

private static final ThreadLocal<WebDriver> threadLocal = new ThreadLocal<WebDriver>();

public static WebDriver getDriver() {
    return threadLocal.get();
}

public static void setDriver(WebDriver driver) {
    threadLocal.set(driver);
}

public static void closeDriver() {
    if (getDriver() != null) {
        try {
            getDriver().close();
        } catch (Exception e) {
            e.printStackTrace();
        }
        try {
            getDriver().quit();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
    threadLocal.remove();
}

}

于 2016-10-03T03:37:16.543 回答