研究Selenium + python 自动化测试有近两个月了,不能说非常熟练,起码对selenium自动化的执行有了深入的认识。
从最初无结构的代码,到类的使用,方法封装,从原始函数调用,到重定义函数。从变量驱动,到数据驱动,再到关键字驱动,一步一步的默默走向自动化框架的构建。虽然还有没有投入使用,只是写几个demo,就慢慢的发现了 selenium自动用例脚本,相似功能地方,代码基本都是一样的,界面元素换个查找方式,把原来的使用 xpath方式,改为使用 id 查找,需要对每个用例脚本都要改,虽然几个用例看不出什么工作量,但是重复findElement的代码,已经让我们感到了代码的笨重。如果某些定位发生了改变,我们就得贯穿整个测试代码进行调整元素定位,这样就会导致我们的脚本在后期,难以维护。
        因此通过Page Object Model 我们可以创建更加健壮代码,并减少或者消除重复的测试代码,从而也能够提高代码的可读性,减少编写脚本的工作量。Page Object Model的实现,就是通过分离测试对象和测试脚本的抽象来实现的。
PO项目实战演示Demo源码:(源码会随着结构调整不断更新,但是整体结构不会变动)
 Page Object Model 简称POM 
下面通过一个实例演示POM的实现
在此之前,先看一个段比较普通的测试用例代码片段:
  1.  ....
    #测试用例
    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("")
    driver.find_element_by_id("loginBtn").click()
    ....
这段脚本实现,是126邮箱登录功能的一个测试用例,单纯从这段代码来看,并不能直观知道他实现了什么业务,而且对于同一个元素,都出现重复的定位代码,只一个简单的登录功能,一个用例只操作3个元素,就暴露出来了一些代码冗余和可读性的问题。那么如果有更多功能和更多用例,那么用例代码更加厚重,可读性也是大大降低,我们需要想办法对此代码进行优化,通过POM实现登录功能。
 
首先,我们要分离测试对象(元素对象)和测试脚本(用例脚本),那么我们分别创建两个脚本文件, LoginPage.py 用于定义页面元素对象,每一个元素都封装成组件(可以看做存放页面元素对象的仓库)  CaseLoginTest.py 测试用例脚本。我们的实现思想,一切元素和元素的操作组件化定义在Page页面,用例脚本页面,通过调用Page中的组件对象,进行拼凑成一个登录脚本。
在写这两个脚本之前,我先对WebDriver中的一些方法进行重定义,以方便我们在写PO的时候,更简洁,快速。
BasePage.py:
  1.  #-*- 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:

  2.  #-*- 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 ="" #用例执行体
    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理解有所帮助>

Selenium的PO模式(Page Object Model)|(Selenium Webdriver For Python)的更多相关文章

  1. Page Object Model (Selenium, Python)

    时间 2015-06-15 00:11:56  Qxf2 blog 原文  http://qxf2.com/blog/page-object-model-selenium-python/ 主题 Sel ...

  2. Selenium的PO模式(Page Object Model)[python版]

     Page Object Model 简称POM  普通的测试用例代码: .... #测试用例 def test_login_mail(self): driver = self.driver driv ...

  3. selenium page object model

    Page Object Model (POM) & Page Factory in Selenium: Ultimate Guide 来源:http://www.guru99.com/page ...

  4. Python+Selenium框架设计--- Page Object Model

    POM(Page Object Model):页面对象模型,POM是一种最近几年非常流行的自动化测试模型,或者思想,POM不是一个框架,就是一个解决问题的思想.采用POM的目的,是为了解决前端中UI变 ...

  5. selenium 的页面对象模型Page Object

    页面对象模型page object model是selenium中的一种脚本设计模式,它能将页面元素封装起来,与业务操作分隔开, 在页面变化改变时,无需去修改业务逻辑代码,提高脚本维护的效率. 1.p ...

  6. Appium+Python之PO模型(Page object Model)

    思考:我们进行自动化测试时,如果把代码都写在一个脚本中,代码的可读性会变差,且后期代码维护也麻烦,最好的想法就是测试对象和测试用例可以分离,可以很快定位问题,代码可读性高,也比较容易理解.这里推荐大家 ...

  7. Java&Selenium自动化测试之Page Object Model

    PO是什么: 1.页面对象模型(PO)是一种设计模式,用来管理维护一组web元素的对象库 2.在PO下,应用程序的每一个页面都有一个对应的page class 3.每一个page class维护着该w ...

  8. 基于Python Selenium Unittest PO设计模式详解

    本文章会讲述以下几个内容: 1.什么是PO设计模式(Page Object Model) 2.为什么要使用PO设计模式 3.使用PO设计模式要点 4.PO设计模式实例 1.什么是PO设计模式 (Pag ...

  9. python+selenium自动化软件测试(第7章):Page Object模式

    什么是Page ObjectModel模式Page Objects是selenium的一种测试设计模式,主要将每个页面看作是一个class.class的内容主要包括属性和方法,属性不难理解,就是这个页 ...

随机推荐

  1. merge 本地 master 分支代码提示 “Already up-to-date”

    在使用 git 的过程中由于误操作,导致从本地 master 分支 merge 代码到当前分支失败,虽然当前分支和 master 分支代码不同步,但是仍然提示 Already up-to-date. ...

  2. centos7 mariadb

    闲置已久的空间环境配置忘得差不多了,今天得空整理,重置了磁盘重新搭建环境,首先在CentOS 7.0安装MariaDB的数据库,在这里记录下安装过程,以便以后查看. 1.安装MariaDB 安装命令 ...

  3. java常见面试题及答案 11-20(JVM篇)

    11.JVM内存分哪几个区,每个区的作用是什么? Java虚拟机主要分为以下一个区: 方法区:1. 有时候也成为永久代,在该区内很少发生垃圾回收,但是并不代表不发生GC,在这里进行的GC主要是对方法区 ...

  4. [1]IP地址查询

    今天起开始玩百度APIStore里面的免费API.以前用过的有12306的:数据.接口,有时间整理出来,12306的有点乱就是了.还有扇贝以及有道的API,之前用在留言板里自动翻译,公司用过百度地图以 ...

  5. Python实现简单的记账本功能

    目标: 1.使用序列化cPickle 2.账户中钱要大于花费的钱,否则提示请存钱 2.编写函数,实现存钱,花钱,查询及退出功能 1.序列化 pickle是python实现序列化的模块,次模块存在使用C ...

  6. ASP.NET Core – 2300% More Requests Served Per Second

    http://www.ageofascent.com/asp-net-core-exeeds-1-15-million-requests-12-6-gbps/ ASP.NET Core – Excee ...

  7. 生产/消费 发送和接收消息---基于kombu和redis交互

    from kombu import Connection, Exchange, Queue media_exchange = Exchange('media', 'direct', durable=T ...

  8. 如何把Excel中的某列数值如何转换成文本格式,且兼容性最好?

    假设要把下方A列数值变成文本格式.有多种方法,其中下方的方法兼容性最好: 第一步:选择 A 列,然后,运行菜单 数据/分列... 命令,如下图: 第二步:默认 下一步: 第三步:默认 下一步: 第四步 ...

  9. C#模拟Http与Https请求框架实例

    using System.Text; using System.Net; using System.IO; using System.Text.RegularExpressions; using Sy ...

  10. 化繁为简,最简易的SqlDataSource配合Cache使用~

    最近一个小项目用了比较方便傻瓜的DevExpress控件,实践中DevExpress的控件确实很方便. 在用ASPxGridView控件的时候,为了偷懒结合了SqlDataSource使用,但一直在大 ...