在我们学习Page Object Model之前,我们先了解一下Page Object Model(以下简称POM).
为什么要POM
用UiAutomator启动UI自动化测试不是一件困难的任务。你只需要查找元素,对元素操作就可以了。来看一个登录APP的简单的脚本。
1 package com.gqou.testdemo; 2 3 import android.support.test.InstrumentationRegistry; 4 import android.support.test.runner.AndroidJUnit4; 5 import android.support.test.uiautomator.UiDevice; 6 import android.support.test.uiautomator.UiObject; 7 import android.support.test.uiautomator.UiSelector; 8 9 import org.junit.Test; 10 import org.junit.runner.RunWith; 11 12 import static junit.framework.TestCase.assertEquals; 13 14 /** 15 * Created by guangqian on 2016/4/22. 16 */ 17 @RunWith(AndroidJUnit4.class) 18 public class SimpleTest { 19 20 @Test 21 public void 测试登录成功() throws Exception{ 22 // 查找图标并点击启动app 23 UiDevice uiDevice = UiDevice.getInstance(InstrumentationRegistry.getInstrumentation()); 24 uiDevice.findObject(new UiSelector().text("TestDemo")).clickAndWaitForNewWindow(); 25 // 输入用户名,密码并点击Login 26 UiObject usernameObj = uiDevice.findObject(new UiSelector().resourceId("com.gqou.testdemo:id/name_txt")); 27 UiObject pwdObj = uiDevice.findObject(new UiSelector().resourceId("com.gqou.testdemo:id/pwd_txt")); 28 UiObject loginBtn = uiDevice.findObject(new UiSelector().resourceId("com.gqou.testdemo:id/login_btn")); 29 UiObject welcomeTip = uiDevice.findObject(new UiSelector().resourceId("com.gqou.testdemo:id/welcome_tip")); 30 31 usernameObj.click(); 32 usernameObj.setText("username"); 33 pwdObj.click(); 34 pwdObj.setText("password"); 35 36 loginBtn.clickAndWaitForNewWindow(); 37 38 // 校验登录成功信息 39 assertEquals("congratulations", welcomeTip.getText()); 40 } 41 }
你可以看到,我们所做的就是查找元素并为那些元素填值,点击等操作。
这是一个比较简单的脚本,脚本维护看起来也很容易。但是随着时间增长,你会编写更多的脚本代码,维护也会变得越来越困难。
脚本维护的主要面临的问题就是:比如有10份脚本使用同一个界面元素,一旦界面元素发生了变化,你需要更改10份脚本。这不仅浪费时间而且还很容易出错。
本文介绍的脚本维护比较好的方法就是针对元素的查找,操作等操作来创建一个独立的类文件。这个类文件可以被使用那些元素的所有脚本重用。将来如果界面元素发生变化,只需要修改一份类文件就可以了,不需要再去修改10份脚本文件。
这个方法就叫做Page Object Model。它可以让我们的代码更加易于阅读,维护和重用。
什么是POM
1. Page Object Model是一种设计模式,为界面对象创建一个对象库
2. 在这个模式下,应用的每一个界面都应该有一个相应的界面类
3. 界面类会查找界面的元素也包含对元素操作的方法
4. 方法的命名应该根据它的行为来命名,例如:如果程序等待支付网关出现在界面上,方法应该被命名类似waitForPaymentScreenDisplay().
POM的优点
1. POM意味着对界面上的元素的操作从原来的测试用例代码中分离出来,这样使我们的代码清晰并且易于理解。
2. 第二个好处是对象库是独立的。因此我们可以通过不同的工具使用相同的对象库来达到不同的目的。比如:我们可以集成TestNG/JUnit来做功能测试,同时也可以使用JBehave/Cucumber做验收测试。
3. 由于方法的可重用,我们的代码会变得更少,有效性更高。
4. 方法能真实的映射出用户在界面上的操作行为,比如:点击过按钮之后并跳到主界面,方法名就会类似"gotoHomePage".
怎么实现POM
POM的基本结构是应用的所有元素和元素的操作方法都被封装在一个类文件中,测试用例代码只需要调用这个类文件中的方法即可。
完整的例子:
第一步:打开应用app
第二步:输入用户名,密码,点击Login
第三步: 验证登录成功
有两个界面:登录界面,欢迎界面,所以我们要创建两个界面的类文件LoginActivity,WelcomeActivity
LoginActivity.java
1 package com.gqou.testdemo.pages; 2 3 import android.support.test.uiautomator.UiDevice; 4 import android.support.test.uiautomator.UiObject; 5 import android.support.test.uiautomator.UiSelector; 6 7 /** 8 * Created by guangqian on 2016/4/22. 9 */ 10 public class LoginActivity { 11 12 UiDevice uiDevice; 13 UiObject mUserName; 14 UiObject mPasswd; 15 UiObject mLogin; 16 17 public LoginActivity(UiDevice uiDevice){ 18 this.uiDevice = uiDevice; 19 mUserName = this.uiDevice.findObject(new UiSelector().resourceId("com.gqou.testdemo:id/name_txt")); 20 mPasswd = this.uiDevice.findObject(new UiSelector().resourceId("com.gqou.testdemo:id/pwd_txt")); 21 mLogin = this.uiDevice.findObject(new UiSelector().resourceId("com.gqou.testdemo:id/login_btn")); 22 } 23 24 /** 25 * 输入用户名 26 * @param userName 27 * @throws Exception 28 */ 29 private void setmUserName(String userName) throws Exception{ 30 mUserName.click(); 31 mUserName.setText(userName); 32 } 33 34 /** 35 * 输入密码 36 * @param pwd 37 * @throws Exception 38 */ 39 private void setmPasswd(String pwd) throws Exception{ 40 mPasswd.click(); 41 mPasswd.setText(pwd); 42 } 43 44 /** 45 * 点击登录按钮 46 * @throws Exception 47 */ 48 private void clickLogin()throws Exception{ 49 mLogin.clickAndWaitForNewWindow(); 50 } 51 52 /** 53 * 封装登录App操作 54 * @param username 55 * @param pwd 56 * @throws Exception 57 */ 58 public void loginApp(String username, String pwd) throws Exception{ 59 this.setmUserName(username); 60 this.setmPasswd(pwd); 61 this.clickLogin(); 62 } 63 }
WelcomeActivity.java
1 package com.gqou.testdemo.pages; 2 3 import android.support.test.uiautomator.UiDevice; 4 import android.support.test.uiautomator.UiObject; 5 import android.support.test.uiautomator.UiSelector; 6 7 /** 8 * Created by guangqian on 2016/4/22. 9 */ 10 public class WelcomeActivity { 11 UiDevice uiDevice; 12 UiObject welcomeTip; 13 14 public WelcomeActivity(UiDevice uiDevice){ 15 this.uiDevice = uiDevice; 16 welcomeTip = this.uiDevice.findObject(new UiSelector().resourceId("com.gqou.testdemo:id/welcome_tip")); 17 } 18 19 /** 20 * 获取欢迎提示语内容 21 * @return 22 * @throws Exception 23 */ 24 public String getWelcomeTip() throws Exception{ 25 return welcomeTip.getText(); 26 } 27 28 }
POMForLoginTest.java
1 package com.gqou.testdemo.testcases; 2 3 import android.support.test.InstrumentationRegistry; 4 import android.support.test.runner.AndroidJUnit4; 5 import android.support.test.uiautomator.UiDevice; 6 import android.support.test.uiautomator.UiSelector; 7 8 import com.gqou.testdemo.pages.LoginActivity; 9 import com.gqou.testdemo.pages.WelcomeActivity; 10 11 import org.junit.Before; 12 import org.junit.Test; 13 import org.junit.runner.RunWith; 14 15 import static junit.framework.TestCase.assertEquals; 16 /** 17 * Created by guangqian on 2016/4/22. 18 */ 19 @RunWith(AndroidJUnit4.class) 20 public class POMForLoginTest { 21 22 public static UiDevice uiDevice = UiDevice.getInstance(InstrumentationRegistry.getInstrumentation()); 23 private static LoginActivity objLogin; 24 private static WelcomeActivity objWelcome; 25 26 27 @Before 28 public void setUp() throws Exception { 29 // 打开应用 30 uiDevice.findObject(new UiSelector().text("TestDemo")).clickAndWaitForNewWindow(); 31 // 创建界面实例对象 32 objLogin = new LoginActivity(uiDevice); 33 objWelcome = new WelcomeActivity(uiDevice); 34 35 } 36 37 38 39 @Test 40 public void 测试App登录成功() throws Exception { 41 // 登录app 42 objLogin.loginApp("Tom", "123456"); 43 // 获取主界面提示语 44 String tips = objWelcome.getWelcomeTip(); 45 // 判断提示语是否为预期 46 assertEquals("congratulations", tips); 47 } 48 }
Good Luck!
水平有限,欢迎各位大牛点评,多谢支持!