pythonwebdriver自动化测试实战.docx

上传人:小飞机 文档编号:4888464 上传时间:2023-05-21 格式:DOCX 页数:25 大小:245.80KB
返回 下载 相关 举报
pythonwebdriver自动化测试实战.docx_第1页
第1页 / 共25页
pythonwebdriver自动化测试实战.docx_第2页
第2页 / 共25页
pythonwebdriver自动化测试实战.docx_第3页
第3页 / 共25页
pythonwebdriver自动化测试实战.docx_第4页
第4页 / 共25页
pythonwebdriver自动化测试实战.docx_第5页
第5页 / 共25页
亲,该文档总共25页,到这儿已超出免费预览范围,如果喜欢就下载吧!
资源描述

《pythonwebdriver自动化测试实战.docx》由会员分享,可在线阅读,更多相关《pythonwebdriver自动化测试实战.docx(25页珍藏版)》请在三一办公上搜索。

1、python webdriver 项目实战第5章测试模型与测试脚本优化第一节、测试模型介绍线性测试通过录制或编写脚本,一个脚本完成用户一套完整的操作,通过对脚本的回放来进行自动化测试。这 是早期进行自动化测试的一种形式;我们在上一章中练习使用webdriver API所编写的脚本也是这种形式。脚本一from selenium import webdriverimport timedriver = webdriver.Firefox()driver.get(wwww.xxx.)driver.find_element_by_id(tbUserName).send_keys(username)dri

2、ver.find_element_by_id(tbPassword).send_keys(123456)driver.find_element_by_id(btnLogin).click()#执行具体用例操作driver.quit ()脚本二from selenium import webdriverimport timedriver = webdriver.Firefox()driver.get(wwww.xxx.)driver.find_element_by_id(tbUserName).send_keys(username)driver.find_element_by_id(tbPass

3、word).send_keys(123456)driver.find_element_by_id(btnLogin).click()#执行具体用例操作driver.quit ()通过上面的两个脚本,我们很明显的发现它的问题:一个用例对应一个脚本,假如界面发生变化,用户名的属性发生改变,不得不需要对每一个脚本进行 修改,测试用例形成一种规模,我们可能将大量的工作用于脚本的维护,从而失去自动化的意义。这种模式下数据和脚本是混在一起的,如果数据发生变也也需要对脚本进行修改。这种模式下脚本的可重复使用率很低。模块化与库我们会清晰的发现在上面的脚本中,其实有不少内容是重复的;于是就有了下面的改进logi

4、n.py#登录模块def login():driver.find_element_by_id(tbUserName).send_keys(username)driver.find_element_by_id(tbPassword).send_keys(456123)driver.find_element_by_id(btnLogin).click()quit.py#退出模块def quit_():测试用例:#coding = utf-8from selenium import webdriver通过上面的代码发现,我们可以把脚本中相同的部分独立出来,形成模块或库;当脚本需要进行调 用。这样做有

5、两个好处:一方面提高了开发效率,不用重复的编写相同的脚本;另一方面提高了代码的复用。数据驱动数据驱动应该是自动化的一个进步;从它的本意来讲,数据的改变(更新)驱动自动化的执行,从而 引起结果改变。这显然是一个非常高级的概念和想法。其实,我们能做到的是下面的形式。d:abcdata.txtP dat t xt -记事本文件 编辑 格式 查看g)帮助也)chongshiselenium bokeyuan不管我们读取的是txt文件,还是csv、excel文件的之类,又或者是数组、字典函数。我们实现了数 据与脚本的分离,换句话说,我们实现了参数化。我们仍一千条数据,通过脚本的执行,可以返回一千条 结果

6、出来。同样的脚本执行不同的数据从而得到了不同的结构。是不是增强的脚本的复用性呢!其实,这对开发来说是完全没有什么技术含量的;对于当初QTP自动化工具来说确是一个买点,因 为它面对的大多是不懂开发的测试。关键字驱动理解了数据驱动,无非是把“数据”换成“关键字”,关键字的改变引起测试结果的改变。关键字驱动用编程方式就不太容易表现了。QTP、robot framework等自动化工具都提供了关键字 驱动(填表格)。好吧!我能说selenium IDE也是关键字驱动么?Untitled C ijRini :=LTniT :=LT g H tuperi/clickcall eli d=q手机7V/7Fl

7、ilJ手机图5.x转化成表格是这样的:http:/niviY-t aebaoxom.CommaudTargetValueopentypeId=q手机c-jckAndWaitCs=biin:on.ise3fch-su.bmitveri ftTextPresem手机图4.xSelenium IDE 脚本分:命令(command)、对象(command)、值(value)格式就那里不偏不移,通过这样的格式去描述不同的对象,从而引起最终结果的改变。也就是说一切 以对象为出发点。当然,这样的脚本,显然对于不懂代码的同学非常直观!我要找谁(对象)?怎么做(命令)?做什 么(值)?更高级的关键字驱动,可以自

8、己定义keyword然后“注册”到框架;从而实现更强大的功能和扩展性。 关键字更详细的理解可以看我偶像的那偏文章。这里简单介绍了自动化测试的几种不同的模型,虽然简单阐述了他们的优缺点,但他们并非后后 者淘汰前者的关系,在实施自动化更多的是以需求为出发点,混合的来使用以上模型去解决问题;使我们 的脚本更易于开发与维护。第二节、登录模块化通过上一节对测试模型的学习可以看到,在我们的目前的脚本中还是有很多可以模块化的地方,比如 登录模块。我们的每一个用例的执行都需要登录脚本,那可我们是否可以将登录脚本独立到单独的文件调 用。下面以快播私有云的登录退出测试用例为例:webcloud.py#coding

9、 = utf-8from selenium import webdriverfrom mon.by import Byfrom mon.keys import Keysfrom selenium.webdriver.support.ui import Selectfrom mon.exceptions import NoSuchElementException import unittest, timeclass Login(unittest.TestCase):def setUp(self):self.driver = webdriver.Firefox()self.driver.impli

10、citly_wait(30)self.base_url = passport.kuaibo.self.verificationErrors =self.accept_next_alert = True#私有云登录用例def test_login(self):driver = self.driverdriver.get(self.base_url + 7login/?referrer=http%3A%2F%2Fwebcloud.kuaibo.%2F)driver.maximize_window()#登陆driver.find_element_by_id(user_name).clear()dri

11、ver.find_element_by_id(user_name).send_keys(username)driver.find_element_by_id(user_pwd).clear()driver.find_element_by_id(user_pwd).send_keys(123456)driver.find_element_by_id(dl_an_submit).click()time.sleep(3)#新功能引导driver.find_element_by_class_name(guide-ok-btn).click()time.sleep(3)#退出driver.find_el

12、ement_by_class_name(Usertool).click()time.sleep(2)driver.find_element_by_link_text(退出).click()time.sleep(2)def tearDown(self):self.driver.quit()self.assertEqual(, self.verificationErrors)if _name_ = main_:unittest.main()从业务流程及用例分析,每一个自动化测试用例的执行过程为:先执行登录操作,然后执行具体的操 作(如文件/文件夹的创建、删除、移动、重命名等操作),最后执行退出操作

13、。如上面的测试用例,登录 与退出操作是相对固定的,那么我们可以把登录与退出操作模块化出去,然后调用,一方面不用写重复代 码,另一方面可以使测试用例更关注具体的用例代码。login.py在与webcloud.py相同的文件夹下创建login.py文件:#coding = utf-8from selenium import webdriverfrom mon.exceptions import NoSuchElementException import unittest, timedef login(self):driver = self.driverdriver.maximize_window(

14、)driver.find_element_by_id(user_name).clear()driver.find_element_by_id(user_name).send_keys(username)driver.find_element_by_id(user_pwd).clear()driver.find_element_by_id(user_pwd).send_keys(123456)driver.find_element_by_id(dl_an_submit).click()time.sleep(3)webcloud.py#coding = utf-8from selenium imp

15、ort webdriverfrom mon.by import By from mon.keys import Keysfrom selenium.webdriver.support.ui import Selectfrom mon.exceptions import NoSuchElementExceptionimport unittest, timeimport login #导入登录文件class Login(unittest.TestCase):def setUp(self):self.driver = webdriver.Firefox()self.driver.implicitly

16、_wait(30)self.base_url = passport.kuaibo.self.verificationErrors =self.accept_next_alert = True#私有云登录用例def test_login(self):driver = self.driverdriver.get(self.base_url + 7login/?referrer=http%3A%2F%2Fwebcloud.kuaibo.%2F)#调用登录模块login.login(self)#新功能引导driver.find_element_by_class_name(guide-ok-btn).c

17、lick()time.sleep(3)#退出driver.find_element_by_class_name(Usertool).click()time.sleep(2)driver.find_element_by_link_text(退出).click()time.sleep(2)def tearDown(self):self.driver.quit()self.assertEqual(, self.verificationErrors)if _name_ = main_:unittest.main()进行到这里,我们有必要补充一下python语言中函数、类、方法的使用,这将有助于我们自动

18、化测 试脚本的开发。下面打开python IDLE :函数的使用:#例1 def add(a,b):c=a + bprint c add(1,3)#例2 def add2(a = 1,b=3):c=a + breturn c d=add2() print d通过def关键字可创建函数,在例1中我们创建了 add()函数,默认接收两个参数化a、b,对a、b相 加结果给c,并将结果函数内打印。例2中创建了 add2()函数,这一次对a、b设置了默认值,同样对a、b做加法,并将结果用return返 回;d在接收add2()时用的是默认值,将后将d接收的结果打印。类与方法的使用:通过class关键字我

19、们创建了一个 Counter类,定义了 add()和subtract。两个方法分别来完成加法和减 法运算,并将计算结果打印。通过上面的例子我们明显的发现类的方法与函数有一个明显的区别,在类的方法中必须有个额外的第 一个参数(self),但在调用类的方法时却不必为这个参数赋值self参数所指的是对象本身,所以习惯性 地命名为self。为何Python给self赋值而你不必给self赋值?创建了一个类MyClass,实例化MyClass得到了 MyObject这个对象,然后调用这个对象的方法 MyObject.method(a,b),在这个过程中,Pytho口会自动转为 Myclass.metho

20、d(MyObject,a,b),这就是 Python 的self的原理。即使你的类的方法不需要任何参数,但还是得给这个方法定义一个self参数,虽然我们在 实例化调用的时候不用理会这个参数。下面回到用例本身来讨论如何模块化和调用的,在login.py文件中:def login(self):driver = self.driver这里用到的是方法,(driver = self.driver)drive为对象身的driver,这一句很重要,否则我们无法在 ligin()方法中使用driver操作浏览器。首先导入login文件,然后对文件中的login ()方法进行调用。下面笔者动手把退出的相关操作

21、也模块化出去吧!第三节、数据驱动(参数化)在测试模型一节的数据驱动中我们已经介绍了如何通过p ython的readlines()函数对百度输入信息进行 参数化设置,将其它循环的读取data.txt文件中每一行数据。这里再回顾一下实现参数化的方式。baidu_read_data.py#coding = utf-8from selenium import webdriverimport os,timesource = open(D:abcdata.txt, r)values = source.readlines()source.close()#执行循环for serch in values:bro

22、wser = webdriver.Firefox()browser.get(.baidu.)browser.find_element_by_id(kw).send_keys(serch)browser.find_element_by_id(su).click()browser.quit()open方法以只读方式(r)打开本地的data.txt文件,readlines方法是逐行的读取文件内容。通过for循环,serch可以每次获取到文件中的一行数据,在定位到百度的输入框后,将数据传入 send_keys(serch)。这样通过循环调用,直到文件的中的所有内容全被读取。登录参数化(读取txt文件)

23、现在按照上面的思路,对自动化脚本中用户、名密码进行参数化,通过python文档我们发现python 读取文件的方式有:整个文件读取、逐行读取、固定字节读取。并没有找到一次读取两条数据的好方法。仓腱两个文件,分别存放用户名密码,如图6.x:HErrL:aTi5.立本文档1 KBtiiEEiVirii. twt 攵*档文件()编辑(X)神文件印编竭其格式查看帮。testing36Q1231156图6.x打开之前编写的login.py文件,做如下修改:#coding二utf-8from selenium import webdriverfrom mon.exceptions import NoSuc

24、hElementException import unittest, time, ossource 二 open(D:selenium_pythondatausername.txt, r) #用户名文件 un 二 source.read() #读取用户名source.close()source2 二 open(D:selenium_pythondatapassword.txt, r) #密码文件 pw 二 source2.read() #读取密码source2.close()def login(self):driver 二 self.driverdriver.maximize_window()

25、driver.find_element_by_id(user_name).clear()driver.find_element_by_id(user_name).send_keys(un) driver.find_element_by_id(user_pwd).clear() driver.find_element_by_id(user_pwd).send_keys(pw) driver.find_element_by_id(dl_an_submit).click() time.sleep(3)分别打开两个txt文件,通过un和pw来接收用户名和密码信息,将接收的数据通过send_key(xx

26、)转 入到执行程序中。运行我们前面创建的webcloud.py文件,程序可以正常的执行。虽然这样做比较丑,但是确实达到了 数据与脚本分离的目的。缺点:虽然目的达到了这,但这样的实现有很多问题:1、用户名密码分别在不同的文件里,修改用户名和密码比较麻烦。2、username.txt和password.txt文件中只能保存一用户密码,无能很好的循环读取。登录参数化(函数)函数是我们前面刚介绍的python知识,函数可以预先给参数化赋值,借助这个特性,我们可以通过调用函数的方式对用户名密码进行参数化userinfo.pydef fun(un = testing,pw=123456):print su

27、ccess reader username and password!return un,pw我们为两个参数un和?可赋了初值,赋值内容如果是字符串需要加引号,如果是数字可以不需要引号。 再次打开login.py文件,做如下修改:#coding = utf-8from selenium import webdriverfrom mon.exceptions import NoSuchElementExceptionimport unittest, timeimport userinfo #导入函数#通过两个变量,来接收调用函数获得用户名&密码us,pw = userinfo.fun()#打印两

28、个变量print us,pwdef login(self):driver = self.driverdriver.maximize_window()driver.find_element_by_id(user_name).clear()driver.find_element_by_id(user_name).send_keys(un)driver.find_element_by_id(user_pwd).clear()driver.find_element_by_id(user_pwd).send_keys(pw)driver.find_element_by_id(dl_an_submit).

29、click()time.sleep(3)单独运行login.py文件:=RESTARTsuccess reader username and password!testing 123456说明我们对函数的参数调用是成功的,将un、pw两个变量的值传入用户名密码的send_keys()方法中 即可。登录参数化(读取字典)既然我们是固定的读取用户名和密码两个值,那么可以借助python字典的方式来完成这个需求。字典由大括号内的多键值对组成;下面继续在python IDLE交互模式下演示字典的创建与使用。 data = abc:123,def:456 print data|abc: 123, def

30、: 456 data.keys()abc, def data.values()123, 456 data.items()(abc, 123), (def, 456)创建字典用大括号,数据由key/value键值对组成,keys()方法返回字典中的键列表。values()返回字 典中的值列表,items()返回(key, value)元组。F面创建一存放字典的函数文件userinfo.py :def zidian():data=username:123456,testing360:123123print success reader username and password!return da

31、ta字典的可以方便的存放k,v键值对,一个键对应一个值;注意,如果密码中有非数字,需要加引号。需要说明的是我们的需求并不适合循环的读取用户名密码,不过我们可以写一小程序单独验证这种循环读取的方式:loop_reader.py#coding = utf-8import userinfo #导入函数#获取字典数据info = userinfo.zidian()#通过items()循环读取元组(键/值对)for us,pw in info.items():print usprint pw表 参数化 (csv)假如我有自动化脚本中要参数化一张表单,表单需要填写用户名、,年龄,性别等信息,使用上面的 方

32、法就很难来解决这个问题,下面通过读取csv文件的方法来解决这个问题。创建userinfo.csv文件,如图5.x图5.x通过WPS或excel创建表格,文件另存为选择CSV格式,下面修改loop_reader.py文件进行循环读取:#coding = utf-8import csv #导入 csv 包#读取本地CSV文件my_file=D:selenium_pythondatauserinfo.csvdata=csv.reader(file(my_file,rb)#循环输出每一行信息for user in data:print user0print user1print user2print

33、user3运行结果: = = = = = = = = = = = = = = = = = = = = = = = = = RESTART = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =testing123456126.23男testing2123123qq.18testing3456123gmail.29csv.reader()用于读取CSV文件,user0表示表格中第一列的数据(用户名),user0表示表格中第二 列的数据(),后面类推。通过CSV读取文件比较灵活,可以循环读取每一条数据,从而又不局限每次所读取数据的个数

34、。我们这里举例了多种方式来进行参数化,虽然最终看来CSV的方式最灵活,这里更多的是想告诉读者 解决问题的方法是多样的,我们可以用python的各种技巧来选择最简单的方法来解决问题。从这个过程 中我们也学到了不少python编程技术。我这里可以说是用python最基础的知识点解决了问题,这里只算提供一个思路,温习一下python, 你一定可以找到更完美优雅的方法;解决问题的方法是多样的,使用最贴合需求的方法,简单解决问题。 这一节写的比较多,对构建自动化框架来说,参数化是非常重要的一个知识点。脚本的模块化与参数化是我们在自动化脚本开发中用到最多的两个技巧,希望读者能认真体会,灵活 的运行到具体的

35、项目中。再说那个分层测试。吴博士的分层和乙醇的测试金字塔是一个道理就是从uiunit越是往ui层 投入产出比越低-test-可笑(372499885) 18:46:07你用一天时间写一个测登陆的所有的场景开发要改得是一个节点还好他要因为需求变动 把逻辑都改了你这就真的蛋疼了。所以我坐ui自动化 从来都是只测正向逻辑 重头戏应 该放在更深层的地方-PY- MR. 18:55:04胡胡 关于你说的登陆挂了,删除不能测试了。还是跑回原点了,你是在纸上谈兵,真正跑用 例的时候,登陆肯定是套件里面的一个单独用例,并且登陆用例先执行完毕才能继续向下, 这里就是整个一套测试系统的设计了。我们不要单独去讨论一个用例的好坏。一定要放在一个真实的场景中,包括业务逻辑,产品特性,设备情况,人员能力和素质。如果抛开这些, 我们讨论的东西会很美好,但不一定能用起来

展开阅读全文
相关资源
猜你喜欢
相关搜索

当前位置:首页 > 生活休闲 > 在线阅读


备案号:宁ICP备20000045号-2

经营许可证:宁B2-20210002

宁公网安备 64010402000987号