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. Java 注解简介

    一,什么叫注解 用一个词就可以描述注解,那就是元数据,即一种描述数据的数据.所以,可以说注解就是源代码的元数据.比如,下面这段代码: 1 2 3 4 @Override public String t ...

  2. 量化学习 | Tushare 基本面选股 (二)

    量化投资比较重要的是策略,可是你得先选个好股,价值投资需要认同他的价值,值得投资的股票才有投资的机会,现在简单介绍一下基于基本面的选股,其实我现实生活中也有炒股,都是经验之说的选股原则. 首先从tus ...

  3. STM32 初学不知道外设对应的APB1还是APB2

    STM32参考手册搜索“系统架构”或者“系统结构”,即可查看外设挂在哪个时钟下,也就知道开启哪个时钟了.

  4. Natas25 Writeup(目录遍历、头部注入)

    Natas25: 打开页面,是一段引文以及可以选择语言的下拉list.查看源码,发现关键代码: function setLanguage(){ //选择语言 /* language setup */ ...

  5. 使用 Docker 部署 Spring Boot 项目

    Docker 介绍 Docker 属于 Linux 容器的一种封装,提供简单易用的容器使用接口.它是目前最流行的 Linux 容器解决方案. Docker 将应用程序与该程序的依赖,打包在一个文件里面 ...

  6. 强连通分量SCC 2-SAT

    强连通分量SCC 2-SAT 部分资料来自: 1.https://blog.csdn.net/whereisherofrom/article/details/79417926 2.https://ba ...

  7. 将SublimeText加入右键菜单

    将SublimeText加入右键菜单 Windows Registry Editor Version 5.00 [HKEY_CLASSES_ROOT\*\shell\SublimeText] @=&q ...

  8. 单调栈-Maximum Width Ramp

    2020-01-23 19:39:26 问题描述: 问题求解: public int maxWidthRamp(int[] A) { Stack<Integer> stack = new ...

  9. 超参数、验证集和K-折交叉验证

    本文首发自公众号:RAIS ​前言 本系列文章为 <Deep Learning> 读书笔记,可以参看原书一起阅读,效果更佳. 超参数 参数:网络模型在训练过程中不断学习自动调节的变量,比如 ...

  10. coding++ :HttpClientUtils 封装

    1.关键 JAR  <!-- <<===================>> httpClient <<===================>> ...