selenium3 使用PageObject进行自动化测试

1、前提
做过一段时间UI层自动化,就会深深的体会到一个痛点:版本迭代太快,UI层元素的属性经常变换,导致维护人员需要花大把的时间去维护代码,为了节省维护成本及时间,就可以利用PageObject 这种设计模式,它就大大的减少了维护时间。
PageObject设计模式:是将某个页面的所有"元素(包含控件)属性"及"元素操作"封装在某个特定的类(Class)里面,目的就是测试代码与被测页面对象代码分离,后期如果有页面元素发生了更改,只需要修改相应Page类里面的获取属性的代码,测试层代码无需修改。
2、场景
使用selenium实现自动打开XXXX 首页,然后输入手机号、密码,点击登录后退出
浏览器:chrome 操作系统: OSX 10 开发工具:IntelliJ IDEA 测试框架:TestNG

3、以下以自己的一个例子作为讲下,如下截图是代码结构



*report:存放运行后的测试报告
*core:公共基础类
*pages:存放页面对象
*tests:运行测试用例
*utils:打印日志信息的一些配置格式
*resources:log4j配置以及testng 运行配置
步骤一、首先在pom.xml 里面需要引入jar包,以下实例用到selenium-java、TestNg、Log4j

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
<!-- test -->
    <groupId>webautotest</groupId>
    <artifactId>webautotest</artifactId>
    <version>1.0-SNAPSHOT</version>
    <packaging>jar</packaging>
    <url>http://maven.apache.org</url>
<properties>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
    </properties>
    <dependencies>
        <dependency>
            <groupId>org.seleniumhq.selenium</groupId>
            <artifactId>selenium-java</artifactId>
            <version>3.5.2</version>
        </dependency>
        <dependency>
            <groupId>org.testng</groupId>
            <artifactId>testng</artifactId>
            <version>6.9.6</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.seleniumhq.selenium</groupId>
            <artifactId>selenium-firefox-driver</artifactId>
            <version>3.5.2</version>
        </dependency>
        <dependency>
            <groupId>org.seleniumhq.selenium</groupId>
            <artifactId>selenium-chrome-driver</artifactId>
            <version>3.5.2</version>
        </dependency>
        <dependency>
            <groupId>log4j</groupId>
            <artifactId>log4j</artifactId>
            <version>1.2.17</version>
        </dependency>
    </dependencies>
    <build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-surefire-plugin</artifactId>
                <version>2.17</version>
                <configuration>
                    <suiteXmlFiles>
                        <suiteXmlFile>src/test/java/pages/weiDianTest.xml</suiteXmlFile>
                    </suiteXmlFiles>
                </configuration>
            </plugin>
        </plugins>
    </build>
</project>

步骤二、编写公共模块,初始化准备

public class TestBase {
    {
      // System.setProperty("webdriver.firefox.marionette", "/Users/chenxiaoqin/Downloads/geckodriver");
         System.setProperty("webdriver.chrome.driver", "/Users/chenxiaoqin/Downloads/chromedriver");
    }
   // protected WebDriver driver = new FirefoxDriver();//打开火狐浏览器
   ChromeOptions options =new ChromeOptions();
    protected WebDriver driver = new ChromeDriver(options);
  //获取当前类的类名传值给logger,该句的作用就是用log4j打印日志时知道是哪个类下面的打印输出信息
    public TestBase(){
        try {
            _newTest1();
        } catch (MalformedURLException e) {
            e.printStackTrace();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
    protected String getUrl() {
        return "https://www.XXX.com";
    }
    protected Logger logger =  LogManager.getLogger(getClass().getSimpleName());
    /**
     *
     * 程序入口,打开需要测试的url地址
     * */
    protected void _newTest1() throws MalformedURLException, InterruptedException {
        driver.get(getUrl());//打开需要测试页面的url
        Thread.sleep(2000);
        //chrome最大化
         options.addArguments("--start-maximized");
        //driver.manage().window().maximize();//获取当前窗口最大化,这个方法是不支持IE跟谷歌浏览器
        Thread.sleep(1000);
    }
    @AfterMethod
    protected void tearDown(){
        driver.quit();
    }
}

步骤三、使用PageFactory初始化pageObject对象,它存在于org.openqa.selenium.support库里面,它提供的方法都是静态的,可以直接调用,提供以下4种方法:

initElements(WebDriver driver, Class<T> pageClassToProxy)
initElements(WebDriver driver, Object page)
initElements(ElementLocatorFactory factory, Object page)
initElements(FieldDecorator decorator, Object page)

一般在实际应用中,我们可以这样使用:

PageFactory.initElements(dr, XXX.class);

或者这样使用:

PageFactory.initElements(new AjaxElementLocatorFactory(dr, 10) ,XXX.class);
后者加入了初始化元素时等待时间

通过initElements方法初始化的各个页面对象,AjaxElementLocatorFactory方法可以查找元素时都会在指定的TIMEOUT时间内不断重试,如果在指定时间内定位到元素则马上继续,如果指定时间内未找到则抛出NoSuchElementException异常。具体事例如下:

package pages;
import core.TestBase;
import org.apache.log4j.LogManager;
import org.apache.log4j.Logger;
import org.openqa.selenium.By;
import org.openqa.selenium.WebDriver;
import org.openqa.selenium.chrome.ChromeDriver;
import org.openqa.selenium.firefox.FirefoxDriver;
import org.openqa.selenium.support.PageFactory;
import org.openqa.selenium.support.pagefactory.AjaxElementLocatorFactory;
import org.openqa.selenium.support.pagefactory.ElementLocatorFactory;

/**
 * Created by jean.
 */
public class GeneralPage {
    protected  WebDriver driver;
    public GeneralPage (WebDriver driver) {
        this.driver = driver;
        //通过initElements方法初始化的各个页面对象,AjaxElementLocatorFactory方法可以查找元素时都会在指定的TIMEOUT时间内不断重试,如果在指定时间内定位到元素则马上继续,如果指定时间内未找到则抛出NoSuchElementException异常。
        PageFactory.initElements(new AjaxElementLocatorFactory(driver, 3000), this);
    }
    /**
     * 封装sendKey文本框输入方法
     * 封装click点击事件方法
     *
     * */
    protected void sendKeys(By by, String value) {
        driver.findElement(by).sendKeys(value);
    }
    protected void click(By by) {
        driver.findElement(by).click();
    }
}

步骤四、创建LogonPage,使用@FindBy来查找页面元素,支持的类型有:id
、name、className、css、tagName、linkText、partialLinkText、xpath
package pages;

import core.TestBase;
import org.openqa.selenium.By;
import org.openqa.selenium.WebDriver;
import org.openqa.selenium.WebElement;
import org.openqa.selenium.chrome.ChromeDriver;
import org.openqa.selenium.support.FindBy;
import org.testng.annotations.Test;
import java.net.MalformedURLException;
/**
 * @author jean
 *         测试场景:打开页面输入手机号、密码点击登录
 */
public class LogonPage extends GeneralPage{
  @FindBy(id="nickName0")
    private WebElement nickName;
    @FindBy(id="logPsw")
    private WebElement passWord;
    @FindBy(css="button.ant-btn.ant-btn-primary.login-btn")
    private WebElement loginButton;
public LogonPage(WebDriver driver) {
        super(driver);
    }
    public LogonPage nameInput() throws InterruptedException {             
     nickName.sendKeys("XXXXXXXXXXX");
        Thread.sleep(3000);
        return this;
    }
    public LogonPage passWordInput() throws InterruptedException {
        passWord.sendKeys("XXXXXXXXX");
        Thread.sleep(3000);
        return this;
    }
   public LogonPage buttonClick() throws InterruptedException {
        loginButton.click();
        Thread.sleep(3000);
        return this;
    }
}

步骤五、创建执行Test类

package tests;
import core.TestBase;
import org.testng.annotations.Test;
import pages.LogonPage;
/**
 * Created by chenxiaoqin on 9/10/17.
 */
public class LoginTest extends TestBase {
    @Test(priority = 0)
    public void login1() {

        try {
            LogonPage logonPage = new LogonPage(driver)
                    .nameInput()
                    .passWordInput()
                    .buttonClick();
        } catch (Exception e) {
            e.printStackTrace();
        }
       logger.info("测试成功");
    }
}

对于日志的打印输出、数据的存储、多个用例的顺序执行,后续将持续更新...

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,711评论 5 468
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,932评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,770评论 0 330
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,799评论 1 271
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,697评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,069评论 1 276
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,535评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,200评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,353评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,290评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,331评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,020评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,610评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,694评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,927评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,330评论 2 346
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,904评论 2 341

推荐阅读更多精彩内容