1. 程式人生 > 其它 >單元測試(一):我的第一個單元測試

單元測試(一):我的第一個單元測試

一.建立專案並在被測試專案中編寫程式碼

     分別建立兩個類庫專案:被測試專案(Demo)和 測試專案(命名規範:在被專案專案後新增.Tests   Demo.Tests)  

     在Demo中新增如下程式碼:

 1 namespace Demo
 2 {
 3     public class FileVerify
 4     {
 5         public bool IsValidFileName(string filename)
 6         {
 7             if(filename.EndsWith(".txt"))
 8             {
9 return true; 10 } 11 return false; 12 } 13 } 14 }

 

二.安裝測試框架NUit,需要安裝兩個包(下圖所示)

NUint是編寫單元測試框架用的包

NUnit3TestAdapter是適配VS中的測試資源管理器的 

注意:如果只安裝NUnit包,在測試資源管理器中點選執行測試 則不會執行 而會提示:測試未執行

 

三.在Demo.Tests中編寫單元測試

 1 using NUnit.Framework;
 2 
 3 namespace
Demo.Tests 4 { 5 [TestFixture] 6 public class FileVerifyTests 7 { 8 [Test] 9 public void IsValidFileName_BadExtensions_ReturnFalse() 10 { 11 FileVerify fileVerify = MakFileVerify(); 12 13 var result = fileVerify.IsValidFileName("filename.too
"); 14 15 Assert.False(result); 16 } 17 18 19 private FileVerify MakFileVerify() 20 { 21 return new FileVerify(); 22 } 23 } 24 }

1.Demo.Tests專案需先引用專案Demo

2.需引用NUnit.Framework名稱空間

3.新增Attribute

   [TestFixture]:新增到類上,標識該類包含自動化測試 

   [Test]:新增到方法上,標識該方法是一個需要呼叫的自動化測試   

   標註了這兩個Attribute 測試資源管理器中就可以找到該測試方法了

4.測試方法訪問修飾符必須為public 返回值型別必須為void 

5.測試方法名稱【IsValidFileName_BadExtensions_ReturnFalse】 命名三個部分:

    ①【工作單元名】如果工作單元是一個方法,那就是方法名;如果工作單元是一組方法,那名稱需要抽象一點,涵蓋這一組方法。

    ②【測試進行的假設條件】假設條件可以從兩方面來理解:一是描述傳給方法的引數,例如本例中的【BadExtensions】二是描述系統的初始狀態

    ③【預期】對測試方法的預期。測試方法有3中行為:返回一個值(真實值或異常)【例如本例中的RetureFalse】、改變系統狀態、呼叫第三方系統

    然後將①②③用_連線起來  這樣可讀性會很高

6.方法體包含三個行為:建立物件(Line 11) 操作物件(Line 13) 斷言(Line 15)  

   建議:行為之間空一行,並且不在斷言中進行函式呼叫,這樣有良好的可讀性,讓更多的人可以讀懂測試

7.MakeFileVerify方法是一個工廠方法,這樣做的目的是為了:當之後FileVerify的建構函式發生變化後 只需要改動這一個地方  為單元測試的可維護性打下了基礎

8.Assert.False()是NUnit框架中斷言的方法

 

四.對該測試新增正驗證

     對於上面的測試從邏輯上講不是完整的,我們還需要考慮副檔名的大小寫,所以我們新增大寫和小寫字尾的測試,程式碼如下

        [Test]
        public void IsValidFileName_GoodExtensionLower_ReturnTrue()
        {
            FileVerify fileVerify = MakFileVerify();

            var result = fileVerify.IsValidFileName("filename.txt");

            Assert.True(result);
        }

        [Test]
        public void IsValidFileName_GoodExtensionUpper_ReturnTrue()
        {
            FileVerify fileVerify = MakFileVerify();

            var result = fileVerify.IsValidFileName("filename.TXT");

            Assert.True(result);
        }

     然後執行測試,發現大寫測試失敗,測試結果如下

     

     這個時候我們需要修改產品程式碼,修改位置在Line 3,忽略大小寫,程式碼修改如下

1         public bool IsValidFileName(string filename)
2         {
3             if(filename.EndsWith(".txt",StringComparison.CurrentCultureIgnoreCase))
4             {
5                 return true;
6             }
7             return false;
8         }

      這個時候重新執行測試,3個測試就全部通過了

 

五.使用NUnit的【引數化測試】功能重構測試 

 1         [TestCase("filename.txt")]
 2         [TestCase("filename.TXT")]
 3         public void IsValidFileName_GoodExtension_ReturnTrue(string filename)
 4         {
 5             FileVerify fileVerify = MakFileVerify();
 6 
 7             var result = fileVerify.IsValidFileName(filename);
 8 
 9             Assert.True(result);
10         }

     1.將方法上標記的[Test]替換為[TestCase("引數")]

     2.重新命名測試方法將GoodExtension後的大小寫去除,變得更通用

     3.將測試方法引數中定義一個引數filename

     4.把測試中硬編碼的值替換成這個測試方法的引數

     5.把替換掉的值放到[TestCase(param1)]中

     測試執行器會將TestCase括號中的引數賦值給測試方法的引數;可以在一個測試方法上新增多個[TestCase]

六.測試預期異常   保證當異常應該丟擲時,被測試的方法能夠正確的丟擲異常

     當用戶輸入的檔名為null或空時,這個時候應該丟擲ArgumentException異常。如果程式碼沒有丟擲異常,那麼測試就是失敗的

     修改後的產品程式碼為

 1         public bool IsValidFileName(string filename)
 2         {
 3             if (string.IsNullOrWhiteSpace(filename))
 4             {
 5                 throw new ArgumentException();
 6             }
 7             if (filename.EndsWith(".txt", StringComparison.CurrentCultureIgnoreCase))
 8             {
 9                 return true;
10             }
11             return false;
12         }

     新增的測試為

1         [Test]
2         public void IsValidFileName_EmptyFileName_Throw()
3         {
4             FileVerify fileVerify = MakFileVerify();
5 
6             var ex = Assert.Catch<ArgumentException>(() => fileVerify.IsValidFileName(""));
7 
8             StringAssert.Contains("filename has to be provided", ex.Message);
9         }

      這樣一個簡單的測試就完成了!

      如有錯誤之處,請指出!