手把手教你java+selenium数据驱动测试框架搭建与实践

news2025/1/23 1:02:24

最近在看Java+selenium+TestNg+Excel的数据驱动,如何使用TestNg和Excel进行数据驱动测试。我其实是个自动化测试小白,工作之余看看这方面的书,照着敲敲代码,慢慢理解,希望通过自己坚持不懈的努力,在测试这个职位上越来越优秀。

#前言

本文主要实现的功能为登录126邮箱,登录成功后,在通讯录中新建联系人,登录邮箱的账号、密码以及新建的联系人的信息都写在Excel文件中,调用Excel文件中的数据,测试用例执行之后并将测试结果写入Excel文件的最后一列。

#环境准备

(1)新建一个Java工程,命名为“Test126mailDataDriver”;

(2)在该工程中配置好WebDriver和TestNG环境,并导入Excel操作及Log4j相关的JAR文件到工程中;

(3)在工程中新建4个Package:

--cn.wangxy.appModules:主要用于实现复用的业务逻辑封装方法;

--cn.wangxy.pageObjects:主要用于实现被测试对象的页面对象;

--cn.wangxy.testScripts:主要用于实现具体的测试逻辑脚本;

--cn.wangxy.util:主要用于实现测试过程中调用的工具类方法,如文件操作、页面元素的操作方法等。

(4)创建一个Excel文件,命名为“126邮箱的数据驱动测试数据.xlsx”,文件的内容如下:其中登录名和密码是我真实的数据,我在这就隐藏掉了哦!

#代码

一、元素定位表达式

在项目根目录下新建一个objectMap.properties文件,编写页面元素的定位表达式。

解释:以下代码分别表示的页面元素为126邮箱的登录页面部分元素和给通讯录新建联系人时用到的元素,其中定位类型和定位表达式用">"分开。

  1. 126mail.loginPage.iframe=xpath>//iframe[contains(@id,'x-URS-iframe')]

  2. 126mail.loginPage.username=xpath>//input[@data-placeholder='邮箱帐号或手机号码']

  3. 126mail.loginPage.password=xpath>//*[@data-placeholder='输入密码']

  4. 126mail.loginPage.loginbutton=id>dologin

  5. 126mail.loginPage.addressbook=xpath>//div[contains(text(),'通讯录')]

  6. 126mail.addressbook.createContactPerson=xpath>//div/div/*[contains(@id,'_mail_button_')]/span[contains(.,'新建联系人')]

  7. 126mail.addressbook.createContactPersonName=xpath>//dt[contains(.,'姓名')]/following-sibling::dd/div/input

  8. 126mail.addressbook.createContactPersonEmail=xpath>//dt[contains(.,'电子邮箱')]/following-sibling::dd/div/input

  9. 126mail.addressbook.createContactPersonMobile=xpath>//dt[contains(.,'手机号码')]/following-sibling::dd/div/input

  10. 126mail.addressBook.saveButton=xpath>//*[contains(@id,'_mail_button_')]/span[contains(.,'确 定')]

二、读取配置文件

在cn.wangxy.util包下新建一个类ObjectMap,用于实现在外部配置文件中配置页面元素的定位表达式。

2.1 读取配置文件代码编写

  1. Properties properties;

  2. public ObjectMap(String propFile){

  3. properties = new Properties();

  4. try {

  5. Reader in = new InputStreamReader(new FileInputStream(propFile),"UTF-8");

  6. properties.load(in);

  7. in.close();

  8. } catch (IOException e) {

  9. System.out.println("读取对象文件出错");

  10. e.printStackTrace();

  11. }

  12. }

2.2 根据配置文件中传入的对象,判断需要使用哪种定位表达式

  1. public By getLocator(String ElementNameInporFile) throws Exception{

  2. //根据传入的变量,从属性配置文件中读取对应的配置对象

  3. String locator = properties.getProperty(ElementNameInporFile);

  4. //将配置对象中的定位类型存入locatorType变量,将定位表达式的值存入locatorValue变量

  5. String locatorType = locator.split(">")[0];

  6. String locatorValue = locator.split(">")[1];

  7. //输出locatorType和locatorValue的值,验证赋值是否正确

  8. System.out.println("获取的定位类型"+locatorType+"\t获取的定位表达式"+locatorValue);

  9. //根据locatorType的变量值内容判断返回何种定位方式的By对象

  10. if (locatorType.toLowerCase().equals("id")){

  11. return By.id(locatorValue);

  12. } else if(locatorType.toLowerCase().equals("name")){

  13. return By.name(locatorValue);

  14. }else if(locatorType.toLowerCase().equals("class")||locatorType.toLowerCase().equals("classname")){

  15. return By.className(locatorValue);

  16. }else if(locatorType.toLowerCase().equals("tagname")||locatorType.toLowerCase().equals("tag")){

  17. return By.tagName(locatorValue);

  18. }else if(locatorType.toLowerCase().equals("linktext")||locatorType.toLowerCase().equals("link")){

  19. return By.linkText(locatorValue);

  20. }else if(locatorType.toLowerCase().equals("partialLinkText")){

  21. return By.partialLinkText(locatorValue);

  22. }else if(locatorType.toLowerCase().equals("cssSelector")||locatorType.toLowerCase().equals("css")){

  23. return By.cssSelector(locatorValue);

  24. }else if(locatorType.toLowerCase().equals("xpath")){

  25. return By.xpath(locatorValue);

  26. }

  27. else {

  28. throw new Exception("输入的locator Type未在程序中被定义:"+locatorType);

  29. }

  30. }

三、获取登录页面的元素对象

在cn.wangxy.pageObjects包下新建一个LoginPage类,用于实现126邮箱登录页面的PageObject对象。

  1. public class LoginPage {

  2. private WebElement element = null;

  3. //指定页面元素定位表达式配置文件的绝对路径

  4. private ObjectMap objectMap = new ObjectMap("D:\\wangxiaoyu\\workspace\\Test126mailDataDriver\\objectMap.properties");

  5. private WebDriver driver;

  6. public LoginPage(WebDriver driver){

  7. this.driver = driver;

  8. }

  9. //进入iframe

  10. public void switchToFrame() throws Exception{

  11. Thread.sleep(5000);

  12. driver.switchTo().frame(driver.findElement(objectMap.getLocator("126mail.loginPage.iframe")));

  13. }

  14. //退出iframe

  15. public void defaultToFrame(){

  16. driver.switchTo().defaultContent();

  17. }

  18. //返回登录页面中的用户名输入框页面元素对象

  19. public WebElement userName() throws Exception{

  20. element = driver.findElement(objectMap.getLocator("126mail.loginPage.username"));

  21. return element;

  22. }

  23. //返回登录页面中密码输入框页面元素对象

  24. public WebElement password() throws Exception{

  25. element = driver.findElement(objectMap.getLocator("126mail.loginPage.password"));

  26. return element;

  27. }

  28. //返回登录页面中的登录按钮页面元素对象

  29. public WebElement loginButton() throws Exception{

  30. element = driver.findElement(objectMap.getLocator("126mail.loginPage.loginbutton"));

  31. return element;

  32. }

  33. }

四、封装登录的测试逻辑

在cn.wangxy.appModules包下新建一个Login_Action类,用来封装登录的测试逻辑,这样的话,在任何一个测试用例中使用登录操作时,直接调用就行,方便后期的代码维护。

  1. public class Login_Action {

  2. public static void execute(WebDriver driver,String userName,String passWord) throws Exception{

  3. driver.get("https://www.126.com/");

  4. LoginPage loginPage = new LoginPage(driver);

  5. loginPage.switchToFrame();

  6. loginPage.userName().sendKeys(userName);

  7. loginPage.password().sendKeys(passWord);

  8. loginPage.loginButton().click();

  9. Thread.sleep(5000);

  10. loginPage.defaultToFrame();

  11. Thread.sleep(3000);

  12. }

  13. }

五、小试牛刀-登录测试

在cn.wangxy.testScripts包下新建一个TestNg的测试类TestMail126Login。登录测试的脚本编写如下所示,邮箱的登录账号和密码用自己真实的数据,下面代码中我自己的隐藏了。

  1. public class TestMail126Login {

  2. public WebDriver driver;

  3. String baseUrl = "https://www.126.com/";

  4. @Test

  5. public void testMailLogin() throws Exception {

  6. Login_Action.execute(driver, "邮箱用户名", "邮箱密码");

  7. Assert.assertTrue(driver.getPageSource().contains("未读邮件"));

  8. }

  9. @BeforeMethod

  10. public void beforeMethod() {

  11. //加载Firefox驱动程序

  12. System.setProperty("webdriver.gecko.driver","D:\\geckodriver.exe");

  13. //打开Firefox浏览器

  14. driver=new FirefoxDriver();

  15. driver.manage().timeouts().implicitlyWait(10, TimeUnit.SECONDS);

  16. }

  17. @AfterMethod

  18. public void afterMethod() {

  19. driver.quit();

  20. }

  21. }

现在代码写的差不多了,就让我们怀着激动的心情在这个测试类上点击【右键】→【Run As】→【TestNg Test】,将现在的代码跑起来吧!只有一个超级简单的登录操作,运行正常且通过的话,我们开始进行下一步的操作!

上面的代码是我边敲,同时边写这篇文章的,代码都是我运行成功过的,从实际项目中复制过来的,所以正常情况下的运行是没有问题的。

六、获取登录后主页的页面元素对象

在cn.wangxy.pageObjects包下新建一个HomePage的类,用来获取登录后主页的页面元素对象,因为我们后面要进行的是在通讯录里面添加联系人,所以登录后需要点击主页中的通讯录按钮进入添加联系人的操作页面。以下代码只获取通讯录按钮的元素对象,如果需要在该页面操作更多的链接元素,可以根据需要进行自定义。

  1. public class HomePage {

  2. private WebElement element = null;

  3. private ObjectMap objectMap = new ObjectMap("D:\\workspace\\Test126mailDataDriven\\objectMap.properties");

  4. private WebDriver driver;

  5. public HomePage(WebDriver driver){

  6. this.driver = driver;

  7. }

  8. //获取登录后主页中的“通讯录”链接

  9. public WebElement addressLink() throws Exception{

  10. element = driver.findElement(objectMap.getLocator("126mail.loginPage.addressbook"));

  11. return element;

  12. }

  13. }

七、获取添加联系人的页面元素对象

在cn.wangxy.pageObjects包下新建一个AddressBookPage类,用来获取添加联系人的页面元素对象。

  1. public class AddressBookPage {

  2. private WebElement element = null;

  3. private ObjectMap objectMap = new ObjectMap("D:\\workspace\\Test126mailDataDriven\\objectMap.properties");

  4. private WebDriver driver;

  5. public AddressBookPage(WebDriver driver){

  6. this.driver =driver;

  7. }

  8. //获取新增联系人按钮

  9. public WebElement createContactPersonButton() throws Exception{

  10. element = driver.findElement(objectMap.getLocator("126mail.addressbook.createContactPerson"));

  11. return element;

  12. }

  13. //获取新增联系人界面中的姓名输入框

  14. public WebElement contactPersonName() throws Exception{

  15. element =driver.findElement(objectMap.getLocator("126mail.addressbook.createContactPersonName"));

  16. return element;

  17. }

  18. //获取新增联系人界面中的电子邮件输入框

  19. public WebElement contactPersonEmail() throws Exception{

  20. element =driver.findElement(objectMap.getLocator("126mail.addressbook.createContactPersonEmail"));

  21. return element;

  22. }

  23. //获取新增联系人界面中的手机号输入框

  24. public WebElement contactPersonMobil()throws Exception{

  25. element =driver.findElement(objectMap.getLocator("126mail.addressbook.createContactPersonMobile"));

  26. return element;

  27. }

  28. //获取新增联系人界面中保存信息的“确定”按钮

  29. public WebElement saveButton() throws Exception{

  30. element =driver.findElement(objectMap.getLocator("126mail.addressBook.saveButton"));

  31. return element;

  32. }

  33. }

八、封装新建联系人的操作逻辑

在cn.wangxy.appModules包下新建一个AddContactPerson_Action类,用来封装新建联系人的操作逻辑。

  1. public class AddContactPerson_Action {

  2. public static void execute(WebDriver driver,String userName,String password,String contactName,String contactEmail,String contactMobil)throws Exception{

  3. Login_Action.execute(driver, userName, password);

  4. Thread.sleep(3000);

  5. Assert.assertTrue(driver.getPageSource().contains("未读邮件"));

  6. HomePage homePage = new HomePage(driver);

  7. homePage.addressLink().click();

  8. AddressBookPage addressBookPage = new AddressBookPage(driver);

  9. Thread.sleep(3000);

  10. addressBookPage.createContactPersonButton().click();

  11. Thread.sleep(1000);

  12. addressBookPage.contactPersonName().sendKeys(contactName);

  13. Thread.sleep(3000);

  14. addressBookPage.contactPersonEmail().sendKeys(contactEmail);

  15. Thread.sleep(3000);

  16. addressBookPage.contactPersonMobil().sendKeys(contactMobil);

  17. Thread.sleep(3000);

  18. addressBookPage.saveButton().click();

  19. Thread.sleep(5000);

  20. }

  21. }

九、Excel操作工具类

接下来重点来了,执行测试用例时要调用Excel文件中的数据并且将执行结果写入Excel数据文件中,需要封装对Excel文件的读取、写入等其它操作方法。

1、设定要操作的Excel文件路径和Excel文件中的Sheet名称

在读/写Excel文件时,均需先调用此方法,设定要操作的Excel文件路径和要操作的Sheet名称。

  1. public static void setExcelFile(String Path,String SheetName) throws Exception{

  2. FileInputStream ExcelFile;

  3. try {

  4. //实例化Excel文件的FileInputStream对象

  5. ExcelFile = new FileInputStream(Path);

  6. //实例化Excel文件的XSSFWorkbook对象

  7. ExcelWbBook = new XSSFWorkbook(ExcelFile);

  8. //实例化XSSFSheet对象,指定Excel文件中的Sheet名称,用于后续sheet中行、列和单元格的操作

  9. ExcelWSheet = ExcelWbBook.getSheet(SheetName);

  10. } catch (Exception e) {

  11. throw(e);

  12. }

  13. }

2、读取Excel文件指定单元格内容的方法

  1. public static String getCellData(int RowNum,int ColNum){

  2. try{

  3. //通过函数参数指定单元格的行号和列号,获取指定的单元格对象

  4. Cell = ExcelWSheet.getRow(RowNum).getCell(ColNum);

  5. String CellData = Cell.getCellType()==XSSFCell.CELL_TYPE_STRING?Cell.getStringCellValue()+""

  6. :String.valueOf(Math.round(Cell.getNumericCellValue()));

  7. return CellData;

  8. }catch(Exception e){

  9. return "";

  10. }

  11. }

下面一行代码主要用来判断单元格内容的类型。如果单元格的内容为字符串类型,则使用getStringCellValue方法获取单元格的内容,如果单元格的内容为数字类型,则使用getNumericCellValue方法获取单元格的内容,注意getNumericCellValue方法返回值是double类型,转换字符串类型必须在Cell.getNumericCellValue前面增加"",用于强制转换double类型为String类型,不加""的话,则会抛出double类型无法转换为String类型的异常。

  1. String CellData = Cell.getCellType()==XSSFCell.CELL_TYPE_STRING?Cell.getStringCellValue()+""

  2. :String.valueOf(Math.round(Cell.getNumericCellValue()));

3、在Excel文件的执行单元格中写入数据

以下代码只支持扩展名为“.xlsx”的Excel文件

  1. public static void setCellData(int RowNum,int ColNum,String Result) throws Exception{

  2. try{

  3. //获取Excel文件的行对象

  4. Row = ExcelWSheet.getRow(RowNum);

  5. //如果单元格为空,则返回null

  6. Cell = Row.getCell(ColNum,Row.RETURN_BLANK_AS_NULL);

  7. if (Cell==null) {

  8. //当单元格对象是null的时候,则创建单元格

  9. //如果单元格为空,无法直接调用单元格对象的setCellValue方法设定单元格的值

  10. Cell = Row.createCell(ColNum);

  11. //创建单元格后,可以调用单元格对象的setCellValue方法设定单元格的值

  12. Cell.setCellValue(Result);

  13. }else{

  14. //单元格中有内容,则可以直接调用单元格对象的setCellValue方法设定单元格的值

  15. Cell.setCellValue(Result);

  16. }

  17. //实例化写入Excel文件的文件输出流对象

  18. FileOutputStream fileOut = new FileOutputStream(Constant.TestDataExcelFilePath);

  19. //将内容写入excel文件

  20. ExcelWBook.write(fileOut);

  21. //调用flush方法强制刷新写入页面

  22. fileOut.flush();

  23. //关闭文件输出流对象

  24. fileOut.close();

  25. }catch (Exception e){

  26. throw (e);

  27. }

  28. }

4、从Excel文件获取测试数据的静态方法

用于为TestNG提供数据驱动的数据集数组

  1. public static Object[][] getTestData(String excelFilePath,String sheetName) throws IOException{

  2. //根据参数传入的数据文件路径和文件名称,组合出excel数据文件的绝对路径

  3. //声明一个File文件的对象

  4. File file = new File(excelFilePath);

  5. //创建FileInputStream对象用于读取Excel文件

  6. FileInputStream inputStream = new FileInputStream(file);

  7. //声明WorkBook对象

  8. Workbook Workbook =null;

  9. //获取文件名参数的扩展名,判断是“.xlsx”文件还是“.xls”文件

  10. String fileExtensionName = excelFilePath.substring(excelFilePath.indexOf("."));

  11. //文件类型如果是“.xlsx”,则使用XSSFWorkbook

  12. //文件类型如果是".xls",则使用

  13. if (fileExtensionName.equals(".xlsx")) {

  14. Workbook = new XSSFWorkbook(inputStream);

  15. } else if(fileExtensionName.equals(".xls")){

  16. Workbook = new HSSFWorkbook(inputStream);

  17. }

  18. //通过sheetName参数,生成Sheet对象

  19. Sheet Sheet = Workbook.getSheet(sheetName);

  20. /*获取excel数据文件Sheet1中数据的行数,用getLastRowNum方法获取数据的最后一行行号,用getFirstNum

  21. 方法获取数据的第一行行号,相减之后算出数据的行数

  22. 注意:excel文件的行号和列号都是从0开始的*/

  23. int rowCount = Sheet.getLastRowNum()-Sheet.getFirstRowNum();

  24. //创建名为records的List对象来存储从excel文件读取的数据

  25. ArrayList<Object[]> records = new ArrayList<Object[]>();

  26. //使用两个for循环遍历excel文件的所有数据(除了第一行,第一行是数据列名称),所以i从1开始,而不是从0开始

  27. for (int i = 1; i < rowCount+1; i++) {

  28. Row row = Sheet.getRow(i);

  29. /*

  30. * 声明一个数组,用来存储excel测试数据文件每行中的测试用例和数据,数组的大小用getLastCellNum-2来进行

  31. * 动态声明,实现测试数据个数和数组大小的一致,因为excel测试数据文件中测试数据行的最后一个单元格为测试执行结果,倒数第二个

  32. * 单元格为此测试数据行是否运行的状态位,所以最后两列的单元格数据并不需要传入测试方法,因此,使用getLastCellNum-2

  33. * 的方法去掉每行中最后两个单元格的数据,计算出需要存储的测试数据个数,并作为测试数据数组的初始化大小

  34. *

  35. */

  36. String fields[] = new String[row.getLastCellNum()-2];

  37. /*

  38. * if用于判断数据行是否要参与测试的执行,excel文件的倒数第二列为数据行的状态位,标记为y表示要执行,非y则不参与测试脚本的执行,会跳过

  39. */

  40. if (row.getCell(row.getLastCellNum()-2).getStringCellValue().equals("y")) {

  41. for (int j = 0; j < row.getLastCellNum()-2; j++) {

  42. //判断Excel文件的单元格字段是数字还是字符串,字符串格式调用getStringCellValue方法获取

  43. //数字格式调用getNumericCellValue方法获取

  44. fields[j] = (String)(row.getCell(j).getCellType()==XSSFCell.CELL_TYPE_STRING?

  45. row.getCell(j).getStringCellValue():""+row.getCell(j).getNumericCellValue());

  46. }

  47. //将fields的数据对象存储到records的List中

  48. records.add(fields);

  49. }

  50. }

  51. //定义函数返回值,即Object[][]

  52. //将存储测试数据的List转换为一个Object的二维数组

  53. Object[][] results = new Object[records.size()][];

  54. //设置二维数据每行的值,每行是一个Object对象

  55. for (int i = 0; i < records.size(); i++) {

  56. results[i] = records.get(i);

  57. }

  58. return results;

  59. }

5、返回Excel测试数据文件最后一列的列号

最后一列用来写入用例执行结果

  1. public static int getLastColumnNum(){

  2. //返回Excel测试数据文件最后一列的列号,如果有11列,则结果返回10

  3. return ExcelWSheet.getRow(0).getLastCellNum()-1;

  4. }

十、添加联系人测试类

在cn.wangxy.testScripts包下新建测试类TestMail126AddContactPerson,用来编写用户登录邮箱之后在通讯录中添加联系人,同时,测试的数据都来自于Excel文件,执行结果也写入Excel文件最后一列。

  1. public class TestMail126AddContactPerson {

  2. public WebDriver driver;

  3. //调用Constant类中的常量

  4. String baseUrl = Constant.Url;

  5. //定义dataprovider,并命名为“testData”

  6. @DataProvider(name = "testData")

  7. public static Object[][] data() throws IOException{

  8. /*

  9. * 调用ExcelUtil类中的getTestData静态方法,获取excel文件中倒数第二列标记为y的测试数据行,

  10. * 函数参数为常量TestDataExcelFilePath和TestDataExcelFileSheet,指定测试文件的路径和

  11. * sheet名称

  12. */

  13. return ExcelUtil.getTestData(Constant.TestDataExcelFilePath, Constant.TestDataExcelFileSheet);

  14. }

  15. //使用名称为“testData”的dataProvider作为测试方法的测试数据集

  16. //测试方法一共使用10和参数,对应exce文件第1~10列

  17. @Test(dataProvider = "testData")

  18. public void testAddressBook(String CaseRowNumber,String testCaseName,String maliUserName,

  19. String mailPassWord,String contactPersonName,String contactPersonEmail,

  20. String contactPersonMobile,String assertContactPersonName,

  21. String assertContactPersonEmail,String assertContactPersonMobile) throws Exception {

  22. driver.get(baseUrl);

  23. //使用变量maliUserName、mailPassWord,contactPersonName,contactPersonEmail,contactPersonMobile作为添加联系人动作的参数

  24. try {

  25. AddContactPerson_Action.execute(driver, maliUserName, mailPassWord, contactPersonName, contactPersonEmail, contactPersonMobile);

  26. } catch (AssertionError error) {

  27. System.out.println("添加联系人失败");

  28. /*

  29. * 执行AddContactPerson_Action类的execute方法失败时,catch语句会捕获AssertionError

  30. * 类型的异常,并设置其测试执行结果失败,由于excel文件中的序号格式被默认设定为带有一位小数,

  31. * 所以使用split("[.]")[0]语句获取序号的整数部分,并传给setCellData函数,在对应序号

  32. * 的测试数据行的最后一列设定为“测试执行失败”

  33. */

  34. ExcelUtil.setCellData(Integer.parseInt(CaseRowNumber.split(".")[0]), ExcelUtil.getLastColumnNum(), "测试执行失败");

  35. //调用Assert类的fail方法将此测试用例设定为执行失败,后续代码将不再执行

  36. Assert.fail("执行AddContactPerson_Action类的execute方法失败");

  37. }

  38. Thread.sleep(3000);

  39. try {

  40. Assert.assertTrue(driver.getPageSource().contains(assertContactPersonName));

  41. } catch (AssertionError error) {

  42. ExcelUtil.setCellData(Integer.parseInt(CaseRowNumber.split("[.]")[0]), ExcelUtil.getLastColumnNum(), "测试执行失败");

  43. Assert.fail("断言通讯录的页面是否包含联系人姓名的关键字失败");

  44. }

  45. try {

  46. Assert.assertTrue(driver.getPageSource().contains(assertContactPersonEmail));

  47. } catch (AssertionError error) {

  48. ExcelUtil.setCellData(Integer.parseInt(CaseRowNumber.split("[.]")[0]), ExcelUtil.getLastColumnNum(), "测试执行失败");

  49. Assert.fail("断言通讯录的页面是否包含联系人邮箱的关键字失败");

  50. }

  51. try {

  52. Assert.assertTrue(driver.getPageSource().contains(assertContactPersonMobile));

  53. } catch (AssertionError error) {

  54. ExcelUtil.setCellData(Integer.parseInt(CaseRowNumber.split("[.]")[0]), ExcelUtil.getLastColumnNum(), "测试执行失败");

  55. Assert.fail("断言通讯录的页面是否包含联系人手机号的关键字失败");

  56. }

  57. ExcelUtil.setCellData(Integer.parseInt(CaseRowNumber.split("[.]")[0]), ExcelUtil.getLastColumnNum(), "执行成功");

  58. }

  59. @BeforeMethod

  60. public void beforeMethod() {

  61. //加载Firefox驱动程序

  62. System.setProperty("webdriver.gecko.driver","D:\\wangxiaoyu\\geckodriver.exe");

  63. //打开Firefox浏览器

  64. driver=new FirefoxDriver();

  65. driver.manage().window().maximize();

  66. driver.manage().timeouts().implicitlyWait(10, TimeUnit.SECONDS);

  67. }

  68. @AfterMethod

  69. public void afterMethod() {

  70. driver.quit();

  71. }

  72. @BeforeClass

  73. public void BeforeClass() throws Exception{

  74. //使用Constant类中的常量,设定测试数据文件的文件路径和测试数据所在的sheet名称

  75. ExcelUtil.setExcelFile(Constant.TestDataExcelFilePath,Constant.TestDataExcelFileSheet);

  76. }

  77. }

十一、执行结果

打开Excel测试数据文件,我们可以看到序号为1和3的测试数据行的最后一列均显示“执行成功”,序号为2的测试数据行的最后一列依旧显示“/”,表示此数据行并未被测试方法调用。Excel测试数据文件的具体内容如图:

十二、数据驱动测试框架优点

(1)通过配置文件,实现页面元素定位表达式和测试代码的分离;

(2)使用ObjectMap方式,简化与页面元素定位相关的代码量;

(3)使用PageObject模式,封装了网页中的页面元素,方便测试代码调用,也实现了一处维护、全局生效的目标。

(4)在cn.wangxy.appModules的包下封装了常用的页面对象操作方法,简化了测试脚本编写的工作量。

(5)在Excel文件中定义多个测试数据,测试框架可自动调用测试数据完成数据驱动测试。

(6)在Excel文件的测试数据中,通过设定“测试数据是否执行”列的内容为“y”或者“n”,可自定义选择测试数据。测试执行结束后会在“测试结果”列中显示测试执行的结果,方便测试人员的查看。

十三、感悟

自学自动化测试其实真的挺累的,但是请不要放弃,坚持下去,一定会成功的,照着书本或者视频上一步一步的练习,我目前就是这样的,不过每一步都要扎实,要不然就会纯粹的浪费时间。我上面代码也是照着书上练习,后来都理解了,代码跑通之后,再写出来的,想着以后即使是参考的话也会有很大的用处。

行动吧,在路上总比一直观望的要好,未来的你肯定会感 谢现在拼搏的自己!如果想学习提升找不到资料,没人答疑解惑时,请及时加入扣群: 455787643,里面有各种软件测试+开发资料和技术可以一起交流学习哦。

总结:

感谢每一个认真阅读我文章的人!!!

作为一位过来人也是希望大家少走一些弯路,如果你不想再体验一次学习时找不到资料,没人解答问题,坚持几天便放弃的感受的话,在这里我给大家分享一些自动化测试的学习资源,希望能给你前进的路上带来帮助。

软件测试面试文档

我们学习必然是为了找到高薪的工作,下面这些面试题是来自阿里、腾讯、字节等一线互联网大厂最新的面试资料,并且有字节大佬给出了权威的解答,刷完这一套面试资料相信大家都能找到满意的工作。

 

          视频文档获取方式:
这份文档和视频资料,对于想从事【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴我走过了最艰难的路程,希望也能帮助到你!以上均可以分享,点下方小卡片即可自行领取。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2153195.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Python语言基础教程(下)4.0

✨博客主页&#xff1a; https://blog.csdn.net/m0_63815035?typeblog &#x1f497;《博客内容》&#xff1a;.NET、Java.测试开发、Python、Android、Go、Node、Android前端小程序等相关领域知识 &#x1f4e2;博客专栏&#xff1a; https://blog.csdn.net/m0_63815035/cat…

记录一下(goland导入其他包方法编译不了爆红但能正常使用)

在goLand里面新建go文件,里面放了一个方法,首字符也大写了,但是在别的包里面报错显示无法识别,爆红显示,但是项目能正常运行,这里说下我的解决方案 即可解决

软件设计师——操作系统

&#x1f4d4;个人主页&#x1f4da;&#xff1a;秋邱-CSDN博客☀️专属专栏✨&#xff1a;软考——软件设计师&#x1f3c5;往期回顾&#x1f3c6;&#xff1a;C: 类和对象&#xff08;上&#xff09;&#x1f31f;其他专栏&#x1f31f;&#xff1a;C语言_秋邱 一、操作系统…

粘接黑科技标杆专业展会-ASE CHINA 2024 震撼开幕!

2024年9月19日&#xff0c;第27届国际胶粘剂及密封剂展暨第19届国际胶粘带与薄膜展&#xff08;以下简称ASE CHINA 2024&#xff09;在上海新国际博览中心N3-N4-N5馆璀璨揭幕。ASE CHINA作为粘接新材料产业风向标&#xff0c;历经27年的辛苦耕耘&#xff0c;与业界同仁并肩而行…

发布策略说明

发布策略说明 发布策略 区别 标准发布 在部署新版本应用时删除旧版本应用。发布过程中&#xff0c;您的服务会出现短暂中断。 蓝绿发布 应用更新时生成蓝绿两个版本&#xff0c;两个版本互相热备&#xff0c;通过切换路由权重的方式实现不同版本应用的上下线。 该发布策略具…

Moshi: a speech-text foundation model for real time dialogue

视频号 挺神奇的东西 整下来 kyutai-labs/moshi (github.com) git clone https://github.com/kyutai-labs/moshi.git 在线体验 moshi.chat 结束后 点击Download audio Download video 可以下载音频与视频 &#xff08;不过是webm格式&#xff09; 发行版 已上传至资源 小…

Qt (19)【Qt 线程安全 | 互斥锁QMutex QMutexLocker | 条件变量 | 信号量】

阅读导航 引言一、互斥锁1. QMutex&#xff08;1&#xff09;基本概念&#xff08;2&#xff09;使用示例基本需求⭕thread.h⭕thread.cpp⭕widget.h⭕widget.cpp 2. QMutexLocker&#xff08;1&#xff09;基本概念&#xff08;2&#xff09;使用示例 3. QReadWriteLocker、QR…

AI论文写作PPT思维导图PC小程序开发

AI论文写作PPT思维导图PC小程序开发 AI智能PPT功能 一键生成PPT大纲、一键扩写大纲内容、单独扩写某个大纲内容、一键生成内容关键词、单项内容关键词生成、新增大纲项、修改大纲、删除大纲、选择PPT模板、单页模板一键切换、在线编辑模板&#xff1b;支持导出PPTX、JPEG、&am…

nodejs 013:Prect 样式复用(multiple classes)例子

Prect 简单示例 Prect 为使用相同的现代 API 的快速 3kB React 替代方案。代码形式与 React 基本相同。部分语法区别可见 prect-differences-to-react。以下是一个 Prect 简单示例。 Button目录Button.css&#xff1a; .this {display: inline-block;padding: 3px 8px;margi…

执行网络攻击模拟的 7 个步骤

在进攻和防守策略方面&#xff0c;我们可以从足球队和美式足球队身上学到很多东西。球员们会分析对方球队的策略&#xff0c;找出弱点&#xff0c;相应地调整进攻策略&#xff0c;最重要的是&#xff0c;练习、练习、再练习。作为最低要求&#xff0c;网络安全部门也应该这样做…

基于微信的设备故障报修管理系统设计与实现+ssm论文源码调试讲解

2相关技术 2.1微信小程序 小程序是一种新的开放能力&#xff0c;开发者可以快速地开发一个小程序。小程序可以在微信内被便捷地获取和传播&#xff0c;同时具有出色的使用体验。尤其拥抱微信生态圈&#xff0c;让微信小程序更加的如虎添翼&#xff0c;发展迅猛。 2.2 MYSQL数据…

一文彻底搞懂大模型 - OpenAI o1(最强推理模型)

最近这一两周看到不少互联网公司都已经开始秋招提前批面试了。 不同以往的是&#xff0c;当前职场环境已不再是那个双向奔赴时代了。求职者在变多&#xff0c;HC 在变少&#xff0c;岗位要求还更高了。 最近&#xff0c;我们又陆续整理了很多大厂的面试题&#xff0c;帮助一些…

举例说明:自然语言处理实战项目

自然语言处理&#xff08;Natural Language Processing, NLP&#xff09;是人工智能领域的一个重要分支&#xff0c;旨在使计算机能够理解、解释和生成人类语言。以下是一些NLP实战项目的示例&#xff1a; 1. 情感分析&#xff08;Sentiment Analysis&#xff09; 项目描述: …

用 HTML + JavaScript DIY 一个渐进式延迟法定退休年龄测算器

为减轻社会和个人因退休年龄变化带来的冲击&#xff0c;近日&#xff0c;全国人民代表大会常务委员会正式发布了关于实施渐进式延迟法定退休年龄的重要决定。 根据该决定&#xff0c;我国将同步启动对男、女职工法定退休年龄的延迟计划。这一调整将采取渐进式的方式进行&#…

09年408考研真题-数据结构

数据结构 10.【2009统考真题】为解决计算机主机与打印机之间速度不匹配的问题&#xff0c;通常设置一个打印数据缓冲区&#xff0c;主机将要输出的数据依次写入该缓冲区&#xff0c;而打印机则依次从该缓冲区中取出数据。该缓冲区的逻辑结构应该是(B&#xff09;。 A.栈 …

unix中如何查询和修改进程的资源限制

一、前言 一个进程在运行时&#xff0c;会用到各种资源&#xff0c;比如cpu的使用时间、内存空间、文件等等。那么&#xff0c;一个进程能够占用多少资源呢&#xff1f;cpu使用的时间有多长&#xff1f;进程空间有多大&#xff1f;能够创建多少个文件&#xff1f;这个就是本文…

数字IC设计\FPGA 职位经典笔试面试整理--基础篇1

注&#xff1a; 资料都是基于网上一些博客分享和自己学习整理而成的 1&#xff1a;什么是同步逻辑和异步逻辑&#xff1f; 同步逻辑是时钟之间有固定的因果关系。异步逻辑是各时钟之间没有固定的因果关系。 同步时序 逻辑电路的特点&#xff1a;各触发器的时钟端全部连接在一…

蚁群算法(ACO算法)求解实例---旅行商问题 (TSP)

目录 一、采用ACO求解 TSP二、 旅行商问题2.1 实际例子&#xff1a;求解 6 个城市的 TSP2.2 **求解该问题的代码&#xff0c;代码&#xff08;完整代码关注底部微信公众号获取&#xff09;**2.3 代码运行过程截屏2.4 代码运行结果截屏&#xff08;后续和其他算法进行对比&#…

Acwing数据结构:单链表

单链表 主要思想&#xff1a;使用数组实现链表(而不用结构体&#xff0c;结构体代码更长&#xff0c;后续图论也是基于数组实现&#xff09;&#xff0c;即静态链表。因为动态链表使用new申请空间需要较多的时间&#xff0c;而算法要求的是以较少的时间完成任务。 单链表&…

软件测试八股文

Part1 1、你的测试职业发展是什么&#xff1f; 测试经验越多&#xff0c;测试能力越高。所以我的职业发展是需要时间积累的&#xff0c;一步步向着高级测试工程师奔去。而且我也有初步的职业规划&#xff0c;前3年积累测试经验&#xff0c;按如何做好测试工程师的要点去要求自…