自動化測試框架TestNG


測試框架有很多,比如常用的

  • UI自動化測試框架
  1. java+selenium/appium+testNG/Junit+Maven/Ant/Gradle+Jenkins+MySQL+testlink/redmine
  2. python+selenium/appium+unittest/pytest+Git+Jenkins+MySQL+testlink/redmine
  3. python+rebot framework+unittest/pytest+Git+Jenkins+MySQL+testlink/redmine
  • 接口自動化框架
  1. java+testNG/Junit+Maven/Ant/Gradle+Jenkins+MySQL+testlink/redmine
  2. python+unittest/pytest+Git+Jenkins+MySQL+testlink/redmine
  3. python+rebot framework+unittest/pytest+Git+Jenkins+MySQL+testlink/redmine
  4. jmeter+Maven/Ant+Jenkins+MySQL+testlink/redmine

由於我對java比較熟悉,所以就從TestNG框架開始學習自動化接口測試。和Junit相比,TestNG比較適合測試人員使用,因為

  • TestNG比Junit涵蓋功能更全面的測試
  • Junit更適合隔離性比較強的單元測試
  • TestNG更適合復雜的集成測試

TestNG的官方網址: https://testng.org/doc/index.html 。個人覺得學習一個新的框架,結合着官網的介紹更容易理解。

1. TestNG的引入:

我使用的Eclipse開發工具,所以引入TestNG需要安裝Eclipse的TestNG插件,具體方法請參考https://testng.org/doc/download.html 。 這里給出了不同的開發環境的引入方法。這里就不再詳細敘述,只記錄如何使用TestNG測試框架進行項目的測試。

2. 創建一個Java Project. 在Eclipse中,點擊 File->New->Java Project. 

3.新建一個package,選擇該package,點擊右鍵->TestNG->Create TestNG class. 輸入Class name點擊finish。

 

4.添加TestNG庫

 

5.在包含@Test注解的方法里完成測試過程。運行,並輸出結果。這里簡單的輸出一個字符串。

 

以上是如何使用TestNG框架進行項目的測試。

接下來學習TestNG的基本知識點:注解。在新建TestNG類的時候,可以給該類添加很多注解。

在MyFirstTestNG這個類中f()方法上的注解@Test是最基本的注解,用來將方法標記為測試方法。該注解很多屬性,以下列舉出一些常用的屬性:

 

  • 關於BeforeSuite和AfterSuite的解析如下:

假設有很多TestNG的class,其中一個SuiteConfig.class類中有BeforeSuite和AfterSuite注解的方法, 

 SuiteConfig.class

 1 public class SuiteConfig {
 2     @BeforeSuite
 3     public void beforeSuite() {
 4         System.out.println("BeforeSuite");
 5     }
 6     @AfterSuite
 7     public void afterSuite() {
 8         System.out.println("AfterSuite");
 9     }
10 }

 loginTest.java

 1 public class LoginTest {
 2   @Test
 3   public void loginSuccess() {
 4       System.out.println("Login success test");
 5   } 
 6   @Test
 7   public void loginFailed() {
 8       System.out.println("login failed test");
 9   } 
10   
11   @BeforeMethod
12   public void beforeMethod() {
13       System.out.println("before loginTest");
14   }
15 
16   @AfterMethod
17   public void afterMethod() {
18       System.out.println("after loginTest");
19   }
20 }

 

PayTest.java

 1 public class PayTest {
 2   @Test
 3   public void payTest() {
 4       System.out.println("PayTest");
 5   }
 6   @BeforeMethod
 7   public void beforeMethod() {
 8       System.out.println("before payTest");
 9   }
10 
11   @AfterMethod
12   public void afterMethod() {
13       System.out.println("after payTest");
14   }
15 }

FavTest.java

 1 public class FavTest {
 2   @Test
 3   public void favTest() {
 4       System.out.println("FavTest");
 5   }
 6   
 7   @BeforeMethod
 8   public void beforeMethod() {
 9       System.out.println("before FavTest");
10   }
11 
12   @AfterMethod
13   public void afterMethod() {
14       System.out.println("after FavTest");
15   }
16 }

新建一個suite.xml文件,並將LoginTest和PayTest放在同一個測試套件testSuite中:

 1 <?xml version="1.0" encoding="UTF-8"?>
 2 <suite name="Suite" parallel="false">
 3   <test name="Test">
 4     <classes>
 5       <class name="com.yywang.suite.SuiteConfig"/>
 6       <class name="com.yywang.suite.LoginTest"/>
 7       <class name="com.yywang.suite.PayTest"/>
 8     </classes>
 9   </test> <!-- Test -->
10 </suite> <!-- Suite -->

運行該suite.xml結果如下:

BeforeSuite
before loginTest
login failed test
after loginTest
before loginTest
Login success test
after loginTest
before payTest
PayTest
after payTest
AfterSuite

===============================================
Suite
Total tests run: 3, Failures: 0, Skips: 0
===============================================

由此可以看出,testSuite測試套件可以包含很多測試用例,並且beforeSuite和afterSuite是在所有的測試用例方法運行之前/后運行的。

 

  • 常用注解及屬性的實例
 1  @Test
 2     public void testCase1() {
 3         System.out.println("Test - This is testCase1 with @Test");
 4     }
 5 
 6     @Test
 7     public void testCase2() {
 8         System.out.println("Test - This is testCase2 with @Test");
 9     }
10     
11     @Test(enabled = false)
12     public void ignoreTest() {
13         System.out.println("ignoreTest - This testcase will be ignored");
14     }
15     
16     @Test(expectedExceptions = RuntimeException.class)
17     public void runTimeExceptionFailed() {
18         System.out.println("This is a failed exception test");
19     }
20 
21     @Test(expectedExceptions = RuntimeException.class)
22     public void runTimeExceptionSuccess() {
23         System.out.println("This is a success exception test");
24         throw new RuntimeException();
25     }
26     
27     @Test
28     public void dependedTestCase() {
29         System.out.println("This is depended TestCase");
30     }
31    
32     @Test(dependsOnMethods = {"dependedTestCase"})
33     public void dependTestCase() {
34         System.out.println("This is depend TestCase");
35     }

 

  •  DataProvider實例

 

 1 public class DataProviderTest {
 2 
 3     @Test(dataProvider = "paramter")
 4     public void testDataProvider(String name, int age) {
 5         System.out.println("testDataProvider - name="+name +",age=" + age );
 6     }
 7 
 8     @DataProvider(name="paramter")
 9     public  Object[][] dataProvider() {
10         System.out.println("dataProvider");
11         Object[][] obj = new Object[][] {
12             {"zhangsan", 10},{"lisi", 20},{"wangwu", 30}
13         };
14         return obj;
15     }
16     
17     @Test(dataProvider = "methodParamter")
18     public void testMethodDataProvider1(String name, int age) {
19         System.out.println("testMethodDataProvider1 - name="+name +",age=" + age );
20     }
21     @Test(dataProvider = "methodParamter")
22     public void testMethodDataProvider2(String name, int age) {
23         System.out.println("testMethodDataProvider2 - name="+name +",age=" + age );
24     }
25     
26     @DataProvider(name="methodParamter")
27     public  Object[][] methodDataProvider(Method method) {
28         System.out.println("methodDataProvider");
29         Object[][] obj = null;
30         if(method.getName().equals("testMethodDataProvider1")) {
31             obj = new Object[][] {
32                 {"bo", 10},{"hao", 20},{"yuan", 30}
33             };
34         }else if(method.getName().equals("testMethodDataProvider2")) {
35             obj = new Object[][] {
36                 {"chen", 10},{"long", 20},{"wang", 30}
37             };
38         }
39         return obj;
40     }
41 }

 

  • xml文件實現的多線程測試實例

創建一個java測試類和xml文件

 1 public class MultThreadTest {
 2   @Test
 3   public void test1() {
 4       System.out.println("test1 " + Thread.currentThread().getId());
 5   }
 6   @Test
 7   public void test2() {
 8       System.out.println("test2 " + Thread.currentThread().getId());
 9   }
10   @Test
11   public void test3() {
12       System.out.println("test3 " + Thread.currentThread().getId());
13   }
14 }
 1 <?xml version="1.0" encoding="UTF-8"?>
 2 <suite name="Suite" parallel="tests" thread-count = "2">
 3 <!-- 
 4 parallel="methods": 所有用例都可以在不同的線程執行
 5 
 6 parallel="tests": 不同的test tag下的用例可以在不同的線程執行,相同的test tag下的用例只能在相同的線程執行    
 7 
 8 parallel="classes": 不同的class下的用例可以在不同的線程執行,相同的class下的用例在相同的線程執行
 9 
10 thread-count : 最大並發線程數
11  -->
12   <test name="Test">
13     <classes>
14       <class name="com.yywang.multthread.MultThreadTest"/>
15     </classes>
16   </test> <!-- Test -->
17   
18   
19    <test name="Test2">
20     <classes>
21       <class name="com.yywang.multthread.MultThreadTest"/>
22     </classes>
23   </test> <!-- Test -->
24   
25 </suite> <!-- Suite -->

運行xml文件,可以看到在不同的parallel屬性值下,輸出的線程id是不一樣的。

 

如果喜歡作者的文章,請關注"寫代碼的猿"訂閱號以便第一時間獲得最新內容。本文版權歸作者所有,歡迎轉載. 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM