"unittest" --- 單元測試框架
***************************

**源代码：** Lib/unittest/__init__.py

======================================================================

（假如你已經熟悉相關基礎的測試概念，你可能會希望跳過以下段落，直接參考
assert 方法清單。）

"unittest" 原生的單元測試框架最初由 JUnit 開發，和其他程式語言相似有主
要的單元測試框架。支援自動化測試，對測試分享安裝與關閉程式碼，集合所有
匯總的測試，並且獨立各個測試報告框架。

"unittest" 用來作為實現支援一些重要的物件導向方法的概念。

test fixture
   *test fixture* 表示为了开展一项或多项测试所需要进行的准备工作，以及
   所有相关的清理操作。举个例子，这可能包含创建临时或代理的数据库、目
   录，再或者启动一个服务器进程。

test case（測試用例）
   一個 *test case* 是一個獨立的單元測試。這是用來確認一個特定設定的輸
   入的特殊回饋。 "unittest" 提供一個基礎類別，類別 "TestCase"，可以用
   來建立一個新的測試條例。

test suite（測試套件）
   *test suite* 是一個搜集測試條例，測試套件，或是兩者皆有。它需要一起
   被執行並用來匯總測試。

test runner（測試執行器）
   *test runner* 是一個編排測試執行與提供結果給使用者的一個元件。執行
   器可以使用圖形化介面，文字介面或是回傳一個特別值用來標示出執行測試
   的結果。

也參考:

  "doctest" 模組
     另一個執行測試的模組，但使用不一樣的測試方法與規範。

  Simple Smalltalk Testing: With Patterns
     Kent Beck 的原始論文討論使用 "unittest" 這樣模式的測試框架。

  pytest
     第三方单元测试框架，提供轻量化的语法来编写测试，例如："assert
     func(10) == 42"。

  The Python Testing Tools Taxonomy
     一份詳細的 Python 測試工具列表，包含 functional testing 框架和
     mock object 函式庫。

  Testing in Python Mailing List
     一個專門興趣的群組用來討論 Python 中的測試方式與測試工具。

  随 Python 源代码分发的脚本 "Tools/unittestgui/unittestgui.py" 是一个
  一个用于发现和执行测试的用户图形界面工具。这为了极大地方便单元测试新
  手使用。建议在生产环境中测试应由一个持续集成系统来启动，如
  Buildbot, Jenkins 或者`Travis-CI <https://travis-ci.com>`_, 或者
  AppVeyor。


簡單範例
========

"unittest" 模組提供一系列豐富的工具用來建構與執行測試。本節將展示這一
系列工具中一部份，它們已能滿足大部份使用者需求。

這是一段簡短的腳本用來測試 3 個字串方法：

   import unittest

   class TestStringMethods(unittest.TestCase):

       def test_upper(self):
           self.assertEqual('foo'.upper(), 'FOO')

       def test_isupper(self):
           self.assertTrue('FOO'.isupper())
           self.assertFalse('Foo'.isupper())

       def test_split(self):
           s = 'hello world'
           self.assertEqual(s.split(), ['hello', 'world'])
           # check that s.split fails when the separator is not a string
           with self.assertRaises(TypeError):
               s.split(2)

   if __name__ == '__main__':
       unittest.main()

測試用例 (testcase) 可以透過繼承 "unittest.TestCase" 類別來建立。這裡
定義了三個獨立的物件方法，名稱皆以 "test" 開頭。這樣的命名方式能告知
test runner 哪些物件方法為定義的測試。

每個測試的關鍵為呼叫 "assertEqual()" 來確認是否為期望的結果；
"assertTrue()" 或是 "assertFalse()" 用來驗證一個條件式；
"assertRaises()" 用來驗證是否觸發一個特定的 exception。使用這些物件方
法來取代 "assert" 陳述句，將能使 test runner 收集所有的測試結果並產生
一個報表。

通过 "setUp()" 和 "tearDown()" 方法，可以设置测试开始前与完成后需要执
行的指令。 在  组织你的测试代码 中，对此有更为详细的描述。

最後將顯示一個簡單的方法去執行測試 "unittest.main()" 提供一個命令執行
列介面測試腳本。當透過命令執行列執行，輸出結果將會像是：

   ...
   ----------------------------------------------------------------------
   Ran 3 tests in 0.000s

   OK

在測試時加入 "-v" 選項將指示 "unittest.main()" 提高 verbosity 層級，產
生以下的輸出：

   test_isupper (__main__.TestStringMethods) ... ok
   test_split (__main__.TestStringMethods) ... ok
   test_upper (__main__.TestStringMethods) ... ok

   ----------------------------------------------------------------------
   Ran 3 tests in 0.001s

   OK

以上的例子顯示大多數使用 "unittest" 特徵足以滿足大多數日常測試的需求。
接下來第一部分文件的剩餘部分將繼續探索完整特徵設定。


命令執行列介面 (Command-Line Interface)
=======================================

單元測試模組可以透過命令執行列執行測試模組，物件甚至個別的測試方法：

   python -m unittest test_module1 test_module2
   python -m unittest test_module.TestClass
   python -m unittest test_module.TestClass.test_method

你可以通過一個串列與任何模組名稱的組合，完全符合類別與方法的名稱。

測試模組可以根據檔案路徑指定：

   python -m unittest tests/test_something.py

這允許你使用 shell 檔案名稱補完功能 (filename completion) 來指定測試模
組。給定的檔案路徑必須亦能被當作模組 import。此路徑轉換為模組名稱的方
式為移除 '.py' 並將路徑分隔符 (path separator) 轉換成 '.'。 假如你的測
試檔案無法被 import 成模組，你應該直接執行該測試檔案。

通過增加 -v 的旗標數，可以在你執行測試時得到更多細節（更高的 verbosity
）：

   python -m unittest -v test_module

若執行時不代任何引數，將執行 Test Discovery（測試探索）：

   python -m unittest

列出所有命令列選項：

   python -m unittest -h

3.2 版更變: 在早期的版本可以個別執行測試方法和不需要模組或是類別。


命令列模式選項
--------------

**unittest** supports these command-line options:

-b, --buffer

   Standard output 與 standard error stream 將在測試執行被緩衝
   (buffer)。這些輸出在測試通過時被丟棄。若是測試錯誤或失則，這些輸出
   將會正常地被印出，並且被加入至錯誤訊息中。

-c, --catch

   "Control-C" 測試執行過程中等待正確的測試結果並回報目前為止所有的測
   試結果。第二個 "Control-C" 拋出一般例外 "KeyboardInterrupt"。

   參照 Signal Handling 針對函式提供的功能。

-f, --failfast

   在第一次錯誤或是失敗停止執行測試。

-k

   只运行匹配模式或子串的测试方法和类。可以多次使用这个选项，以便包含
   匹配子串的所有测试用例。

   包含通配符（*）的模式使用 "fnmatch.fnmatchcase()" 对测试名称进行匹
   配。另外，该匹配是大小写敏感的。

   模式对测试加载器导入的测试方法全名进行匹配。

   例如，"-k foo" 可以匹配到 "foo_tests.SomeTest.test_something"  和
   "bar_tests.SomeTest.test_foo" ，但是不能匹配到
   "bar_tests.FooTest.test_something" 。

--locals

   透過 traceback 顯示本地變數。

3.2 版新加入: 增加命令列模式選項 "-b" 、 "-c" 與 "-f"。

3.5 版新加入: 命令列選項 "--locals"。

3.7 版新加入: 命令行选项 "-k" 。

對執行所有的專案或是一個子集合測試，命令列模式可以可以被用來做測試探索
。


Test Discovery（測試探索）
==========================

3.2 版新加入.

單元測試支援簡單的 test discovery（測試探索）。為了相容於測試探索，所
有的測試檔案都要是模組或是套件（包含 *namespace packages*），並能從專
案的最上層目錄中 import（代表它們的檔案名稱必須是有效的 identifiers）
。

Test discovery（測試探索）實作在 "TestLoader.discover()"，但也可以被用
於命令列模式。基本的命令列模式用法如下：

   cd project_directory
   python -m unittest discover

備註:

  "python -m unittest" 作為捷徑，其功能相當於 "python -m unittest
  discover"。假如你想傳遞引數至探索測試的話，一定要明確地加入
  "discover" 子指令。

"discover" 子指令有以下幾個選項：

-v, --verbose

   詳細（verbose）輸出

-s, --start-directory directory

   開始尋找的資料夾（預設為 "."）

-p, --pattern pattern

   匹配測試檔案的模式（預設為 "test*.py"）

-t, --top-level-directory directory

   專案的最高階層目錄 (defaults to start directory)

"-s", "-p", 和 "-t" 選項依照傳遞位置作為引數排序順序。以下兩個命令列被
視為等價:

   python -m unittest discover -s project_directory -p "*_test.py"
   python -m unittest discover project_directory "*_test.py"

正如可以传入路径那样，传入一个包名作为起始目录也是可行的，如
"myproject.subpackage.test" 。你提供的包名会被导入，它在文件系统中的位
置会被作为起始目录。

警示:

  探索性测试通过导入测试对测试进行加载。在找到所有你指定的开始目录下的
  所有测试文件后，它把路径转换为包名并进行导入。如 "foo/bar/baz.py" 会
  被导入为 "foo.bar.baz" 。如果你有一个全局安装的包，并尝试对这个包的
  副本进行探索性测试，可能会从错误的地方开始导入。如果出现这种情况，测
  试会输出警告并退出。如果你使用包名而不是路径作为开始目录，搜索时会假
  定它导入的是你想要的目录，所以你不会收到警告。

测试模块和包可以通过 load_tests protocol 自定义测试的加载和搜索。

3.4 版更變: Test discovery supports *namespace packages* for start
directory. Note that you need to the top level directory too. (e.g.
"python -m unittest discover -s root/namespace -t root").


组织你的测试代码
================

单元测试的构建单位是 *test cases* ：独立的、包含执行条件与正确性检查的
方案。在 "unittest" 中，测试用例表示为 "unittest.TestCase" 的实例。通
过编写 "TestCase" 的子类或使用 "FunctionTestCase" 编写你自己的测试用例
。

一个 "TestCase" 实例的测试代码必须是完全自含的，因此它可以独立运行，或
与其它任意组合任意数量的测试用例一起运行。

"TestCase" 的最简单的子类需要实现一个测试方法（例如一个命名以 "test"
开头的方法）以执行特定的测试代码：

   import unittest

   class DefaultWidgetSizeTestCase(unittest.TestCase):
       def test_default_widget_size(self):
           widget = Widget('The widget')
           self.assertEqual(widget.size(), (50, 50))

可以看到，为了进行测试，我们使用了基类 "TestCase" 提供的其中一个
"assert*()" 方法。若测试不通过，将会引发一个带有说明信息的异常，并且
"unittest" 会将这个测试用例标记为测试不通过。任何其它类型的异常将会被
当做错误处理。

可能同时存在多个前置操作相同的测试，我们可以把测试的前置操作从测试代码
中拆解出来，并实现测试前置方法 "setUp()" 。在运行测试时，测试框架会自
动地为每个单独测试调用前置方法。

   import unittest

   class WidgetTestCase(unittest.TestCase):
       def setUp(self):
           self.widget = Widget('The widget')

       def test_default_widget_size(self):
           self.assertEqual(self.widget.size(), (50,50),
                            'incorrect default size')

       def test_widget_resize(self):
           self.widget.resize(100,150)
           self.assertEqual(self.widget.size(), (100,150),
                            'wrong size after resize')

備註:

  多个测试运行的顺序由内置字符串排序方法对测试名进行排序的结果决定。

在测试运行时，若 "setUp()" 方法引发异常，测试框架会认为测试发生了错误
，因此测试方法不会被运行。

相似的，我们提供了一个 "tearDown()" 方法在测试方法运行后进行清理工作。

   import unittest

   class WidgetTestCase(unittest.TestCase):
       def setUp(self):
           self.widget = Widget('The widget')

       def tearDown(self):
           self.widget.dispose()

若 "setUp()" 成功运行，无论测试方法是否成功，都会运行 "tearDown()" 。

这样的一个测试代码运行的环境被称为 *test fixture* 。一个新的 TestCase
实例作为一个测试脚手架，用于运行各个独立的测试方法。在运行每个测试时，
"setUp()" 、"tearDown()" 和 "__init__()" 会被调用一次。

建议你根据所测试的功能，将测试用 TestCase 实现集合起来。"unittest" 为
此提供了机制：*test suite*，以 "unittest" 的类 "TestSuite" 为代表。大
部分情况下，调用 "unittest.main()" 即可，并且它会为你集合所有模块的测
试用例并执行。

然而，如果你需要自定义你的测试套件的话，你可以参考以下方法组织你的测试
：

   def suite():
       suite = unittest.TestSuite()
       suite.addTest(WidgetTestCase('test_default_widget_size'))
       suite.addTest(WidgetTestCase('test_widget_resize'))
       return suite

   if __name__ == '__main__':
       runner = unittest.TextTestRunner()
       runner.run(suite())

你可以把测试用例和测试套件放在与被测试代码相同的模块中（比如
"widget.py"），但将测试代码放在单独的模块中（比如 "test_widget.py"）有
几个优势。

* 测试模块可以从命令行被独立调用。

* 更容易在分发的代码中剥离测试代码。

* 降低没有好理由的情况下修改测试代码以通过测试的诱惑。

* 测试代码应比被测试代码更少地被修改。

* 被测试代码可以更容易地被重构。

* 对用 C 语言写成的模块无论如何都得单独写成一个模块，为什么不保持一致
  呢？

* 如果测试策略发生了改变，没有必要修改源代码。


复用已有的测试代码
==================

一些用户希望直接使用 "unittest" 运行已有的测试代码，而不需要把已有的每
个测试函数转化为一个 "TestCase" 的子类。

因此， "unittest" 提供 "FunctionTestCase" 类。这个 "TestCase" 的子类可
用于打包已有的测试函数，并支持设置前置与后置函数。

假定有一个测试函数：

   def testSomething():
       something = makeSomething()
       assert something.name is not None
       # ...

可以创建等价的测试用例如下，其中前置和后置方法是可选的。

   testcase = unittest.FunctionTestCase(testSomething,
                                        setUp=makeSomethingDB,
                                        tearDown=deleteSomethingDB)

備註:

  可以用 "FunctionTestCase" 快速将现有的测试转换成基于 "unittest" 的测
  试，但不推荐。花点时间继承 "TestCase" 会让以后重构测试无比轻松。

在某些情况下，现有的测试可能是用 "doctest" 模块编写的。 如果是这样，
"doctest" 提供了一个 "DocTestSuite" 类，可以从现有基于 "doctest" 的测
试中自动构建 "unittest.TestSuite" 用例。


跳过测试与预计的失败
====================

3.1 版新加入.

Unittest 支持跳过单个或整组的测试用例。它还支持把测试标注成“预期失败”
的测试。这些坏测试会失败，但不会算进 "TestResult" 的失败里。

要跳过测试只需使用 "skip()" *decorator* 或其附带条件的版本，在
"setUp()" 内部使用 "TestCase.skipTest()"，或是直接引发 "SkipTest"。

跳过测试的基本用法如下:

   class MyTestCase(unittest.TestCase):

       @unittest.skip("demonstrating skipping")
       def test_nothing(self):
           self.fail("shouldn't happen")

       @unittest.skipIf(mylib.__version__ < (1, 3),
                        "not supported in this library version")
       def test_format(self):
           # Tests that work for only a certain version of the library.
           pass

       @unittest.skipUnless(sys.platform.startswith("win"), "requires Windows")
       def test_windows_support(self):
           # windows specific testing code
           pass

       def test_maybe_skipped(self):
           if not external_resource_available():
               self.skipTest("external resource not available")
           # test code that depends on the external resource
           pass

在啰嗦模式下运行以上测试例子时，程序输出如下：

   test_format (__main__.MyTestCase) ... skipped 'not supported in this library version'
   test_nothing (__main__.MyTestCase) ... skipped 'demonstrating skipping'
   test_maybe_skipped (__main__.MyTestCase) ... skipped 'external resource not available'
   test_windows_support (__main__.MyTestCase) ... skipped 'requires Windows'

   ----------------------------------------------------------------------
   Ran 4 tests in 0.005s

   OK (skipped=4)

跳过测试类的写法跟跳过测试方法的写法相似:

   @unittest.skip("showing class skipping")
   class MySkippedTestCase(unittest.TestCase):
       def test_not_run(self):
           pass

"TestCase.setUp()"  也可以跳过测试。可以用于所需资源不可用的情况下跳过
接下来的测试。

使用 "expectedFailure()" 装饰器表明这个测试预计失败。:

   class ExpectedFailureTestCase(unittest.TestCase):
       @unittest.expectedFailure
       def test_fail(self):
           self.assertEqual(1, 0, "broken")

你可以很容易地编写在测试时调用 "skip()" 的装饰器作为自定义的跳过测试装
饰器。 下面这个装饰器会跳过测试，除非所传入的对象具有特定的属性:

   def skipUnlessHasattr(obj, attr):
       if hasattr(obj, attr):
           return lambda func: func
       return unittest.skip("{!r} doesn't have {!r}".format(obj, attr))

以下的装饰器和异常实现了跳过测试和预期失败两种功能：

@unittest.skip(reason)

   跳过被此装饰器装饰的测试。 *reason*  为测试被跳过的原因。

@unittest.skipIf(condition, reason)

   当 *condition* 为真时，跳过被装饰的测试。

@unittest.skipUnless(condition, reason)

   跳过被装饰的测试，除非 *condition* 为真。

@unittest.expectedFailure

   Mark the test as an expected failure or error.  If the test fails
   or errors it will be considered a success.  If the test passes, it
   will be considered a failure.

exception unittest.SkipTest(reason)

   引发此异常以跳过一个测试。

   通常来说，你可以使用 "TestCase.skipTest()" 或其中一个跳过测试的装饰
   器实现跳过测试的功能，而不是直接引发此异常。

被跳过的测试的 "setUp()"  和 "tearDown()" 不会被运行。被跳过的类的
"setUpClass()" 和 "tearDownClass()" 不会被运行。被跳过的模组的
"setUpModule()" 和 "tearDownModule()" 不会被运行。


使用子测试区分测试迭代
======================

3.4 版新加入.

当你的几个测试之间的差异非常小，例如只有某些形参不同时，unittest 允许
你使用 "subTest()" 上下文管理器在一个测试方法体的内部区分它们。

例如，以下测试:

   class NumbersTest(unittest.TestCase):

       def test_even(self):
           """
           Test that numbers between 0 and 5 are all even.
           """
           for i in range(0, 6):
               with self.subTest(i=i):
                   self.assertEqual(i % 2, 0)

可以得到以下输出:

   ======================================================================
   FAIL: test_even (__main__.NumbersTest) (i=1)
   ----------------------------------------------------------------------
   Traceback (most recent call last):
     File "subtests.py", line 32, in test_even
       self.assertEqual(i % 2, 0)
   AssertionError: 1 != 0

   ======================================================================
   FAIL: test_even (__main__.NumbersTest) (i=3)
   ----------------------------------------------------------------------
   Traceback (most recent call last):
     File "subtests.py", line 32, in test_even
       self.assertEqual(i % 2, 0)
   AssertionError: 1 != 0

   ======================================================================
   FAIL: test_even (__main__.NumbersTest) (i=5)
   ----------------------------------------------------------------------
   Traceback (most recent call last):
     File "subtests.py", line 32, in test_even
       self.assertEqual(i % 2, 0)
   AssertionError: 1 != 0

如果不使用子测试，程序遇到第一次错误之后就会停止。而且因为``i``的值不
显示，错误也更难找。

   ======================================================================
   FAIL: test_even (__main__.NumbersTest)
   ----------------------------------------------------------------------
   Traceback (most recent call last):
     File "subtests.py", line 32, in test_even
       self.assertEqual(i % 2, 0)
   AssertionError: 1 != 0


类与函数
========

本节深入介绍了 "unittest" 的 API。


测试用例
--------

class unittest.TestCase(methodName='runTest')

   "TestCase" 类的实例代表了 "unittest" 宇宙中的逻辑测试单元。 该类旨
   在被当作基类使用，特定的测试将由其实体子类来实现。 该类实现了测试运
   行器所需的接口以允许它驱动测试，并实现了可被测试代码用来检测和报告
   各种类型的失败的方法。

   每个 "TestCase" 实例将运行一个单位的基础方法：即名为 *methodName*
   的方法。 在使用 "TestCase" 的大多数场景中，你都不需要修改
   *methodName* 或重新实现默认的 "runTest()" 方法。

   3.2 版更變: "TestCase" 不需要提供 *methodName* 即可成功初始化。 这
   使得从交互式解释器试验 "TestCase" 更为容易。

   "TestCase" 的实例提供了三组方法：一组用来运行测试，另一组被测试实现
   用来检查条件和报告失败，还有一些查询方法用来收集有关测试本身的信息
   。

   第一组（用于运行测试的）方法是：

   setUp()

      为测试预备而调用的方法。 此方法会在调用测试方法之前被调用；除了
      "AssertionError" 或 "SkipTest"，此方法所引发的任何异常都将被视为
      错误而非测试失败。 默认的实现将不做任何事情。

   tearDown()

      在测试方法被调用并记录结果之后立即被调用的方法。 此方法即使在测
      试方法引发异常时仍会被调用，因此子类中的实现将需要特别注意检查内
      部状态。 除 "AssertionError" 或 "SkipTest" 外，此方法所引发的任
      何异常都将被视为额外的错误而非测试失败（因而会增加总计错误报告数
      ）。 此方法将只在 "setUp()" 成功执行时被调用，无论测试方法的结果
      如何。 默认的实现将不做任何事情。

   setUpClass()

      在一个单独类中的测试运行之前被调用的类方法。 "setUpClass" 会被作
      为唯一的参数在类上调用且必须使用 "classmethod()" 装饰器:

         @classmethod
         def setUpClass(cls):
             ...

      查看 Class and Module Fixtures 获取更详细的说明。

      3.2 版新加入.

   tearDownClass()

      在一个单独类的测试完成运行之后被调用的类方法。 "tearDownClass"
      会被作为唯一的参数在类上调用且必须使用 "classmethod()" 装饰器:

         @classmethod
         def tearDownClass(cls):
             ...

      查看 Class and Module Fixtures 获取更详细的说明。

      3.2 版新加入.

   run(result=None)

      运行测试，将结果收集至作为 *result* 传入的 "TestResult"。 如果
      *result* 被省略或为 "None"，则会创建一个临时的结果对象（通过调用
      "defaultTestResult()" 方法）并使用它。 结果对象会被返回给
      "run()" 的调用方。

      同样的效果也可通过简单地调用 "TestCase" 实例来达成。

      3.3 版更變: 之前版本的 "run" 不会返回结果。 也不会调用实例。

   skipTest(reason)

      在测试方法或 "setUp()" 执行期间调用此方法将跳过当前测试。 详情参
      见 跳过测试与预计的失败。

      3.1 版新加入.

   subTest(msg=None, **params)

      返回一个上下文管理器以将其中的代码块作为子测试来执行。 可选的
      *msg* 和 *params* 是将在子测试失败时显示的任意值，以便让你能清楚
      地标识它们。

      一个测试用例可以包含任意数量的子测试声明，并且它们可以任意地嵌套
      。

      查看 使用子测试区分测试迭代 获取更详细的信息。

      3.4 版新加入.

   debug()

      运行测试而不收集结果。 这允许测试所引发的异常被传递给调用方，并
      可被用于支持在调试器中运行测试。

   "TestCase" 类提供了一些断言方法用于检查并报告失败。 下表列出了最常
   用的方法（请查看下文的其他表来了解更多的断言方法）:

   +-------------------------------------------+-------------------------------+-----------------+
   | 方法                                      | 检查对象                      | 引入版本        |
   |===========================================|===============================|=================|
   | "assertEqual(a, b)"                       | "a == b"                      |                 |
   +-------------------------------------------+-------------------------------+-----------------+
   | "assertNotEqual(a, b)"                    | "a != b"                      |                 |
   +-------------------------------------------+-------------------------------+-----------------+
   | "assertTrue(x)"                           | "bool(x) is True"             |                 |
   +-------------------------------------------+-------------------------------+-----------------+
   | "assertFalse(x)"                          | "bool(x) is False"            |                 |
   +-------------------------------------------+-------------------------------+-----------------+
   | "assertIs(a, b)"                          | "a is b"                      | 3.1             |
   +-------------------------------------------+-------------------------------+-----------------+
   | "assertIsNot(a, b)"                       | "a is not b"                  | 3.1             |
   +-------------------------------------------+-------------------------------+-----------------+
   | "assertIsNone(x)"                         | "x is None"                   | 3.1             |
   +-------------------------------------------+-------------------------------+-----------------+
   | "assertIsNotNone(x)"                      | "x is not None"               | 3.1             |
   +-------------------------------------------+-------------------------------+-----------------+
   | "assertIn(a, b)"                          | "a in b"                      | 3.1             |
   +-------------------------------------------+-------------------------------+-----------------+
   | "assertNotIn(a, b)"                       | "a not in b"                  | 3.1             |
   +-------------------------------------------+-------------------------------+-----------------+
   | "assertIsInstance(a, b)"                  | "isinstance(a, b)"            | 3.2             |
   +-------------------------------------------+-------------------------------+-----------------+
   | "assertNotIsInstance(a, b)"               | "not isinstance(a, b)"        | 3.2             |
   +-------------------------------------------+-------------------------------+-----------------+

   这些断言方法都支持 *msg* 参数，如果指定了该参数，它将被用作测试失败
   时的错误消息 (另请参阅 "longMessage")。 请注意将 *msg* 关键字参数传
   给 "assertRaises()", "assertRaisesRegex()", "assertWarns()",
   "assertWarnsRegex()" 的前提是它们必须被用作上下文管理器。

   assertEqual(first, second, msg=None)

      测试 *first* 和 *second* 是否相等。 如果两个值的比较结果是不相等
      ，则测试将失败。

      此外，如果 *first* 和 *second* 的类型完全相同且属于 list, tuple,
      dict, set, frozenset 或 str 或者属于通过 "addTypeEqualityFunc()"
      注册子类的类型则将会调用类型专属的相等判断函数以便生成更有用的默
      认错误消息 (另请参阅 类型专属方法列表)。

      3.1 版更變: 增加了对类型专属的相等判断函数的自动调用。

      3.2 版更變: 增加了 "assertMultiLineEqual()" 作为用于比较字符串的
      默认类型相等判断函数。

   assertNotEqual(first, second, msg=None)

      测试 *first* 和 *second* 是否不等。 如果两个值的比较结果是相等，
      则测试将失败。

   assertTrue(expr, msg=None)
   assertFalse(expr, msg=None)

      测试 *expr* 是否为真值（或假值）。

      请注意这等价于 "bool(expr) is True" 而不等价于 "expr is True" (
      后者要使用 "assertIs(expr, True)")。 当存在更专门的方法时也应避
      免使用此方法 (例如应使用 "assertEqual(a, b)" 而不是
      "assertTrue(a == b)")，因为它们在测试失败时会提供更有用的错误消
      息。

   assertIs(first, second, msg=None)
   assertIsNot(first, second, msg=None)

      测试 *first* 和 *second* 是（或不是）同一个对象。

      3.1 版新加入.

   assertIsNone(expr, msg=None)
   assertIsNotNone(expr, msg=None)

      测试 *expr* 是 (或不是) "None"。

      3.1 版新加入.

   assertIn(member, container, msg=None)
   assertNotIn(member, container, msg=None)

      测试 *member* 是 (或不是) *container* 的成员。

      3.1 版新加入.

   assertIsInstance(obj, cls, msg=None)
   assertNotIsInstance(obj, cls, msg=None)

      测试 *obj* 是 (或不是) *cls* (此参数可以为一个类或包含类的元组，
      即 "isinstance()" 所接受的参数) 的实例。 要检测是否为指定类型，
      请使用 "assertIs(type(obj), cls)"。

      3.2 版新加入.

   还可以使用下列方法来检查异常、警告和日志消息的产生:

   +-----------------------------------------------------------+----------------------------------------+--------------+
   | 方法                                                      | 检查对象                               | 引入版本     |
   |===========================================================|========================================|==============|
   | "assertRaises(exc, fun, *args, **kwds)"                   | "fun(*args, **kwds)" 引发 *exc*        |              |
   +-----------------------------------------------------------+----------------------------------------+--------------+
   | "assertRaisesRegex(exc, r, fun, *args, **kwds)"           | "fun(*args, **kwds)" 引发了 *exc* 并且 | 3.1          |
   |                                                           | 消息可与正则表达式 *r* 相匹配          |              |
   +-----------------------------------------------------------+----------------------------------------+--------------+
   | "assertWarns(warn, fun, *args, **kwds)"                   | "fun(*args, **kwds)" 引发了 *warn*     | 3.2          |
   +-----------------------------------------------------------+----------------------------------------+--------------+
   | "assertWarnsRegex(warn, r, fun, *args, **kwds)"           | "fun(*args, **kwds)" 引发了 *warn* 并  | 3.2          |
   |                                                           | 且消息可与正则表达式 *r* 相匹 配       |              |
   +-----------------------------------------------------------+----------------------------------------+--------------+
   | "assertLogs(logger, level)"                               | "with" 代码块在 *logger* 上使用最小的  | 3.4          |
   |                                                           | *level* 级别写入了日志                 |              |
   +-----------------------------------------------------------+----------------------------------------+--------------+

   assertRaises(exception, callable, *args, **kwds)
   assertRaises(exception, *, msg=None)

      测试当 *callable* 附带任何同时被传给 "assertRaises()" 的位置或关
      键字参数被调用时是否引发了异常。 如果引发了 *exception* 则测试通
      过，如果引发了另一个异常则报错，或者如果未引发任何异常则测试失败
      。 要捕获一组异常中的任何一个，可以将包含多个异常类的元组作为
      *exception* 传入。

      如果只给出了 *exception* 和可能的 *msg* 参数，则返回一个上下文管
      理器以便被测试的代码可以被写成内联形式而不是被写成函数:

         with self.assertRaises(SomeException):
             do_something()

      当被作为上下文管理器使用时，"assertRaises()" 接受额外的关键字参
      数 *msg*。

      上下文管理器将把捕获的异常对象存入在其 "exception" 属性中。 这适
      用于需要对所引发异常执行额外检查的场合:

         with self.assertRaises(SomeException) as cm:
             do_something()

         the_exception = cm.exception
         self.assertEqual(the_exception.error_code, 3)

      3.1 版更變: 添加了将 "assertRaises()" 用作上下文管理器的功能。

      3.2 版更變: 增加了 "exception" 属性。

      3.3 版更變: 增加了 *msg* 关键字参数在作为上下文管理器时使用。

   assertRaisesRegex(exception, regex, callable, *args, **kwds)
   assertRaisesRegex(exception, regex, *, msg=None)

      与 "assertRaises()" 类似但还会测试 *regex* 是否匹配被引发异常的
      字符串表示形式。 *regex* 可以是一个正则表达式对象或包含正则表达
      式的字符串以提供给 "re.search()" 使用。 例如:

         self.assertRaisesRegex(ValueError, "invalid literal for.*XYZ'$",
                                int, 'XYZ')

      或是：

         with self.assertRaisesRegex(ValueError, 'literal'):
            int('XYZ')

      3.1 版新加入: 以方法名``assertRaisesRegexp``添加。

      3.2 版更變: 重命名为 "assertRaisesRegex()"。

      3.3 版更變: 增加了 *msg* 关键字参数在作为上下文管理器时使用。

   assertWarns(warning, callable, *args, **kwds)
   assertWarns(warning, *, msg=None)

      测试当 *callable* 附带任何同时被传给 "assertWarns()" 的位置或关
      键字参数被调用时是否触发了警告。 如果触发了 *warning* 则测试通过
      ，否则测试失败。 引发任何异常则报错。 要捕获一组警告中的任何一个
      ，可将包含多个警告类的元组作为 *warnings* 传入。

      如果只给出了 *warning* 和可能的 *msg* 参数，则返回一个上下文管理
      器以便被测试的代码可以被写成内联形式而不是被写成函数:

         with self.assertWarns(SomeWarning):
             do_something()

      当被作为上下文管理器使用时，"assertWarns()" 接受额外的关键字参数
      *msg*。

      上下文管理器将把捕获的警告对象保存在其 "warning" 属性中，并把触
      发警告的源代码行保存在 "filename" 和 "lineno" 属性中。 这适用于
      需要对捕获的警告执行额外检查的场合:

         with self.assertWarns(SomeWarning) as cm:
             do_something()

         self.assertIn('myfile.py', cm.filename)
         self.assertEqual(320, cm.lineno)

      无论被调用时警告过滤器是否就位此方法均可工作。

      3.2 版新加入.

      3.3 版更變: 增加了 *msg* 关键字参数在作为上下文管理器时使用。

   assertWarnsRegex(warning, regex, callable, *args, **kwds)
   assertWarnsRegex(warning, regex, *, msg=None)

      与 "assertWarns()" 类似但还会测试 *regex* 是否匹配被触发警告的消
      息文本。 *regex* 可以是一个正则表达式对象或包含正则表达式的字符
      串以提供给 "re.search()" 使用。 例如:

         self.assertWarnsRegex(DeprecationWarning,
                               r'legacy_function\(\) is deprecated',
                               legacy_function, 'XYZ')

      或是：

         with self.assertWarnsRegex(RuntimeWarning, 'unsafe frobnicating'):
             frobnicate('/etc/passwd')

      3.2 版新加入.

      3.3 版更變: 增加了 *msg* 关键字参数在作为上下文管理器时使用。

   assertLogs(logger=None, level=None)

      一个上下文管理器，它测试在 *logger* 或其子对象上是否至少记录了一
      条至少为指定 *level* 以上级别的消息。

      如果给出了 *logger* 则它应为一个 "logging.Logger" 对象或为一个指
      定日志记录器名称的 "str"。 默认为根日志记录器，它将捕获未被非传
      播型后继日志记录器所拦阻的所有消息。

      如果给出了 *level* 则它应为一个用数字表示的日志记录级别或其字符
      串形式 (例如 ""ERROR"" 或 "logging.ERROR")。 默认为
      "logging.INFO"。

      如果在 "with" 代码块内部发出了至少一条与 *logger* 和 *level* 条
      件相匹配的消息则测试通过，否则测试失败。

      上下文管理器返回的对象是一个记录辅助器，它会记录所匹配的日志消息
      。 它有两个属性:

      records

         所匹配的日志消息 "logging.LogRecord" 对象组成的列表。

      output

         由 "str" 对象组成的列表，内容为所匹配消息经格式化后的输出。

      示例:

         with self.assertLogs('foo', level='INFO') as cm:
            logging.getLogger('foo').info('first message')
            logging.getLogger('foo.bar').error('second message')
         self.assertEqual(cm.output, ['INFO:foo:first message',
                                      'ERROR:foo.bar:second message'])

      3.4 版新加入.

   还有其他一些方法可用于执行更专门的检查，例如:

   +-----------------------------------------+----------------------------------+----------------+
   | 方法                                    | 检查对象                         | 引入版本       |
   |=========================================|==================================|================|
   | "assertAlmostEqual(a, b)"               | "round(a-b, 7) == 0"             |                |
   +-----------------------------------------+----------------------------------+----------------+
   | "assertNotAlmostEqual(a, b)"            | "round(a-b, 7) != 0"             |                |
   +-----------------------------------------+----------------------------------+----------------+
   | "assertGreater(a, b)"                   | "a > b"                          | 3.1            |
   +-----------------------------------------+----------------------------------+----------------+
   | "assertGreaterEqual(a, b)"              | "a >= b"                         | 3.1            |
   +-----------------------------------------+----------------------------------+----------------+
   | "assertLess(a, b)"                      | "a < b"                          | 3.1            |
   +-----------------------------------------+----------------------------------+----------------+
   | "assertLessEqual(a, b)"                 | "a <= b"                         | 3.1            |
   +-----------------------------------------+----------------------------------+----------------+
   | "assertRegex(s, r)"                     | "r.search(s)"                    | 3.1            |
   +-----------------------------------------+----------------------------------+----------------+
   | "assertNotRegex(s, r)"                  | "not r.search(s)"                | 3.2            |
   +-----------------------------------------+----------------------------------+----------------+
   | "assertCountEqual(a, b)"                | *a* 和 *b* 具有同样数量的相同元  | 3.2            |
   |                                         | 素，无论其顺序如何。             |                |
   +-----------------------------------------+----------------------------------+----------------+

   assertAlmostEqual(first, second, places=7, msg=None, delta=None)
   assertNotAlmostEqual(first, second, places=7, msg=None, delta=None)

      测试 *first* 是否约等于 *second*，比较的标准是计算差值并舍入到
      *places* 所指定的十进制位数 (默认为 7 位)，再与零相比较。 请注意
      此方法是将结果值舍入到指定的 *十进制位数* (即相当于 "round()" 函
      数) 而非 *有效位数*。

      如果提供了 *delta* 而非 *places* 则 *first* 和 *second* 之间的差
      值必须小于等于 (或大于) *delta*。

      同时提供 *delta* 和 *places* 将引发 "TypeError"。

      3.2 版更變: "assertAlmostEqual()" 会自动将几乎相等的对象视为相等
      。 而如果对象相等则 "assertNotAlmostEqual()" 会自动测试失败。 增
      加了 *delta* 关键字参数。

   assertGreater(first, second, msg=None)
   assertGreaterEqual(first, second, msg=None)
   assertLess(first, second, msg=None)
   assertLessEqual(first, second, msg=None)

      根据方法名分别测试 *first* 是否 >, >=, < 或 <= *second*。 如果不
      是，则测试失败:

         >>> self.assertGreaterEqual(3, 4)
         AssertionError: "3" unexpectedly not greater than or equal to "4"

      3.1 版新加入.

   assertRegex(text, regex, msg=None)
   assertNotRegex(text, regex, msg=None)

      测试一个 *regex* 搜索匹配（或不匹配） *文本*。如果不匹配，错误信
      息中将包含匹配模式和 *文本*（或部分匹配失败的 *文本*）。*regex*
      可以是正则表达式对象或能够用于 "re.search()" 的包含正则表达式的
      字符串。

      3.1 版新加入: 以方法名``assertRegexpMatches``添加。

      3.2 版更變: 方法 "assertRegexpMatches()" 已被改名为
      "assertRegex()"。

      3.2 版新加入: "assertNotRegex()"

      3.5 版新加入: "assertNotRegexpMatches" 这个名字是
      "assertNotRegex()" 的已被弃用的别名。

   assertCountEqual(first, second, msg=None)

      测试序列 *first* 与 *second* 是否包含同样的元素，无论其顺序如何
      。 当存在差异时，将生成一条错误消息来列出两个序列之间的差异。

      重复的元素 *不会* 在 *first* 和 *second* 的比较中被忽略。 它会检
      查每个元素在两个序列中的出现次数是否相同。 等价于:
      "assertEqual(Counter(list(first)), Counter(list(second)))" 但还
      适用于包含不可哈希对象的序列。

      3.2 版新加入.

   "assertEqual()" 方法会将相同类型对象的相等性检查分派给不同的类型专
   属方法。 这些方法已被大多数内置类型所实现，但也可以使用
   "addTypeEqualityFunc()" 来注册新的方法:

   addTypeEqualityFunc(typeobj, function)

      注册一个由 "assertEqual()" 调用的特定类型专属方法来检查恰好为相
      同 *typeobj* (而非子类) 的两个对象是否相等。 *function* 必须接受
      两个位置参数和第三个 msg=None 关键字参数，就像 "assertEqual()"
      那样。 当检测到前两个形参之间不相等时它必须引发
      "self.failureException(msg)" -- 可能还会提供有用的信息并在错误消
      息中详细解释不相等的原因。

      3.1 版新加入.

   以下是 "assertEqual()" 自动选用的不同类型的比较方法。一般情况下不需
   要直接在测试中调用这些方法。

   +-------------------------------------------+-------------------------------+----------------+
   | 方法                                      | 用作比较                      | 引入版本       |
   |===========================================|===============================|================|
   | "assertMultiLineEqual(a, b)"              | 字符串                        | 3.1            |
   +-------------------------------------------+-------------------------------+----------------+
   | "assertSequenceEqual(a, b)"               | 序列                          | 3.1            |
   +-------------------------------------------+-------------------------------+----------------+
   | "assertListEqual(a, b)"                   | 列表                          | 3.1            |
   +-------------------------------------------+-------------------------------+----------------+
   | "assertTupleEqual(a, b)"                  | 元组                          | 3.1            |
   +-------------------------------------------+-------------------------------+----------------+
   | "assertSetEqual(a, b)"                    | 集合                          | 3.1            |
   +-------------------------------------------+-------------------------------+----------------+
   | "assertDictEqual(a, b)"                   | 字典                          | 3.1            |
   +-------------------------------------------+-------------------------------+----------------+

   assertMultiLineEqual(first, second, msg=None)

      测试多行字符串 *first* 是否与字符串 *second* 相等。 当不相等时将
      在错误消息中包括两个字符串之间差异的高亮显示。 此方法会在通过
      "assertEqual()" 进行字符串比较时默认被使用。

      3.1 版新加入.

   assertSequenceEqual(first, second, msg=None, seq_type=None)

      测试两个序列是否相等。 如果提供了 *seq_type*，则 *first* 和
      *second* 都必须为 *seq_type* 的实例否则将引发失败。 如果两个序列
      不相等则会构造一个错误消息来显示两者之间的差异。

      此方法不会被 "assertEqual()" 直接调用，但它会被用于实现
      "assertListEqual()" 和 "assertTupleEqual()"。

      3.1 版新加入.

   assertListEqual(first, second, msg=None)
   assertTupleEqual(first, second, msg=None)

      测试两个列表或元组是否相等。 如果不相等，则会构造一个错误消息来
      显示两者之间的差异。 如果某个形参的类型不正确也会引发错误。 这些
      方法会在通过 "assertEqual()" 进行列表或元组比较时默认被使用。

      3.1 版新加入.

   assertSetEqual(first, second, msg=None)

      测试两个集合是否相等。 如果不相等，则会构造一个错误消息来列出两
      者之间的差异。 此方法会在通过 "assertEqual()" 进行集合或冻结集合
      比较时默认被使用。

      如果 *first* 或 *second* 没有 "set.difference()" 方法则测试失败
      。

      3.1 版新加入.

   assertDictEqual(first, second, msg=None)

      测试两个字典是否相等。 如果不相等，则会构造一个错误消息来显示两
      个字典的差异。 此方法会在对 "assertEqual()" 的调用中默认被用来进
      行字典的比较。

      3.1 版新加入.

   最后 "TestCase" 还提供了以下的方法和属性:

   fail(msg=None)

      无条件地发出测试失败消息，附带错误消息 *msg* 或 "None"。

   failureException

      这个类属性给出测试方法所引发的异常。 如果某个测试框架需要使用专
      门的异常，并可能附带额外的信息，则必须子类化该类以便与框架“正常
      互动”。 这个属性的初始值为 "AssertionError"。

   longMessage

      这个类属性决定当将一个自定义失败消息作为 msg 参数传给一个失败的
      assertXYY 调用时会发生什么。默认值为 "True"。 在此情况下，自定义
      消息会被添加到标准失败消息的末尾。 当设为 "False" 时，自定义消息
      会替换标准消息。

      类设置可以通过在调用断言方法之前将一个实例属性 self.longMessage
      赋值为 "True" 或 "False" 在单个测试方法中进行重载。

      类设置会在每个测试调用之前被重置。

      3.1 版新加入.

   maxDiff

      这个属性控制来自在测试失败时报告 diffs 的断言方法的 diffs 输出的
      最大长度。 它默认为 80*8 个字符。 这个属性所影响的断言方法有
      "assertSequenceEqual()" (包括所有委托给它的序列比较方法),
      "assertDictEqual()" 以及 "assertMultiLineEqual()"。

      将 "maxDiff" 设为 "None" 表示不限制 diffs 的最大长度。

      3.2 版新加入.

   测试框架可使用下列方法在测试时收集信息:

   countTestCases()

      返回此测试对象所提供的测试数量。 对于 "TestCase" 实例，该数量将
      总是为 "1"。

   defaultTestResult()

      返回此测试类所要使用的测试结果类的实例（如果未向 "run()" 方法提
      供其他结果实例）。

      对于 "TestCase" 实例，该返回值将总是为 "TestResult" 的实例；
      "TestCase" 的子类应当在有必要时重载此方法。

   id()

      返回一个标识指定测试用例的字符串。 该返回值通常为测试方法的完整
      名称，包括模块名和类名。

   shortDescription()

      返回测试的描述，如果未提供描述则返回 "None"。 此方法的默认实现将
      在可用的情况下返回测试方法的文档字符串的第一行，或者返回 "None"
      。

      3.1 版更變: 在 3.1 中已修改此方法将测试名称添加到简短描述中，即
      使存在文档字符串。 这导致了与单元测试扩展的兼容性问题因而在
      Python 3.2 中将添加测试名称操作改到 "TextTestResult" 中。

   addCleanup(function, *args, **kwargs)

      在 "tearDown()" 之后添加了一个要调用的函数来清理测试期间所使用的
      资源。 函数将按它们被添加的相反顺序被调用 (LIFO (last-in, first-
      out))。 它们在调用时将附带它们被添加时传给 "addCleanup()" 的任何
      参数和关键字参数。

      如果 "setUp()" 失败，即意味着 "tearDown()" 未被调用，则已添加的
      任何清理函数仍将被调用。

      3.1 版新加入.

   doCleanups()

      此方法会在 "tearDown()" 之后，或者如果 "setUp()" 引发了异常则会
      在 "setUp()" 之后被调用。

      它将负责调用由 "addCleanup()" 添加的所有清理函数。 如果你需要在
      "tearDown()" *之前* 调用清理函数则可以自行调用 "doCleanups()"。

      "doCleanups()" 每次会弹出清理函数栈中的一个方法，因此它可以在任
      何时候被调用。

      3.1 版新加入.

   classmethod addClassCleanup(function, /, *args, **kwargs)

      在Add a function to be called after "tearDownClass()" 之后添加了
      一个要调用的函数来清理测试类运行期间所使用的资源。 函数将按它们
      被添加的相反顺序被调用 (LIFO (last-in, first-out))。 它们在调用
      时将附带它们被添加时传给 "addClassCleanup()" 的任何参数和关键字
      参数。

      如果 "setUpClass()" 失败，即意味着 "tearDownClass()" 未被调用，
      则已添加的任何清理函数仍将被调用。

      3.8 版新加入.

   classmethod doClassCleanups()

      此方法会在 "tearDownClass()" 之后无条件地被调用，或者如果
      "setUpClass()" 引发了异常则会在 "setUpClass()" 之后被调用。

      它将负责访问由 "addClassCleanup()" 添加的所有清理函数。 如果你需
      要在 "tearDownClass()" *之前* 调用清理函数则可以自行调用
      "doClassCleanups()"。

      "doClassCleanups()" 每次会弹出清理函数栈中的一个方法，因此它在任
      何时候被调用。

      3.8 版新加入.

class unittest.IsolatedAsyncioTestCase(methodName='runTest')

   这个类提供了与 "TestCase" 类似的 API 并也接受协程作为测试函数。

   3.8 版新加入.

   coroutine asyncSetUp()

      为测试预备而调用的方法。 此方法会在 "setUp()" 之后被调用。 此方
      法将在调用测试方法之前立即被调用；除了 "AssertionError" 或
      "SkipTest"，此方法所引发的任何异常都将被视为错误而非测试失败。
      默认的实现将不做任何事情。

   coroutine asyncTearDown()

      在测试方法被调用并记录结果之后立即被调用的方法。 此方法会在
      "tearDown()" 之前被调用。 此方法即使在测试方法引发异常时仍会被调
      用，因此子类中的实现将需要特别注意检查内部状态。 除
      "AssertionError" 或 "SkipTest" 外，此方法所引发的任何异常都将被
      视为额外的错误而非测试失败（因而会增加总计错误报告数）。 此方法
      将只在 "asyncSetUp()" 成功执行时被调用，无论测试方法的结果如何。
      默认的实现将不做任何事情。

   addAsyncCleanup(function, /, *args, **kwargs)

      此方法接受一个可被用作清理函数的协程。

   run(result=None)

      设置一个新的事件循环来运行测试，将结果收集至作为 *result* 传入的
      "TestResult"。 如果 *result* 被省略或为 "None"，则会创建一个临时
      的结果对象（通过调用 "defaultTestResult()" 方法）并使用它。 结果
      对象会被返回给 "run()" 的调用方。 在测试结束时事件循环中的所有任
      务都将被取消。

   一个显示先后顺序的例子:

      from unittest import IsolatedAsyncioTestCase

      events = []


      class Test(IsolatedAsyncioTestCase):


          def setUp(self):
              events.append("setUp")

          async def asyncSetUp(self):
              self._async_connection = await AsyncConnection()
              events.append("asyncSetUp")

          async def test_response(self):
              events.append("test_response")
              response = await self._async_connection.get("https://example.com")
              self.assertEqual(response.status_code, 200)
              self.addAsyncCleanup(self.on_cleanup)

          def tearDown(self):
              events.append("tearDown")

          async def asyncTearDown(self):
              await self._async_connection.close()
              events.append("asyncTearDown")

          async def on_cleanup(self):
              events.append("cleanup")

      if __name__ == "__main__":
          unittest.main()

   在运行测试之后，"events" 将会包含 "["setUp", "asyncSetUp",
   "test_response", "asyncTearDown", "tearDown", "cleanup"]"。

class unittest.FunctionTestCase(testFunc, setUp=None, tearDown=None, description=None)

   这个类实现了 "TestCase" 的部分接口，允许测试运行方驱动测试，但不提
   供可被测试代码用来检查和报告错误的方法。 这个类被用于创建使用传统测
   试代码的测试用例，允许它被集成到基于 "unittest" 的测试框架中。


已弃用的别名
~~~~~~~~~~~~

由于历史原因，某些 "TestCase" 方法具有一个或几个已目前已弃用的别名。
下表列出了它们的正确名称和已弃用的别名:

   +--------------------------------+------------------------+-------------------------+
   | 方法名                         | 已弃用的别名           | 已弃用的别名            |
   |================================|========================|=========================|
   | "assertEqual()"                | failUnlessEqual        | assertEquals            |
   +--------------------------------+------------------------+-------------------------+
   | "assertNotEqual()"             | failIfEqual            | assertNotEquals         |
   +--------------------------------+------------------------+-------------------------+
   | "assertTrue()"                 | failUnless             | assert_                 |
   +--------------------------------+------------------------+-------------------------+
   | "assertFalse()"                | failIf                 |                         |
   +--------------------------------+------------------------+-------------------------+
   | "assertRaises()"               | failUnlessRaises       |                         |
   +--------------------------------+------------------------+-------------------------+
   | "assertAlmostEqual()"          | failUnlessAlmostEqual  | assertAlmostEquals      |
   +--------------------------------+------------------------+-------------------------+
   | "assertNotAlmostEqual()"       | failIfAlmostEqual      | assertNotAlmostEquals   |
   +--------------------------------+------------------------+-------------------------+
   | "assertRegex()"                |                        | assertRegexpMatches     |
   +--------------------------------+------------------------+-------------------------+
   | "assertNotRegex()"             |                        | assertNotRegexpMatches  |
   +--------------------------------+------------------------+-------------------------+
   | "assertRaisesRegex()"          |                        | assertRaisesRegexp      |
   +--------------------------------+------------------------+-------------------------+

   3.1 版後已棄用: 在第二列中列出的 fail* 别名已经被弃用。

   3.2 版後已棄用: 在第三列中列出的 assert* 别名已经被弃用。

   3.2 版後已棄用: "assertRegexpMatches" 和 "assertRaisesRegexp" 已经
   被重命名为 "assertRegex()" 和 "assertRaisesRegex()"。

   3.5 版後已棄用: "assertNotRegexpMatches" 这个名称已被弃用并应改用
   "assertNotRegex()"。


分组测试
--------

class unittest.TestSuite(tests=())

   这个类代表对单独测试用例和测试套件的聚合。 这个类提供给测试运行方所
   需的接口以允许其像任何其他测试用例一样运行。 运行一个 "TestSuite"
   实例与对套件执行迭代来逐一运行每个测试的效果相同。

   如果给出了 *tests*，则它必须是一个包含单独测试用例的可迭代对象或是
   将被用于初始构建测试套件的其他测试套件。 额外的方法被提供用来在随后
   向测试集添加测试用例和测试套件。

   "TestSuite" 对象的行为与 "TestCase" 对象很相似，区别在于它们并不会
   真正实现一个测试。 它们会被用来将测试聚合为多个要同时运行的测试分组
   。 还有一些额外的方法会被用来向 "TestSuite" 实例添加测试:

   addTest(test)

      向测试套件添加 "TestCase" 或 "TestSuite"。

   addTests(tests)

      将来自包含 "TestCase" 和 "TestSuite" 实例的可迭代对象的所有测试
      添加到这个测试套件。

      这等价于对 *tests* 进行迭代，并为其中的每个元素调用 "addTest()"
      。

   "TestSuite" 与 "TestCase" 共享下列方法:

   run(result)

      运行与这个套件相关联的测试，将结果收集到作为 *result* 传入的测试
      结果对象中。 请注意与 "TestCase.run()" 的区别，"TestSuite.run()"
      必须传入结果对象。

   debug()

      运行与这个套件相关联的测试而不收集结果。 这允许测试所引发的异常
      被传递给调用方并可被用于支持在调试器中运行测试。

   countTestCases()

      返回此测试对象所提供的测试数量，包括单独的测试和子套件。

   __iter__()

      由 "TestSuite" 分组的测试总是可以通过迭代来访问。 其子类可以通过
      重载 "__iter__()" 来惰性地提供测试。 请注意此方法可在单个套件上
      多次被调用（例如在计数测试或相等性比较时），为此在
      "TestSuite.run()" 之前重复迭代所返回的测试对于每次调用迭代都必须
      相同。 在 "TestSuite.run()" 之后，调用方不应继续访问此方法所返回
      的测试，除非调用方使用重载了 "TestSuite._removeTestAtIndex()" 的
      子类来保留对测试的引用。

      3.2 版更變: 在较早的版本中 "TestSuite" 会直接访问测试而不是通过
      迭代，因此只重载 "__iter__()" 并不足以提供所有测试。

      3.4 版更變: 在较早的版本中 "TestSuite" 会在 "TestSuite.run()" 之
      后保留对每个 "TestCase" 的引用。 其子类可以通过重载
      "TestSuite._removeTestAtIndex()" 来恢复此行为。

   在 "TestSuite" 对象的典型应用中，"run()" 方法是由 "TestRunner" 发起
   调用而不是由最终用户测试来控制。


加载和运行测试
--------------

class unittest.TestLoader

   "TestLoader" 类可被用来基于类和模块创建测试套件。 通常，没有必要创
   建该类的实例；"unittest" 模块提供了一个可作为
   "unittest.defaultTestLoader" 共享的实例。 但是，使用子类或实例允许
   对某些配置属性进行定制。

   "TestLoader" 对象具有下列属性:

   errors

      包含在加载测试期间遇到的非致命错误的列表。 在任何时候都不会被加
      载方重置。 致命错误是通过相关方法引发一个异常来向调用方发出信号
      的。 非致命错误也是由一个将在运行时引发原始错误的合成测试来提示
      的。

      3.5 版新加入.

   "TestLoader" 对象具有下列方法:

   loadTestsFromTestCase(testCaseClass)

      返回一个包含在 "TestCase" 所派生的 "testCaseClass" 中的所有测试
      用例的测试套件。

      会为每个由 "getTestCaseNames()" 指明的方法创建一个测试用例实例。
      在默认情况下这些都是以 "test" 开头的方法名称。 如果
      "getTestCaseNames()" 不返回任何方法，但 "runTest()" 方法已被实现
      ，则会为该方法创建一个单独的测试用例。

   loadTestsFromModule(module, pattern=None)

      返回包含在给定模块中的所有测试用例的测试套件。 此方法会在
      *module* 中搜索从派生自 "TestCase" 的类并为该类定义的每个测试方
      法创建一个类实例。

      備註:

        虽然使用 "TestCase" 所派生的类的层级结构可以方便地共享配置和辅
        助函数，但在不打算直接实例化的基类上定义测试方法并不能很好地配
        合此方法使用。 不过，当配置有差异并且定义在子类当中时这样做还
        是有用处的。

      如果一个模块提供了 "load_tests" 函数则它将被调用以加载测试。 这
      允许模块自行定制测试加载过程。 这就称为 load_tests protocol。
      *pattern* 参数会被作为传给 "load_tests" 的第三个参数。

      3.2 版更變: 添加了对 "load_tests" 的支持。

      3.5 版更變: 未写入文档的非官方 *use_load_tests* 默认参数已被弃用
      并忽略，但是它仍然被接受以便向下兼容。 此方法现在还接受一个仅限
      关键字参数 *pattern*，它会被作为传给 "load_tests" 的第三个参数。

   loadTestsFromName(name, module=None)

      返回由给出了字符串形式规格描述的所有测试用例组成的测试套件。

      描述名称 *name* 是一个“带点号的名称”，它可以被解析为一个模块、一
      个测试用例类、一个测试用例类内部的测试方法、一个 "TestSuite" 实
      例，或者一个返回 "TestCase" 或 "TestSuite" 实例的可调用对象。 这
      些检查将按在此列出的顺序执行；也就是说，一个可能的测试用例类上的
      方法将作为“一个测试用例内部的测试方法”而非作为“一个可调用对象”被
      选定。

      举例来说，如果你有一个模块 "SampleTests"，其中包含一个派生自
      "TestCase" 的类 "SampleTestCase"，其中包含三个测试方法
      ("test_one()", "test_two()" 和 "test_three()")。 则描述名称
      "'SampleTests.SampleTestCase'" 将使此方法返回一个测试套件，它将
      运行全部三个测试方法。 使用描述名称
      "'SampleTests.SampleTestCase.test_two'" 将使它返回一个测试套件，
      它将仅运行 "test_two()" 测试方法。 描述名称可以指向尚未被导入的
      模块和包；它们将作为附带影响被导入。

      本模块可以选择相对于给定的 *module* 来解析 *name*。

      3.5 版更變: 如果在遍历 *name* 时发生了 "ImportError" 或
      "AttributeError" 则在运行时引发该错误的合成测试将被返回。 这些错
      误被包括在由 self.errors 所积累的错误中。

   loadTestsFromNames(names, module=None)

      类似于 "loadTestsFromName()"，但是接受一个名称序列而不是单个名称
      。 返回值是一个测试套件，它支持为每个名称所定义的所有测试。

   getTestCaseNames(testCaseClass)

      返回由 *testCaseClass* 中找到的方法名称组成的已排序的序列；这应
      当是 "TestCase" 的一个子类。

   discover(start_dir, pattern='test*.py', top_level_dir=None)

      通过从指定的开始目录向其子目录递归来找出所有测试模块，并返回一个
      包含该结果的 TestSuite 对象。 只有与 *pattern* 匹配的测试文件才
      会被加载。 （使用 shell 风格的模式匹配。） 只有可导入的模块名称
      （即有效的 Python 标识符）将会被加载。

      所有测试模块都必须可以从项目的最高层级上导入。 如果起始目录不是
      最高层级目录则必须单独指明最高层级目录。

      如果导入某个模块失败，比如因为存在语法错误，则会将其记录为单独的
      错误并将继续查找模块。 如果导入失败是因为引发了 "SkipTest"，则会
      将其记录为跳过而不是错误。

      如果找到了一个包（即包含名为 "__init__.py" 的文件的目录），则将
      在包中查找 "load_tests" 函数。 如果存在此函数则将对其执行调用
      "package.load_tests(loader, tests, pattern)"。 测试发现操作会确
      保在执行期间仅检查测试一次，即使 load_tests 函数本身调用了
      "loader.discover" 也是如此。.

      如果 "load_tests" 存在则发现操作 *不会* 对包执行递归处理，
      "load_tests" 将负责加载包中的所有测试。is responsible for
      loading all tests in the package.

      模式特意地不被当作 loader 属性来保存以使包能够自己继续执行发现操
      作。 *top_level_dir* 则会被保存以使 "load_tests" 不需要将此参数
      传入到 "loader.discover()"。

      *start_dir* 可以是一个带点号的名称或是一个目录。

      3.2 版新加入.

      3.4 版更變: 在导入时引发 "SkipTest" 的模块会被记录为跳过，而不是
      错误。

      3.4 版更變: *start_dir* 可以是一个 *命名空间包*。

      3.4 版更變: 路径在被导入之前会先被排序以使得执行顺序保持一致，即
      使下层文件系统的顺序不是取决于文件名的。

      3.5 版更變: 现在 "load_tests" 会检查已找到的包，无论它们的路径是
      否与 *pattern* 匹配，因为包名称是无法与默认的模式匹配的。

   "TestLoader" 的下列属性可通过子类化或在实例上赋值来配置:

   testMethodPrefix

      给出将被解读为测试方法的方法名称的前缀的字符串。 默认值为
      "'test'"。

      这会影响 "getTestCaseNames()" 以及所有 "loadTestsFrom*()" 方法。

   sortTestMethodsUsing

      将被用来在 "getTestCaseNames()" 以及所有 "loadTestsFrom*()" 方法
      中比较方法名称以便对它们进行排序。

   suiteClass

      根据一个测试列表来构造测试套件的可调用对象。 不需要结果对象上的
      任何方法。 默认值为 "TestSuite" 类。

      这会影响所有 "loadTestsFrom*()" 方法。

   testNamePatterns

      由 Unix shell 风格通配符的测试名称模式组成的列表，供测试方法进行
      匹配以包括在测试套件中（参见 "-v" 选项）。

      如果该属性不为 "None" (默认值)，则将要包括在测试套件中的所有测试
      方法都必须匹配该列表中的某个模式。 请注意匹配总是使用
      "fnmatch.fnmatchcase()" 来执行，因此不同于传给 "-v" 选项的模式，
      简单的子字符串模式将必须使用 "*" 通配符来进行转换。

      这会影响所有 "loadTestsFrom*()" 方法。

      3.7 版新加入.

class unittest.TestResult

   这个类被用于编译有关哪些测试执行成功而哪些失败的信息。

   存放一组测试的结果的 "TestResult" 对象。 "TestCase" 和 "TestSuite"
   类将确保结果被正确地记录；测试创建者无须担心如何记录测试的结果。

   建立在 "unittest" 之上的测试框架可能会想要访问通过运行一组测试所产
   生的 "TestResult" 对象用来报告信息；"TestRunner.run()" 方法是出于这
   个目的而返回 "TestResult" 实例的。

   "TestResult" 实例具有下列属性，在检查运行一组测试的结果的时候很有用
   处。

   errors

      一个包含 "TestCase" 实例和保存了格式化回溯信息的字符串 2 元组的
      列表。 每个元组代表一个引发了非预期的异常的测试。

   failures

      一个包含 "TestCase" 实例和保存了格式化回溯信息的字符串 2 元组的
      列表。 每个元组代表一个使用 "TestCase.assert*()" 方法显式地发出
      失败信号的测试。

   skipped

      一个包含 2-tuples of "TestCase" 实例和保存了跳过测试原因的字符串
      2 元组的列表。

      3.1 版新加入.

   expectedFailures

      一个包含 "TestCase" 实例和保存了格式化回溯信息的 2 元组的列表。
      每个元组代表测试用例的一个已预期的失败或错误。

   unexpectedSuccesses

      一个包含被标记为已预期失败，但却测试成功的 "TestCase" 实例的列表
      。

   shouldStop

      当测试的执行应当被 "stop()" 停止时则设为 "True"。

   testsRun

      目前已运行的测试的总数量。

   buffer

      如果设为真值，"sys.stdout" 和 "sys.stderr" 将在 "startTest()" 和
      "stopTest()" 被调用之间被缓冲。 被收集的输出将仅在测试失败或发生
      错误时才会被回显到真正的 "sys.stdout" 和 "sys.stderr"。 任何输出
      还会被附加到失败/错误消息中。

      3.2 版新加入.

   failfast

      如果设为真值则 "stop()" 将在首次失败或错误时被调用，停止测试运行
      。

      3.2 版新加入.

   tb_locals

      如果设为真值则局部变量将被显示在回溯信息中。

      3.5 版新加入.

   wasSuccessful()

      如果当前所有测试都已通过则返回 "True"，否则返回 "False"。

      3.4 版更變: 如果有任何来自测试的 "unexpectedSuccesses" 被
      "expectedFailure()" 装饰器所标记则返回 "False"。

   stop()

      此方法可被调用以提示正在运行的测试集要将 "shouldStop" 属性设为
      "True" 来表示其应当被中止。 "TestRunner" 对象应当认同此旗标并返
      回而不再运行任何额外的测试。

      例如，该特性会被 "TextTestRunner" 类用来在当用户从键盘发出一个中
      断信号时停止测试框架。 提供了 "TestRunner" 实现的交互式工具也可
      通过类似方式来使用该特性。

   "TestResult" 类的下列方法被用于维护内部数据结构，并可在子类中被扩展
   以支持额外的报告需求。 这特别适用于构建支持在运行测试时提供交互式报
   告的工具。

   startTest(test)

      当测试用例 *test* 即将被运行时被调用。

   stopTest(test)

      在测试用例 *test* 已经执行后被调用，无论其结果如何。

   startTestRun()

      在任何测试被执行之前被调用一次。

      3.1 版新加入.

   stopTestRun()

      在所有测试被执行之后被调用一次。

      3.1 版新加入.

   addError(test, err)

      当测试用例 *test* 引发了非预期的异常时被调用。 *err* 是一个元组
      ，其形式与 "sys.exc_info()" 的返回值相同: "(type, value,
      traceback)"。

      默认实现会将一个元组 "(test, formatted_err)" 添加到实例的
      "errors" 属性，其中 *formatted_err* 是派生自 *err* 的已格式化回
      溯信息。

   addFailure(test, err)

      当测试用例 *test* 发出了失败信号时将被调用。 *err* 是一个元组，
      其形式与 "sys.exc_info()" 的返回值相同: "(type, value,
      traceback)"。

      默认实现会将一个元组 "(test, formatted_err)" 添加到实例的
      "failures" 属性，其中 *formatted_err* 是派生自 *err* 的已格式化
      回溯信息。

   addSuccess(test)

      当测试用例 *test* 成功时被调用。

      默认实现将不做任何操作。

   addSkip(test, reason)

      当测试用例 *test* 被跳过时将被调用。 *reason* 是给出的跳过测试的
      理由。

      默认实现会将一个元组 "(test, reason)" 添加到实例的 "skipped" 属
      性。

   addExpectedFailure(test, err)

      当测试用例 *test* 失败或发生错误，但是使用了 "expectedFailure()"
      装饰器来标记时将被调用。

      默认实现会将一个元组 "(test, formatted_err)" 添加到实例的
      "expectedFailures" 属性，其中 *formatted_err* 是派生自 *err* 的
      已格式化回溯信息。

   addUnexpectedSuccess(test)

      当测试用例 *test* 使用了was marked with the "expectedFailure()"
      装饰器来标记，但是却执行成功时将被调用。

      默认实现会将该测试添加到实例的 "unexpectedSuccesses" 属性。

   addSubTest(test, subtest, outcome)

      当一个子测试结束时将被调用。 *test* 是对应于该测试方法的测试用例
      。 *subtest* 是一个描述该子测试的 "TestCase" 实例。

      如果 *outcome* 为 "None"，则该子测试执行成功。 否则，它将失败并
      引发一个异常，*outcome* 是一个元组，其形式与 "sys.exc_info()" 的
      返回值相同: "(type, value, traceback)"。

      默认实现在测试结果为成功时将不做任何事，并会将子测试的失败记录为
      普通的失败。

      3.4 版新加入.

class unittest.TextTestResult(stream, descriptions, verbosity)

   "TestResult" 的一个具体实现，由 "TextTestRunner" 使用。

   3.2 版新加入: 这个类在之前被命名为 "_TextTestResult"。 这个旧名字仍
   然作为别名存在，但已被弃用。

unittest.defaultTestLoader

   用于分享的 "TestLoader" 类实例。 如果不需要自制 "TestLoader"，则可
   以使用该实例而不必重复创建新的实例。

class unittest.TextTestRunner(stream=None, descriptions=True, verbosity=1, failfast=False, buffer=False, resultclass=None, warnings=None, *, tb_locals=False)

   一个将结果输出到流的基本测试运行器。 如果 *stream* 为默认的 "None"
   ，则会使用 "sys.stderr" 作为输出流。 这个类具有一些配置形参，但实际
   上都非常简单。 运行测试套件的图形化应用程序应当提供替代实现。 这样
   的实现应当在添加新特性到 unittest 时接受 "**kwargs" 作为修改构造运
   行器的接口。

   在默认情况下这个运行器会显示 "DeprecationWarning",
   "PendingDeprecationWarning", "ResourceWarning" 和 "ImportWarning"，
   即使它们 被默认忽略。 由 deprecated unittest methods 所导致的弃用警
   告也会被作为特例，并且当警告过滤器为 "'default'" 或 "'always'" 时，
   对于每个模块它们将仅显示一次，以避免过多的警告消息。 此种行为可使用
   Python 的 "-Wd" 或 "-Wa" 选项 (参见 警告控制) 并让 *warnings* 保持
   为 "None" 来覆盖。

   3.2 版更變: 增加了 "warnings" 参数。

   3.2 版更變: 默认流会在实例化而不是在导入时被设为 "sys.stderr"。

   3.5 版更變: 增加了 tb_locals 形参。

   _makeResult()

      此方法将返回由 "run()" 使用的 "TestResult" 实例。 它不应当被直接
      调用，但可在子类中被重载以提供自定义的 "TestResult"。

      "_makeResult()" 会实例化传给 "TextTestRunner" 构造器的
      "resultclass" 参数所指定的类或可迭代对象。 如果没有提供
      "resultclass" 则默认为 "TextTestResult"。 结果类会使用以下参数来
      实例化:

         stream, descriptions, verbosity

   run(test)

      此方法是 "TextTestRunner" 的主要公共接口。 此方法接受一个
      "TestSuite" 或 "TestCase" 实例。 通过调用 "_makeResult()" 创建
      "TestResult" 来运行测试并将结果打印到标准输出。

unittest.main(module='__main__', defaultTest=None, argv=None, testRunner=None, testLoader=unittest.defaultTestLoader, exit=True, verbosity=1, failfast=None, catchbreak=None, buffer=None, warnings=None)

   从 *module* 加载一组测试并运行它们的命令行程序；这主要是为了让测试
   模块能方便地执行。 此函数的最简单用法是在测试脚本末尾包括下列行:

      if __name__ == '__main__':
          unittest.main()

   你可以通过传入冗余参数运行测试以获得更详细的信息:

      if __name__ == '__main__':
          unittest.main(verbosity=2)

   *defaultTest* 参数是要运行的单个测试名称，或者如果未通过 *argv* 指
   定任何测试名称则是包含多个测试名称的可迭代对象。 如果未指定或为
   "None" 且未通过 *argv* 指定任何测试名称，则会运行在 *module* 中找到
   的所有测试。

   *argv* 参数可以是传给程序的选项列表，其中第一个元素是程序名。 如未
   指定或为 "None"，则会使用 "sys.argv" 的值。

   *testRunner* 参数可以是测试运行器类或是其已创建的实例。 在默认情况
   下 "main" 会调用 "sys.exit()" 并附带一个退出码来指明测试运行是成功
   还是失败。

   *testLoader* 参数必须是一个 "TestLoader" 实例，其默认值为
   "defaultTestLoader"。

   "main" 支持通过传入 "exit=False" 参数以便在交互式解释器中使用。 这
   将在标准输出中显示结果而不调用 "sys.exit()":

      >>> from unittest import main
      >>> main(module='test_module', exit=False)

   *failfast*, *catchbreak* 和 *buffer* 形参的效果与同名的 command-
   line options 一致。

   *warnings* 参数指定在运行测试时所应使用的 警告过滤器。 如果未指定，
   则默认的 "None" 会在将 "-W" 选项传给 **python** 命令时被保留 (参见
   警告控制)，而在其他情况下将被设为 "'default'"。

   调用 "main" 实际上将返回一个 "TestProgram" 类的实例。 这会把测试运
   行结果保存为 "result" 属性。

   3.1 版更變: 增加了 *exit* 形参。

   3.2 版更變: 增加了 *verbosity*, *failfast*, *catchbreak*, *buffer*
   和 *warnings* 形参。

   3.4 版更變: *defaultTest* 形参被修改为也接受一个由测试名称组成的迭
   代器。


load_tests 协议
~~~~~~~~~~~~~~~

3.2 版新加入.

模块或包可以通过实现一个名为 "load_tests" 的函数来定制在正常测试运行或
测试发现期间要如何从中加载测试。

如果一个测试模块定义了 "load_tests" 则它将被
"TestLoader.loadTestsFromModule()" 调用并传入下列参数:

   load_tests(loader, standard_tests, pattern)

其中 *pattern* 会通过 "loadTestsFromModule" 传入。 它的默认值为 "None"
。

它应当返回一个 "TestSuite"。

*loader* 是执行载入操作的 "TestLoader" 实例。 *standard_tests* 是默认
要从该模块载入的测试。 测试模块通常只需从标准测试集中添加或移除测试。
第三个参数是在作为测试发现的一部分载入包时使用的。

一个从指定 "TestCase" 类集合中载入测试的 "load_tests" 函数看起来可能是
这样的:

   test_cases = (TestCase1, TestCase2, TestCase3)

   def load_tests(loader, tests, pattern):
       suite = TestSuite()
       for test_class in test_cases:
           tests = loader.loadTestsFromTestCase(test_class)
           suite.addTests(tests)
       return suite

如果发现操作是在一个包含包的目录中开始的，不论是通过命令行还是通过调用
"TestLoader.discover()"，则将在包 "__init__.py" 中检查 "load_tests"。
如果不存在此函数，则发现将在包内部执行递归，就像它是另一个目录一样。
在其他情况下，包中测试的发现操作将留给 "load_tests" 执行，它将附带下列
参数被调用:

   load_tests(loader, standard_tests, pattern)

这应当返回代表包中所有测试的 "TestSuite"。 ("standard_tests" 将只包含
从 "__init__.py" 获取的测试。)

因为模式已被传入 "load_tests" 所以包可以自由地继续（还可能修改）测试发
现操作。 针对一个测试包的 '无操作' "load_tests" 函数看起来是这样的:

   def load_tests(loader, standard_tests, pattern):
       # top level directory cached on loader instance
       this_dir = os.path.dirname(__file__)
       package_tests = loader.discover(start_dir=this_dir, pattern=pattern)
       standard_tests.addTests(package_tests)
       return standard_tests

3.5 版更變: 发现操作不会再检查包名称是否匹配 *pattern*，因为包名称不可
能匹配默认的模式。


类与模块设定
============

类与模块设定是在 "TestSuite" 中实现的。 当测试套件遇到来自新类的测试时
则来自之前的类（如果存在）的 "tearDownClass()" 会被调用，然后再调用来
自新类的 "setUpClass()"。

类似地如果测试是来自之前的测试的另一个模块则来自之前模块的
"tearDownModule" 将被运行，然后再运行来自新模块的 "setUpModule"。

在所有测试运行完毕后最终的 "tearDownClass" 和 "tearDownModule" 将被运
行。

请注意共享设定不适用于一些 [潜在的] 特性例如测试并行化并且它们会破坏测
试隔离。 它们应当被谨慎地使用。

由 unittest 测试加载器创建的测试的默认顺序是将所有来自相同模块和类的测
试归入相同分组。 这将导致 "setUpClass" / "setUpModule" (等) 对于每个类
和模块都恰好被调用一次。 如果你将顺序随机化，以便使得来自不同模块和类
的测试彼此相邻，那么这些共享的设定函数就可能会在一次测试运行中被多次调
用。

共享的设定不适用与非标准顺序的套件。 对于不想支持共享设定的框架来说
"BaseTestSuite" 仍然可用。

如果在共享的设定函数中引发了任何异常则测试将被报告错误。 因为没有对应
的测试实例，所以会创建一个 "_ErrorHolder" 对象（它具有与 "TestCase" 相
同的接口）来代表该错误。 如果你只是使用标准 unittest 测试运行器那么这
个细节并不重要，但是如果你是一个框架开发者那么这可能会有关系。


setUpClass 和 tearDownClass
---------------------------

这些必须被实现为类方法:

   import unittest

   class Test(unittest.TestCase):
       @classmethod
       def setUpClass(cls):
           cls._connection = createExpensiveConnectionObject()

       @classmethod
       def tearDownClass(cls):
           cls._connection.destroy()

如果你希望在基类上的 "setUpClass" 和 "tearDownClass" 被调用则你必须自
己云调用它们。 在 "TestCase" 中的实现是空的。

如果在 "setUpClass" 中引发了异常则类中的测试将不会被运行并且
"tearDownClass" 也不会被运行。 跳过的类中的 "setUpClass" 或
"tearDownClass" 将不会被运行。 如果引发的异常是 "SkipTest" 异常则类将
被报告为已跳过而非发生错误。


setUpModule 和 tearDownModule
-----------------------------

这些应当被实现为函数:

   def setUpModule():
       createConnection()

   def tearDownModule():
       closeConnection()

如果在 "setUpModule" 中引发了异常则模块中的任何测试都将不会被运行并且
"tearDownModule" 也不会被运行。 如果引发的异常是 "SkipTest" 异常则模块
将被报告为已跳过而非发生错误。

要添加即使在发生异常时也必须运行的清理代码，请使用 "addModuleCleanup":

unittest.addModuleCleanup(function, /, *args, **kwargs)

   在 "tearDownModule()" 之后添加一个要调用的函数来清理测试类运行期间
   所使用的资源。 函数将按它们被添加的相反顺序被调用 (LIFO (last-in,
   first-out))。 它们在调用时将附带它们被添加时传给
   "addModuleCleanup()" 的任何参数和关键字参数。

   如果 "setUpModule()" 失败，即意味着 "tearDownModule()" 未被调用，则
   已添加的任何清理函数仍将被调用。

   3.8 版新加入.

unittest.doModuleCleanups()

   此函数会在 "tearDownModule()" 之后无条件地被调用，或者如果
   "setUpModule()" 引发了异常则会在 "setUpModule()" 之后被调用。

   它将负责调用由 "addCleanupModule()" 添加的所有清理函数。 如果你需要
   在 "tearDownModule()" *之前* 调用清理函数则可以自行调用
   "doModuleCleanups()"。

   "doModuleCleanups()" 每次会弹出清理函数栈中的一个方法，因此它可以在
   任何时候被调用。

   3.8 版新加入.


信号处理
========

3.2 版新加入.

unittest 的 "-c/--catch" 命令行选项，加上 "unittest.main()" 的
"catchbreak" 形参，提供了在测试运行期间处理 control-C 的更友好方式。
在捕获中断行为被启用时 control-C 将允许当前运行的测试能够完成，而测试
运行将随后结束并报告已有的全部结果。 第二个 control-C 将会正常地引发
"KeyboardInterrupt"。

处理 control-C 信号的句柄会尝试与安装了自定义 "signal.SIGINT" 处理句柄
的测试代码保持兼容。 如果是 "unittest" 处理句柄而 *不是* 已安装的
"signal.SIGINT" 处理句柄被调用，即它被系统在测试的下层替换并委托处理，
则它会调用默认的处理句柄。 这通常会是替换了已安装处理句柄并委托处理的
代码所预期的行为。 对于需要禁用 "unittest" control-C 处理的单个测试则
可以使用 "removeHandler()" 装饰器。

还有一些工具函数让框架开发者可以在测试框架内部启用 control-C 处理功能
。

unittest.installHandler()

   安装 control-C 处理句柄。 当接收到 "signal.SIGINT" 时（通常是响应用
   户按下 control-C）所有已注册的结果都会执行 "stop()" 调用。

unittest.registerResult(result)

   注册一个 "TestResult" 对象用于 control-C 的处理。 注册一个结果将保
   存指向它的弱引用，因此这并不能防止结果被作为垃圾回收。

   如果 control-C 未被启用则注册 "TestResult" 对象将没有任何附带影响，
   因此不论是否启用了该项处理测试框架都可以无条件地注册他们独立创建的
   所有结果。

unittest.removeResult(result)

   移除一个已注册的结果。 一旦结果被移除则 "stop()" 将不再会作为针对
   control-C 的响应在结果对象上被调用。

unittest.removeHandler(function=None)

   当不附带任何参数被调用时此函数将移除已被安装的 control-C 处理句柄。
   此函数还可被用作测试装饰器以在测试被执行时临时性地移除处理句柄:

      @unittest.removeHandler
      def test_signal_handling(self):
          ...
