51Testing软件测试论坛

 找回密码
 (注-册)加入51Testing

QQ登录

只需一步,快速开始

微信登录,快人一步

手机号码,快捷登录

查看: 4549|回复: 4
打印 上一主题 下一主题

[转贴] 测试驱动开发与Python

[复制链接]
  • TA的每日心情
    无聊
    昨天 09:05
  • 签到天数: 1050 天

    连续签到: 1 天

    [LV.10]测试总司令

    跳转到指定楼层
    1#
    发表于 2016-7-29 15:28:15 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
    1. 最近在看一本书《Test-Driven Development with Python》,里面非常详细的介绍了如何一步一步通过测试驱动开发(TDD)的方式开发Web项目。刚好这本书中使用了我之前所了解的一些技术,Django、selenium、unittest等。所以,读下来受益匪浅。

    2.   我相信不少开发都写单元测试,不过,一般是先写功能代码,然后,再写单元测试用例,在编写单元测试用例的过程中,可能需要调整功能代码,从而使单元测试用例通过。但是TDD就特别要求先写测试用例,后写实现代码。这一开始确实有些难。

    3.   这里就选择一个简单的例子向各位介绍一下TDD的流程(套路)。



    4. 编写功能测试用例:                                                

    5.   首先,编写功能测试用例,functional_tests.py

    6. 复制代码
    7. from selenium import webdriver

    8. browser = webdriver.Firefox()
    9. browser.get("http://127.0.0.1:8000")

    10. assert "Django" in browser.title
    11. 复制代码
    12.   你没看错,这就是由Selenium编写的功能测试代码。打开Firefox浏览器,并访问http://127.0.0.1:8000,通过assert 判断浏览器标题是否包含“Django”。

    13.   然后,运行该测试用例。

    14. D:\pydj>python functional_tests.py
    15. Traceback (most recent call last):
    16.   File "functional_tests.py", line 6, in <module>
    17.     assert "Django" in browser.title
    18. AssertionError
    19.   测试用例失败了,这是必然的,因为我们还没有创建被测试的项目。但,这同样也是我们想要的结果。TDD的套路就是通过编写功能代码,使测试用例通过。





    20. 创建项目:                                                           

    21.   接下来创建Django项目:

    22.   D:\pydj>django-admin startproject superlists



    23.   当前项目结构如下:

    24. ├─ functional_tests.py
    25. └─ superlists
    26.   ├─ manage.py
    27.   └─ superlists
    28.     ├─ __init__.py
    29.     ├─ settings.py
    30.     ├─ urls.py
    31.     └─ wsgi.py



    32.   进入项目目录,启动项目:

    33. 复制代码
    34. D:\pydj> cd superlists
    35. D:\pydj\superlists>python manage.py runserver
    36. Performing system checks...

    37. System check identified no issues (0 silenced).

    38. You have unapplied migrations; your app may not work properly until they are applied.
    39. Run 'python manage.py migrate' to apply them.
    40. June 13, 2016 - 23:23:29
    41. Django version 1.9.7, using settings 'superlists.settings'
    42. Starting development server at http://127.0.0.1:8000/
    43. Quit the server with CTRL-BREAK.
    44. 复制代码
    45.   再次运行功能测试用例,functional_tests.py



    46. 接下来继续编写功能测试用例。functional_tests.py

    47. 复制代码
    48. #coding=utf-8
    49. from selenium import webdriver
    50. import unittest


    51. class NewVisitorTest(unittest.TestCase):

    52.     def setUp(self):
    53.         self.browser = webdriver.Firefox()
    54.         self.browser.implicitly_wait(3)

    55.     def tearDown(self):
    56.         self.browser.close()

    57.     def test_case_start_a_list_and_retrieve_it_later(self):
    58.         # 小明听说有一个很酷的在线代办事项应用
    59.         # 她去看了这个应用首页
    60.         self.browser.get("http://127.0.0.1:8000")

    61.         # 它注意到网页的标题和头部包含“To-Do”这个词语。
    62.         self.assertIn("To-Do", self.browser.title)


    63. if __name__ == '__main__':
    64.     unittest.main()
    65. 复制代码
    66. 这里用到了unittest 单元测试框架。如果,你不懂Python的单元测试,建议读者去学习unittest。



    67. 执行测试用例:

    68. 复制代码
    69. C:\Python35\python.exe D:/pydj/functional_tests.py
    70. F
    71. ======================================================================
    72. FAIL: test_case_start_a_list_and_retrieve_it_later (__main__.NewVisitorTest)
    73. ----------------------------------------------------------------------
    74. Traceback (most recent call last):
    75.   File "D:/pydj/functional_tests.py", line 21, in test_case_start_a_list_and_retrieve_it_later
    76.     self.assertIn("To-Do", self.browser.title)
    77. AssertionError: 'To-Do' not found in 'Welcome to Django'

    78. ----------------------------------------------------------------------
    79. Ran 1 test in 3.491s

    80. FAILED (failures=1)
    81. 复制代码


    82. 测试用例又在预料之内的失败了!先不要着急解决这个问题,把项目创建完成。

    83. D:\pydj\superlists>python3 manage.py startapp lists

    84. 将functional_tests.py放到superlists项目目录下。







    85. 单元测试与功能测试的区别:                                    

    86.   正如给事物所贴的众多标签一样,单元测试和功能测试之间的界线有时不那么清晰。不过,二者之间有个基本区别:功能测试站在用户的角度从外部测试应用,单元测试则站在程序员的角度从内部测试应用。

    87. 我遵从的 TDD 方法同时使用这两种类型测试应用。采用的工作流程大致如下。

    88.   (1) 先写功能测试,从用户的角度描述应用的新功能。

    89.   (2) 功能测试失败后,想办法编写代码让它通过(或者说至少让当前失败的测试通过)。此时,使用一个或多个单元测试定义希望代码实现的效果,保证为应用中的每一行代码

    90.   (3) 单元测试失败后,编写最少量的应用代码,刚好让单元测试通过。有时,要在第 2 步和第 3 步之间多次往复,直到我们觉得功能测试有一点进展为止。

    91.   (4) 然后,再次运行功能测试,看能否通过,或者有没有进展。这一步可能促使我们编写一些新的单元测试和代码等。

    92.   由此可以看出,这整个过程中,功能测试站在高层驱动开发,而单元测试则从低层驱动我们做些什么。



    93. 打开/lists/tests.py文件,编写单元测试。

    94. 复制代码
    95. from django.test import TestCase


    96. # Create your tests here.
    97. class SmokeTest(TestCase):

    98.     def test_bad_moths(self):
    99.         self.assertEqual(1 + 1,2)
    100. 复制代码
    101.   运行单元测试:

    102. 复制代码
    103. D:\pydj\superlists>python3 manage.py test
    104. Creating test database for alias 'default'...
    105. .
    106. ----------------------------------------------------------------------
    107. Ran 1 test in 0.001s

    108. OK
    109. Destroying test database for alias 'default'...
    110. 复制代码
    111.   OK,说明单元测试没问题。接下来就要编写真正的单元测试了(/lists/tests.py)。

    112. 复制代码
    113. from django.core.urlresolvers import resolve
    114. from django.test import TestCase
    115. from django.http import HttpRequest

    116. from lists.views import home_page

    117. class HomePageTest(TestCase):

    118.     def test_root_url_resolves_to_home_page_view(self):
    119.         found = resolve('/')
    120.         self.assertEqual(found.func, home_page)

    121.     def test_home_page_returns_correct_html(self):
    122.         request = HttpRequest()
    123.         response = home_page(request)
    124.         self.assertTrue(response.content.startswith(b'<html>'))
    125.         self.assertIn(b'<title>To-Do lists</title>', response.content)
    126.         self.assertTrue(response.content.endswith(b'</html>'))
    127. 复制代码
    128.   第一个用例(test_root_url_resolves_to_home_page_view):

    129.    resolve 是 Django 内部使用的函数,用于解析 URL,并将其映射到相应的视图函数上。检查解析网站根路径“ /” 时,是否能找到名为 home_page 的函数。

    130.   第二个用例(test_home_page_returns_correct_html):

    131.   创建了一个 HttpRequest 对象,用户在浏览器中请求网页时, Django 看到的就是HttpRequest 对象。把这个 HttpRequest 对象传给 home_page 视图,得到响应。听说响应对象是 HttpResponse类的实例时,你应该不会觉得奇怪。接下来我们断定响应的 .content 属性(即发送给用户的 HTML)中有特定的内容。

    132.   assertTrue()希望响应以 <html> 标签开头,并在结尾处关闭该标签。注意, response.content 是原始字节,不是 Python 字符串,因此对比时要使用 b'' 句法。b是BYTE字符串

    133.   assertIn()希望响应中有一个 <title> 标签,其内容包含单词“ To-Do”——因为在功能测试中做了这项测试。

    134.   在书中,这里的单元测试也不是一次写成的,这里省略中间过程,一次写成一个相对健全的单元测试。

    135.   根据单元测试编写视图文件lists/views.py

    136. 复制代码
    137. from django.shortcuts import render
    138. from django.http import HttpResponse

    139. # Create your views here.
    140. def home_page(request):
    141.      return HttpResponse('<html><title>To-Do lists</title></html>')
    142. 复制代码
    143.   运行单元测试使其通过:

    144. 复制代码
    145. D:pydj\superlists>python3 manage.py test
    146. Creating test database for alias 'default'...
    147. ..
    148. ----------------------------------------------------------------------
    149. Ran 2 tests in 0.002s

    150. OK
    151. Destroying test database for alias 'default'...
    152. 复制代码
    153.   最后不要忘了,配置superlists/urls.py文件。

    154. urlpatterns = [
    155.     #url(r'^admin/', admin.site.urls),
    156.     url(r'^

    157. , views.home_page),
    158. ]


    159.   最后的最后,启动服务:

    160.   D:\pydj\superlists>python manage.py runserver

    161.   运行功能测试用例使其通过:

    162. 复制代码
    163. C:\Users\fnngj\Desktop\superlists>python3 functional_tests.py
    164. F
    165. ======================================================================
    166. FAIL: test_can_start_a_list_and_retrieve_it_later (__main__.NewVisitorTest)
    167. ----------------------------------------------------------------------
    168. Traceback (most recent call last):
    169.   File "functional_tests.py", line 21, in test_can_start_a_list_and_retrieve_it_later
    170.     self.fail('Finish the test!')
    171. AssertionError: Finish the test!

    172. ----------------------------------------------------------------------
    173. Ran 1 test in 7.070s

    174. FAILED (failures=1)
    175. 复制代码
    复制代码


    分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
    收藏收藏4
    回复

    使用道具 举报

  • TA的每日心情
    擦汗
    2024-11-1 14:56
  • 签到天数: 1182 天

    连续签到: 1 天

    [LV.10]测试总司令

    3#
    发表于 2016-7-29 16:51:02 | 只看该作者
    从另外一个角度去测试,很不错,可以学习下
    回复 支持 反对

    使用道具 举报

    该用户从未签到

    5#
    发表于 2016-8-2 12:31:04 来自手机 | 只看该作者
    什么都没看懂
    回复 支持 反对

    使用道具 举报

    本版积分规则

    关闭

    站长推荐上一条 /1 下一条

    小黑屋|手机版|Archiver|51Testing软件测试网 ( 沪ICP备05003035号 关于我们

    GMT+8, 2024-11-22 07:55 , Processed in 0.069258 second(s), 26 queries .

    Powered by Discuz! X3.2

    © 2001-2024 Comsenz Inc.

    快速回复 返回顶部 返回列表