Selenium的PO模式(Page Object Model)|(Selenium Webdriver For Python)
2014-11-07 00:55
711 查看
研究Selenium + python 自动化测试有近两个月了,不能说非常熟练,起码对selenium自动化的执行有了深入的认识。
从最初无结构的代码,到类的使用,方法封装,从原始函数调用,到重定义函数。从变量驱动,到数据驱动,再到关键字驱动,一步一步的默默走向自动化框架的构建。虽然还有没有投入使用,只是写几个demo,就慢慢的发现了 selenium自动用例脚本,相似功能地方,代码基本都是一样的,界面元素换个查找方式,把原来的使用 xpath方式,改为使用 id 查找,需要对每个用例脚本都要改,虽然几个用例看不出什么工作量,但是重复findElement的代码,已经让我们感到了代码的笨重。如果某些定位发生了改变,我们就得贯穿整个测试代码进行调整元素定位,这样就会导致我们的脚本在后期,难以维护。
因此通过Page Object Model 我们可以创建更加健壮代码,并减少或者消除重复的测试代码,从而也能够提高代码的可读性,减少编写脚本的工作量。Page Object Model的实现,就是通过分离测试对象和测试脚本的抽象来实现的。
PO项目实战演示Demo源码:(源码会随着结构调整不断更新,但是整体结构不会变动)
https://coding.net/u/tsbc/p/PySelenium_PO/git
Page Object Model 简称POM
下面通过一个实例演示POM的实现
在此之前,先看一个段比较普通的测试用例代码片段:
这段脚本实现,是126邮箱登录功能的一个测试用例,单纯从这段代码来看,并不能直观知道他实现了什么业务,而且对于同一个元素,都出现重复的定位代码,只一个简单的登录功能,一个用例只操作3个元素,就暴露出来了一些代码冗余和可读性的问题。那么如果有更多功能和更多用例,那么用例代码更加厚重,可读性也是大大降低,我们需要想办法对此代码进行优化,通过POM实现登录功能。
首先,我们要分离测试对象(元素对象)和测试脚本(用例脚本),那么我们分别创建两个脚本文件, LoginPage.py 用于定义页面元素对象,每一个元素都封装成组件(可以看做存放页面元素对象的仓库) CaseLoginTest.py 测试用例脚本。我们的实现思想,一切元素和元素的操作组件化定义在Page页面,用例脚本页面,通过调用Page中的组件对象,进行拼凑成一个登录脚本。
在写这两个脚本之前,我先对WebDriver中的一些方法进行重定义,以方便我们在写PO的时候,更简洁,快速。
BasePage.py:
LoginPage.py:
通过使用POM进行重新构造代码结构后,发现代码测试用例代码的可读性提高很多,元素写成组件的方式,不需要每次都写findElement直接在脚本中调用组件就可以使用。在CaseLoginTest脚本用例执行体中,一旦我们输入 login_page并敲入一个点时,LoginPage页面中的元素对象组件都显示出来。并且定义好的PageObject组件可以重复在其它的脚本中进行使用,减少了代码的工作量,也方便对脚本进行后期的维护管理,当元素属性发生变化时,我们只需要对一个PageObaject页面中的对象组件定义进行更改即可。
<简单对Page Object Model使用实例进行描述,希望能够对Selenium自动化的POM理解有所帮助>
来自为知笔记(Wiz)
从最初无结构的代码,到类的使用,方法封装,从原始函数调用,到重定义函数。从变量驱动,到数据驱动,再到关键字驱动,一步一步的默默走向自动化框架的构建。虽然还有没有投入使用,只是写几个demo,就慢慢的发现了 selenium自动用例脚本,相似功能地方,代码基本都是一样的,界面元素换个查找方式,把原来的使用 xpath方式,改为使用 id 查找,需要对每个用例脚本都要改,虽然几个用例看不出什么工作量,但是重复findElement的代码,已经让我们感到了代码的笨重。如果某些定位发生了改变,我们就得贯穿整个测试代码进行调整元素定位,这样就会导致我们的脚本在后期,难以维护。
因此通过Page Object Model 我们可以创建更加健壮代码,并减少或者消除重复的测试代码,从而也能够提高代码的可读性,减少编写脚本的工作量。Page Object Model的实现,就是通过分离测试对象和测试脚本的抽象来实现的。
PO项目实战演示Demo源码:(源码会随着结构调整不断更新,但是整体结构不会变动)
https://coding.net/u/tsbc/p/PySelenium_PO/git
Page Object Model 简称POM
下面通过一个实例演示POM的实现
在此之前,先看一个段比较普通的测试用例代码片段:
.... #测试用例 def test_login_mail(self): driver = self.driver driver.get("http://mail.126.com") driver.find_element_by_id("idInput").clear() driver.find_element_by_id("idInput").send_keys("auto_tester") driver.find_element_by_id("pwdInput").clear() driver.find_element_by_id("pwdInput").send_keys("123456") driver.find_element_by_id("loginBtn").click() ....
这段脚本实现,是126邮箱登录功能的一个测试用例,单纯从这段代码来看,并不能直观知道他实现了什么业务,而且对于同一个元素,都出现重复的定位代码,只一个简单的登录功能,一个用例只操作3个元素,就暴露出来了一些代码冗余和可读性的问题。那么如果有更多功能和更多用例,那么用例代码更加厚重,可读性也是大大降低,我们需要想办法对此代码进行优化,通过POM实现登录功能。
首先,我们要分离测试对象(元素对象)和测试脚本(用例脚本),那么我们分别创建两个脚本文件, LoginPage.py 用于定义页面元素对象,每一个元素都封装成组件(可以看做存放页面元素对象的仓库) CaseLoginTest.py 测试用例脚本。我们的实现思想,一切元素和元素的操作组件化定义在Page页面,用例脚本页面,通过调用Page中的组件对象,进行拼凑成一个登录脚本。
在写这两个脚本之前,我先对WebDriver中的一些方法进行重定义,以方便我们在写PO的时候,更简洁,快速。
BasePage.py:
#-*- coding: utf-8 -*- __author__ ='tsbc' from selenium.webdriver.support.wait importWebDriverWait from selenium import webdriver classAction(object): """ BasePage封装所有页面都公用的方法,例如driver, url ,FindElement等 """ #初始化driver、url、等 def __init__(self, selenium_driver, base_url, pagetitle): self.base_url = base_url self.pagetitle = pagetitle self.driver = selenium_driver #打开页面,校验页面链接是否加载正确 def _open(self, url, pagetitle): #使用get打开访问链接地址 self.driver.get(url) self.driver.maximize_window() #使用assert进行校验,打开的链接地址是否与配置的地址一致。调用on_page()方法 assert self.on_page(pagetitle), u"打开开页面失败 %s"% url #重写元素定位方法 def find_element(self,*loc): #return self.driver.find_element(*loc) try: WebDriverWait(self.driver,10).until(lambda driver: driver.find_element(*loc).is_displayed()) return self.driver.find_element(*loc) except: print u"%s 页面中未能找到 %s 元素"%(self, loc) #重写switch_frame方法 def switch_frame(self, loc): return self.driver.switch_to_frame(loc) #定义open方法,调用_open()进行打开链接 def open(self): self._open(self.base_url, self.pagetitle) #使用current_url获取当前窗口Url地址,进行与配置地址作比较,返回比较结果(True False) def on_page(self, pagetitle): return pagetitle in self.driver.title #定义script方法,用于执行js脚本,范围执行结果 def script(self, src): self.driver.execute_script(src) #重写定义send_keys方法 def send_keys(self, loc, vaule, clear_first=True, click_first=True): try: loc = getattr(self,"_%s"% loc) if click_first: self.find_element(*loc).click() if clear_first: self.find_element(*loc).clear() self.find_element(*loc).send_keys(vaule) exceptAttributeError: print u"%s 页面中未能找到 %s 元素"%(self, loc)
LoginPage.py:
#-*- coding: utf-8 -*- __author__ ='tsbc' from selenium.webdriver.common.by importBy importBasePage #继承BasePage类 classLoginPage(BasePage.Action): #定位器,通过元素属性定位元素对象 username_loc =(By.ID,"idInput") password_loc =(By.ID,"pwdInput") submit_loc =(By.ID,"loginBtn") span_loc =(By.CSS_SELECTOR,"div.error-tt>p") dynpw_loc =(By.ID,"lbDynPw") userid_loc =(By.ID,"spnUid") #Action def open(self): #调用page中的_open打开连接 self._open(self.base_url, self.pagetitle) #调用send_keys对象,输入用户名 def input_username(self, username): self.find_element(*self.username_loc).send_keys(username) #调用send_keys对象,输入密码 def input_password(self, password): self.find_element(*self.password_loc).send_keys(password) #调用send_keys对象,点击登录 def click_submit(self): self.find_element(*self.submit_loc).click() #用户名或密码不合理是Tip框内容展示 def show_span(self): return self.find_element(*self.span_loc).text #切换登录模式为动态密码登录(IE下有效) def swich_DynPw(self): self.find_element(*self.dynpw_loc).click() #登录成功页面中的用户ID查找 def show_userid(self): return self.find_element(*self.userid_loc).text CaseLoginTest.py: #-*- coding: utf-8 -*- __author__ ='tsbc' import sys reload(sys) sys.setdefaultencoding('utf-8') import unittest from PO importLoginPage from selenium import webdriver classCaselogin126mail(unittest.TestCase): """ 登录126邮箱的case """ @classmethod def setUpClass(cls): cls.driver = webdriver.Chrome() cls.driver.implicitly_wait(30) cls.url ="http://mail.126.com" cls.username ="auto_tester" cls.password ="123456" #用例执行体 def test_login_mail(self): #声明LoginPage类对象 login_page =LoginPage.LoginPage(self.driver, self.url, u"网易") #调用打开页面组件 login_page.open() #调用用户名输入组件 login_page.input_username(self.username) #调用密码输入组件 login_page.input_password(self.password) #调用点击登录按钮组件 login_page.click_submit() @classmethod def tearDownClass(cls): cls.driver.quit() if __name__ =="__main__": unittest.main()
通过使用POM进行重新构造代码结构后,发现代码测试用例代码的可读性提高很多,元素写成组件的方式,不需要每次都写findElement直接在脚本中调用组件就可以使用。在CaseLoginTest脚本用例执行体中,一旦我们输入 login_page并敲入一个点时,LoginPage页面中的元素对象组件都显示出来。并且定义好的PageObject组件可以重复在其它的脚本中进行使用,减少了代码的工作量,也方便对脚本进行后期的维护管理,当元素属性发生变化时,我们只需要对一个PageObaject页面中的对象组件定义进行更改即可。
<简单对Page Object Model使用实例进行描述,希望能够对Selenium自动化的POM理解有所帮助>
来自为知笔记(Wiz)
相关文章推荐
- Selenium的PO模式(Page Object Model)|(Selenium Webdriver For Python)
- Selenium的PO模式(Page Object Model)[python版]
- Selenium+Python :WebDriver设计模式( Page Object )
- Selenium的PO模式(Page Object Model)[python版]
- Python3-Selenium3使用PO设计模式(Page Object)实现简单的页面登录操作
- Page Object Model (Selenium, Python)(一)
- Selenium Page Object(PO)设计模式
- Page Object Model (Selenium + Python)(三)
- Page Object Model (Selenium, Python)
- PO模式(Page Object Model)
- Webdriver的设计模式:Page Object(页面模型)
- Selenium自动化测试Python五:WebDriver设计模式
- Python+Selenium框架设计--- Page Object Model
- selenium启动IE报错:页面提示:This is the initial start page for the WebDriver server.
- Selenium-webdriver 页面模式在实际项目中的实际运用,大量Page页面如何初始化?
- 轻松自动化---selenium-webdriver(python) (六)
- selenium webdriver (python)
- selenium-webdriver(python) 11
- selenium-webdriver(python) (十六) --unittest 框架
- selenium-webdriver(python) (十四) -- webdriver原理