1、PO模式的思想

  1. 原理:
    1. 将页面的元素定位和元素行为封装成一个page类
    2. 类的属性:元素的定位
    3. 类的行为:元素的操作 
  2. 页面对象和测试用例分离
  3. 测试用例:
    1. 调用所需要页面对象中的行为,组成测试用例
    2. 测试用例中,只需要含有页面函数的调用和断言,不应该出现元素定位等其他的操作,如果写测试用例中出现需要其他的额外的操作,可以想办法封装到页面对象或者元素定位中
  4. 好处:
    1. 当某个页面的元素发生变化,只需要修改该页面对象中的代码即可,测试用例不需要修改
    2. 提高代码的重用率,结构清晰,维护代码容易
    3. 测试用例发生变化是,不需要或者只需要修改少数页面对象代码即可

2、使用unittest框架实现PO模式

  1. 新建一个包,命名为PageObjects,包中用来封装各个页面的功能
    1. 页面对象封装的类中,函数用来实现页面的功能,在类的初始化函数中,使用传参(driver)的方式完成,页面只需要实现对应功能即可,具体传入什么样的测试,在测试用例的前置条件中实现,可以提高函数的重用率
      1. 页面对象封装函数
         from selenium.webdriver.support import expected_conditions as EC
        from selenium.webdriver.support.wait import WebDriverWait
        from selenium.webdriver.common.by import By
        class LoginPage:
        def __init__(self,driver):
        self.driver=driver def login(self,username,pwd,remember_user=True):
        # 等待定位元素出现
        WebDriverWait(self.driver,10).until(EC.visibility_of_element_located((By.XPATH,"//input[@name='phone']")))
        # 输入用户名
        self.driver.find_element_by_xpath("//input[@name='phone']").send_keys(username)
        # 输入密码
        self.driver.find_element_by_xpath("//input[@name='password']").send_keys(pwd)
        # 定位记住手机号元素
        remember_ele=self.driver.find_element_by_xpath("//input[@name='remember_me']")
        # 判断时候记住手机号
        if remember_user==True:
        remember_ele.checked=True
        else:
        remember_ele.checked=False
        # 点击登录
        self.driver.find_element_by_xpath("//button[text()='登录']").click()
      2. 测试用例调用页面对象函数,代码如下

        1     def test_login_success(self):
        2 # 2、步骤
        3 # 实例化LoginPage类
        4 lg=LoginPage(self.driver)
        5 # 调用login方法
        6 lg.login("登录账号","登录密码")
        7 # 3、断言
        8 self.assertTrue(IndexPage(self.driver).isExist_logout_ele())
      3. Index_page页面封装函数代码
        from selenium.webdriver.support import expected_conditions as EC
        from selenium.webdriver.support.wait import WebDriverWait
        from selenium.webdriver.common.by import By
        class IndexPage:
        def __init__(self,driver):
        self.driver=driver def isExist_logout_ele(self):
        try:
        WebDriverWait(self.driver,10).until(EC.visibility_of_element_located((By.XPATH,"//a[text()='退出']")))
        return True
        except:
        return False
      4. 测试用例的前置条件代码
        import unittest
        from selenium import webdriver
        from PageObjects.login_page import LoginPage
        from PageObjects.index_page import IndexPage
        class TestLogin(unittest.TestCase): def setUp(self):
        # 1、前置条件
        url="http://ip:host/Index/login.html"
        self.driver=webdriver.Chrome()
        self.driver.maximize_window()
        self.driver.get(url)
      5. 运行测试用例可以在测试用例最后面使用一下代码,然后运行该代码
        if __name__ == '__main__':
        unittest.main()
      6. 运行结果如下,表示运行成功 
  2. 新建一个包,命名为TestCases,包中用来写测试用例,测试用例中用到那个页面对象中的函数时,只需要调用页面对象中的函数就可以了
  3. 新建一个包,命名为TestDatas,用来写测试数据,将测试数据的类导入到测试用例模块中,进行使用,这样的话,测试数据便于管理,测试数据添加、修改、删除不用修改测试用例
    1. 在TestCases下面新建一个公用的数据管理.py文件Common_Datas.py和模块数据的.py文件login_datas.py,如下t图:

      #Common_Datas.py
      web_login_url="http://ip:port/Index/login.html" #login_datas.py
      #正常场景---测试数据
      success_data={"user":"","pwd":""} # 异常用例--手机号格式不正确(大于11位,小于11位,为空,不在号码段)
      error_data=[
      {"user": "", "pwd": "","check":"请输入正确的手机号"},
      {"user": "", "pwd": "","check":"请输入正确的手机号"},
      {"user": "", "pwd": "","check":"请输入正确的手机号"},
      {"user": "", "pwd": "","check":"请输入手机号"},
      {"user": "", "pwd": "","check":"请输入密码"},
      {"user": "", "pwd": "","check":"请输入手机号"},
      ]
    2. 在测试用例模块使用数据管理模块中的数据
       import unittest
      from selenium import webdriver
      from PageObjects.login_page import LoginPage
      from PageObjects.index_page import IndexPage
      from TestDatas import Common_Datas as CD
      from TestDatas import login_datas as ld
      from ddt import ddt,data @ddt
      class TestLogin(unittest.TestCase):
      @classmethod
      def setUpClass(cls):
      # 1、前置条件
      cls.driver=webdriver.Chrome()
      cls.driver.maximize_window()
      cls.driver.get(CD.web_login_url)
      # 实例化LoginPage类
      cls.lg=LoginPage(cls.driver)
      @classmethod
      def tearDownClass(cls):
      cls.driver.quit() def tearDown(self):
      self.driver.refresh()
      pass
      # 正常用例--登录成功
      def test_login_1_success(self):
      # 2、步骤 # 调用login方法
      self.lg.login(ld.success_data['user'],ld.success_data['pwd'])
      # 3、断言
      self.assertTrue(IndexPage(self.driver).isExist_logout_ele())
      # 异常用例--手机号码格式不正确(大于11位,小于11位,为空,不在号码段) ddt
      @data(*ld.error_data)
      def test_login_0_user_wrongFormat(self,item):
      self.lg.login(item['user'],item['pwd'])
      self.assertEqual(self.lg.get_errorMsg_form_loginArea(),item['check'])

      备注:

      setUpClass(cls)函数和tearDownClass(cls)只在整个用例开始和结束时运行一次,必须使用@classmethod来修饰,如上面的异常测试用例有6条,在用例开始前和结束时只运行一次 
      setUp(self)函数和tearDown(self)在每个测试用例开始和结束时运行,如果使用这两个函数,上面的6条用例,每运行一个,就会运行一次这两个函数,总共运行6次
      tearDown(self)可以和setUpClass(cls)函数和tearDownClass(cls)函数结合使用,如上面的界面刷新self.driver.refresh()和页面关闭cls.driver.quit()
    3. 测试用例类型相同的测试数据可以设计在一起,使用ddt在测试用例运行时分别执行测试用例
    4. 使用unittest框架,运行测试用例,如下:
       import unittest
      from TestCases.test_login import TestLogin
      from HTMLTestRunnerCN import HTMLTestReportCN
      # 创建一个容器,用来存测试用例
      suite=unittest.TestSuite()
      # 加载测试用例的类的实例
      loader=unittest.TestLoader()
      # 将测试用例加载到suite容器中
      suite.addTest(loader.loadTestsFromTestCase(TestLogin))
      # 打开文件,用来写测试报告
      with open("test_result.html",'wb') as file: # runner=unittest.TextTestRunner(verbosity=1) runner=HTMLTestReportCN( stream=file, verbosity=2, title="web自动化测试", description="第一个web自动化测试", tester="wsk")
      # 运行测试用例
      runner.run(suite)
    5. 运行结果,如下图:

      在写测试用例时,如果使用unittest框架可以使用函数名来控制代码的执行顺序,如:

      ok test_login_0_user_wrongFormat_6 (TestCases.test_login.TestLogin)  --------------------  0  
      ok test_login_1_success (TestCases.test_login.TestLogin)------------------------------------------1    前面都一样,0比1小,先运行

  

web自动化测试中的PO模式(一)的更多相关文章

  1. web自动化测试中绕开验证码登陆的方式

    web自动化测试中登陆需验证码是很大的一个困扰.现推荐一种简单的避开验证码登陆的方式,先代码进入登录页,人工输入验证码登录后浏览器自动保存cookie,再在新的标签中登录. 具体代码如下: publi ...

  2. python web自动化测试中失败截图方法汇总

    在使用web自动化测试中,用例失败则自动截图的网上也有,但实际能落地的却没看到,现总结在在实际应用中失败截图的几种方法: 一.使用unittest框架截图方法:   1.在tearDown中写入截图的 ...

  3. JavaScript在web自动化测试中的作用

    前言 JS的全称JavaScript,是一种运行在浏览器中的解释型脚本语言,通常用来实现web前端页面的基本功能,对于前端开发人员是不得不掌握的一门基本技能,但是对于做web自动化测试的人员来说,如果 ...

  4. 说说UI自动化中的PO模式

    PO模式,全称PageObject模式,即页面对象模式.将页面定位与业务操作分离. po模式有以下几个优点: 1.易读性好 2.扩展性高 3.复用性强 4.维护性好 5.代码冗余率低 了解了po模式及 ...

  5. TestNG测试框架在基于Selenium进行的web自动化测试中的应用

    转载请注明出自天外归云的博客园:http://www.cnblogs.com/LanTianYou/ TestNG+Selenium+Ant TestNG这个测试框架可以很好的和基于Selenium的 ...

  6. (转)Web自动化测试中的接口测试

    1.背景 1.1 Web程序中的接口 1.1.1 典型的Web设计架构 web是实现了基于网络通信的浏览器客户端与远程服务器进行交互的应用,通常包括两部分:web服务器和web客户端.web客户端的应 ...

  7. Web自动化测试中的接口测试

    1.2.3 接口可测性分析 接口显而易见要比UI简单的都,只需要知道协议和参数即可完成一次请求,从自动化测试实施难易程度来看,有以下几个特征: 1)驱动执行接口的自动化成本不高:HTTP,RPC,SO ...

  8. 高访问量WEB开发中的架构模式,学习从点滴开始

     当一个Web系统从日访问量10万逐步增长到1000万,甚至超过1亿的过程中,Web系统承受的压力会越来越大,在这个过程中,我们会遇到很多的问题.为了解决这些性能压力带来问题,我们需要在Web系统架构 ...

  9. 对web应用中单一入口模式的理解及php实现

    在我们web应用的开发中,经常会听见或看见单一入口模式,在我开始学习tp框架的时候也不理解为什么要运用一个单一入口模式,只是会使用,最近自己在搞一个小东西的时候才明白为什么在web开发中要运用单一入口 ...

随机推荐

  1. Scala尾递归

    递归函数应用 首先,我们来对比两个递归方法的求值步骤. 假设有方法gcd,用来计算两个数的最大公约数.下面是欧几里得算法的实现: def gcp(a: Int, b: Int): Int = if ( ...

  2. oracle中表空间管理,游标

    一.表空间 oracle使用表空间来管理数据库的对象(表,序列,过程,函数,游标等). oracle的逻辑结构(看不见的):oracle数据库 => 表空间 => 表 序列  过程等对象. ...

  3. ubuntu 16.04服务器安装apache2 + php + mysql

    具体操作 第一步:安装mysql apt-get install mysql-server mysql-client 第二步:安装apache2 apt-get install apache2 第三步 ...

  4. Java 借助poi操作PDF工具类

    ​ 一直以来说写一个关于Java操作PDF的工具类,也没有时间去写,今天抽空写一个简单的工具类,拥有PDF中 换行,字体大小,字体设置,字体颜色,首行缩进,居中,居左,居右,增加新一页等功能,如果需要 ...

  5. ArrayList,HashSet,SortedSet之间的区别是什么?

    今天看Redis官方案例,出现了列表和集合概念,列表在Java中指的就是List,集合在Java中指的就是Set,那么怎么实现列表和集合,以及它们有什么区别呢? 我写了个Demo演示下: import ...

  6. Topshelf+Quartz3.0基于控制台应用程序快速开发可调度windows服务

    1.TopShelf TopShelf是一个开源的跨平台的宿主服务框架.可通过.Net Core/.Net Framwork控制台应用程序快速开发windows服务,更加便于服务调试. 本文基于.Ne ...

  7. Mybatis三剑客介绍

    1.MyBatis generator 利用mybatis-generator自动生成代码 下载地址:  https://download.csdn.net/download/qq_36625806/ ...

  8. Django之Ajax传输数据

    MTV与MVC模型 MTV与MVC都是模型,只不过MTV是django自己定义的,具体看一下他们的意思 MTV模型(django) M:模型层(models.py) T:templates文件夹 V: ...

  9. 《数据库优化》- MySQL视图

    一.什么是视图 视图,是基于一个表或多个表或视图的逻辑表,本身不包含数据,通过它可以对表里面的数据进行查询和修改,视图基于的表称为基表.视图是存储在数据字典里的一条select语句. 通俗地讲,视图就 ...

  10. 01.为什么要学习Go语言

    为什么互联网世界需要Go语言 世界上已经有太多太多的编程语言了,为什么又出来一个Go语言? 硬件限制:摩尔定律已然失效 摩尔定律:当价格不变时,集成电路上可容纳的元器件的数目,约每隔18-24个月便会 ...