该文章主要介绍 testNG(testing next generation,下一代测试技术)框架的使用。

1.首先安装testNG

2.安装完成后,创建maven项目,导入TESTNG和selenium依赖。

3.此时就可以直接创建testNG的测试类了

4.下面通过 百度页面打开、输入关键字、进行搜索,来简单演示一下testNG的使用

 package testNGSelenium.testNGDemo;

 import org.testng.annotations.Test;
import org.testng.annotations.BeforeTest;
import org.testng.annotations.DataProvider;
import org.testng.annotations.Parameters; import java.util.Date;
import java.util.concurrent.TimeUnit; import org.apache.bcel.generic.NEW;
import org.apache.commons.collections.map.StaticBucketMap;
import org.openqa.selenium.By;
import org.openqa.selenium.WebDriver;
import org.openqa.selenium.WebElement;
import org.openqa.selenium.chrome.ChromeDriver;
import org.openqa.selenium.support.ui.ExpectedCondition;
import org.openqa.selenium.support.ui.WebDriverWait;
import org.testng.annotations.AfterTest; public class NewTest {
//声明一个全局变量
private WebDriver driver; @Test
@Parameters({"keyWord"})
public void f(String keywords) {
final By input1 = By.id("kw");
final By button1 = By.id("su"); //智能等待页面的输入框加载出来
try { (new WebDriverWait(driver, 3)).until(new ExpectedCondition<Boolean>() { public Boolean apply(WebDriver input) {
WebElement element = input.findElement(input1);
System.out.println(new Date());
return element!= null;
}
}); } catch (Exception e) {
System.out.println("输入框3S还没加载出来!!!");
e.printStackTrace();
} //智能等待页面的搜索按钮加载出来
try {
(new WebDriverWait(driver, 3)).until(new ExpectedCondition<Boolean>() { public Boolean apply(WebDriver input) {
WebElement element = input.findElement(button1);
System.out.println(new Date());
return element!= null;
}
});
} catch (Exception e) {
System.out.println("搜索按钮等待了3秒还未加载出来");
e.printStackTrace();
} driver.findElement(input1).sendKeys(keywords);
driver.findElement(button1).click(); //等待10S,便于观察结果
try {
Thread.sleep(10000);
} catch (InterruptedException e) { e.printStackTrace();
} }
@BeforeTest
public void beforeTest() {
driver = new ChromeDriver();
//窗口最大化
driver.manage().window().maximize(); //加载页面
driver.get("http://www.baidu.com"); //设置页面完全加载时间为3秒,第二个参数TimeUnit.SECONDS是指定时间单位
driver.manage().timeouts().pageLoadTimeout(3, TimeUnit.SECONDS); } @AfterTest
public void afterTest() {
driver.quit();
} }

上面代码中用到了参数化,是需要在xml文件中设置的:

注意:

1.如果使用了参数化,则必须在XML文件上run as TestNG,否则会提示得不到参数的。

2.运行完成后,我们可以在项目目录中看到多了一个test-output

对其中的emailable-report.html右键---open with---web browser,就可以打开如下的测试报告:

关于配置文件的简单介绍:

通过上面的配置文件介绍,你可能会问,如果parallelism设置为true,是不是就可以跑多线程了??

是的!!!激不激动,开不开心!这个多线程可以支持我们做两件事:

1.多个线程来跑同一套件中的不同方法、不同类、不同组件---------------------这样,当我们的测试代码很庞大的时候,会非常节省运行时间

2.用来测试多线程同步问题。举个栗子-----------当有段代码涉及到:不同线程访问同一对象,并且会对该对象作出修改的时候,我们就可以针对这段代码来设计:多个线程同时跑这段代码,来测试下开发的线程同步是不是已经处理好啦。

下面来介绍下TestNG提供的多线程测试的不同使用方法,为了便于理解,这里使用的都是笔者写的最简单的demo:

一:并行的执行某方法

package testNGSelenium.testMethodParallelism;

import org.testng.annotations.Test;
import org.testng.annotations.BeforeMethod;
import org.testng.annotations.AfterMethod; public class MethodParallelismTest {
@Test
public void f1() {
System.out.println("这里是方法1,执行该方法的当前线程ID为"+ Thread.currentThread().getId()); } @Test
public void f2() {
System.out.println("这里是方法2,执行该方法的当前线程ID为"+ Thread.currentThread().getId());
} @BeforeMethod
public void beforeMethod() {
System.out.println("这里是before,执行该方法的当前线程ID为"+ Thread.currentThread().getId()); } @AfterMethod
public void afterMethod() {
System.out.println("这里是after,执行该方法的当前线程ID为"+ Thread.currentThread().getId());
} }

对应修改XML文件为:

该代码的执行结果为:

二:并行的执行同一套间下同一组件中的不同class

package testNGSelenium.testClassesParallelism;

import org.testng.annotations.Test;
import org.testng.annotations.BeforeClass;
import org.testng.annotations.BeforeMethod;
import org.testng.annotations.DataProvider;
import org.testng.annotations.AfterClass;
import org.testng.annotations.AfterMethod; public class ClassesParallelism1 { @Test
public void f() {
System.out.println("test1" + Thread.currentThread().getId());
} @BeforeClass
public void beforeClass() {
System.out.println("beforeClass1" + Thread.currentThread().getId());
} @AfterClass
public void afterClass() {
System.out.println("afterClass2" + Thread.currentThread().getId());
} }
package testNGSelenium.testClassesParallelism;

import org.testng.annotations.AfterClass;
import org.testng.annotations.BeforeClass;
import org.testng.annotations.Test; public class ClassesParallelism2 {
@Test
public void f() {
System.out.println("test2" + Thread.currentThread().getId());
} @BeforeClass
public void beforeClass() {
System.out.println("beforeClass2" + Thread.currentThread().getId());
} @AfterClass
public void afterClass() {
System.out.println("afterClass2" + Thread.currentThread().getId());
} }

对应XML的设置:

该代码的执行结果为:

三:同一套间中的不同组件并行

package testOneSuitTwoTest;

import org.testng.annotations.Test;
import org.testng.annotations.BeforeMethod;
import org.testng.annotations.AfterMethod;
import org.testng.annotations.DataProvider;
import org.testng.annotations.Parameters;
import org.testng.annotations.BeforeClass;
import org.testng.annotations.AfterClass;
import org.testng.annotations.BeforeTest;
import org.testng.annotations.AfterTest;
import org.testng.annotations.BeforeSuite;
import org.testng.annotations.AfterSuite; public class OneSuitTwoModuleTest { @Test
@Parameters({"location"})
public void f(String p) {
System.out.println(p);
System.out.println("Test---thread" + Thread.currentThread().getId());
} @BeforeMethod
public void beforeMethod() {
System.out.println("beforeMethod---thread" + Thread.currentThread().getId());
} @AfterMethod
public void afterMethod() {
System.out.println("afterMethod---thread" + Thread.currentThread().getId());
} /* @DataProvider
public Object[][] dp() {
return new Object[][] { new Object[] { 1, "a" }, new Object[] { 2, "b" }, };
}*/ @BeforeClass
public void beforeClass() {
System.out.println("beforeClass---thread" + Thread.currentThread().getId());
} @AfterClass
public void afterClass() {
System.out.println("afterClass---thread" + Thread.currentThread().getId());
} @BeforeTest
public void beforeTest() {
System.out.println("beforeTest---thread" + Thread.currentThread().getId());
} @AfterTest
public void afterTest() {
System.out.println("afterTest---thread" + Thread.currentThread().getId());
} @BeforeSuite
public void beforeSuite() {
System.out.println("beforeSuite---thread" + Thread.currentThread().getId());
} @AfterSuite
public void afterSuite() {
System.out.println("afterSuite---thread" + Thread.currentThread().getId());
} }

对应的XML设置为:

上面代码的执行结果为;

四:多线程执行某方法

package testNGSelenium.moreThreadTest;

import org.testng.annotations.Test;
import org.testng.annotations.BeforeTest;
import org.testng.annotations.AfterTest; public class MoreThreadTest {
//这里代表 启动三个线程,该方法总共执行9遍,设置整个测试方法的超时时间为2S
@Test(threadPoolSize=3,invocationCount=9,timeOut=2000)
public void f() {
System.out.println("Test---threadId:"+Thread.currentThread().getId());
} @BeforeTest
public void beforeTest() {
} @AfterTest
public void afterTest() {
} }

对应的XML文件并没有特别设置啥参数。

执行结果为:

注意点:

1.如果有童鞋在运行TESTNG套件的时候,遇到了ElementTravelsal找不到的问题,则在自己的pom.xml文件中引入依赖包:xml-apis

<!-- https://mvnrepository.com/artifact/xml-apis/xml-apis -->
<dependency>
<groupId>xml-apis</groupId>
<artifactId>xml-apis</artifactId>
<version>1.4.01</version>
</dependency>

2.如果遇到运行程序之后,在console窗口遇到了乱码:

请打开eclipse.ini文件,并添加: -Dfile.encoding=UTF-8

3.一个xml中间,只能有一个suite标签;

一个suite可以有多个test,test名称必须不同(否则报错:Two tests in the same suite cannot have the same name);

一个test中间可以包含多个class,

一个class里面可以包含多个方法

4.在testNg的 suite file中,添加上

<!DOCTYPE suite SYSTEM "http://testng.org/testng-1.0.dtd">

相当于标签控制,只有合法标签才能输入。否则报错。

5.如果读者使用的开发工具是IDEA,那么久不需要安装testNG的插件了。IDEA是自带了testNG的插件的。只要在maven中添加testNg的包就可以了。

6.老的版本中

<suite name="suitname1" parallel="true" thread-count="2">
parallel的值是没有“true”的。
分为tests,classes,methods级别
tests:不同test tag下的用例可以在不同的线程执行,相同test tag下的用例只能在同一个线程中执行
classes:不同class tag下的用例可以在不同的线程执行,相同class tag下的用例只能在同一个线程中执行。
methods:所有用例都可以在不同的线程去执行。 7.关于dataprovider的并发

https://www.cnblogs.com/znicy/p/6534893.html

最新文章

  1. Linux下使用NDK编译FFMPEG(libstagefright)
  2. [machine learning] Loss Function view
  3. WCF Host中的BaseAddress 和 Endpoint中的Address的区别
  4. set, list 和map知识总结
  5. Hidden Word
  6. 持久层框架之MyBatis
  7. SQL数据开发(经典) 基本操作
  8. 探寻 webpack 插件机制
  9. Vue (一) --- vue.js的快速入门使用
  10. pandas.read_csv() 报错 OSError: Initializing from file failed,报错原因分析和解决方法
  11. Docker使用Dockerfile构建Asp.Net Core镜像
  12. C# 如何保证对象线程内唯一:数据槽(CallContext)【转载】
  13. Tips about Troubleshooting RAC
  14. jenkins安装教程
  15. 纯小白入手 vue3.0 CLI - 2.2 - 组件 home.vue 的初步改造
  16. 微信 audio 获取 duration 为 NaN 的解决方法
  17. [Objective-C语言教程]指针(15)
  18. Android 关于异步Http请求,以及编码问题
  19. Intellij IDEA如何使用Maven Tomcat Plugin运行web项目
  20. 如何设计Spring读取某种文件的逻辑顺序

热门文章

  1. 2018.11.28 OGNL表达式与struts2框架结合的体现---在配置文件中体现(补充)
  2. 2018.10.6 Hibernate配置文件详解-------ORM元数据配置 &amp;&amp;&amp; hibernate主配置文件
  3. 【洛谷P1169】[ZJOI2007]棋盘制作
  4. iOS开发中用到的第三方库概览
  5. OSI七层模型详解(物理层、数据链路层、网络层、传输层.....应用层协议与硬件)
  6. 新疆大学ACM-ICPC程序设计竞赛五月月赛(同步赛)-B-杨老师游戏
  7. 数论(一)LOJ1282
  8. 【TOJ 1449】Area of Circles II(求不同位置的两圆面积之和)
  9. 安装docker和更改docker镜像下载目录
  10. IDEA开发vue.js卡死问题