【一:实战开发testng的介绍】

本文主要介绍了接口测试的重要性,如便于持续集成、降低研发成本等,并详细讲解了testng自动化测试框架的使用,包括基本注解、依赖测试、超时测试、异常测试、多线程测试以及数据驱动等,帮助读者掌握接口自动化测试的落地实践。
摘要由CSDN通过智能技术生成


1、主要内容

在这里插入图片描述

1.1、为啥要做接口测试

  • 更容易实现持续集成
  • 自动化测试落地性价比更高,比UI更稳定
  • 大型系统更多更复杂,系统间模块越来越多
  • BUG更容易定位
  • 降低研发成本,提高效率

1.2、接口自动化测试落地过程

需求阶段项目立项,产品设计,需求文档
研发阶段UI设计、前端开发,后端开发,测试设计,测试开发
测试阶段环境搭建,多项测试执行,BUG修复,测试报告
项目上线显示回归测试,上线报告,添加监控

1.3、接口测试范围

https://i-blog.csdnimg.cn/blog_migrate/c2b1f2ace213bd5b9f0317df3bac00ce.jpeg)

1.4、手工接口常用的工具

  • postman(Chrome)
  • httpRequest
  • fiddler
  • jemter(统计结果不完善,倾向于性能测试)

1.5、自动化框架的设计

在这里插入图片描述

2、testng自动化测试框架

引入的包

  <dependency>
            <groupId>org.testng</groupId>
            <artifactId>testng</artifactId>
            <version>6.10</version>
 </dependency>
  1. @test:最基本的注解,用来把方法标记为测试的一部分
  2. @BeforeMethod: 测试方法之前运行的方法
  3. @AfterMethod:测试方法之后运行的方法
  4. @BeforeClass: 指的是在类运行之前运行的方法
  5. @AfterClass:指的是在类运行之后运行的方法
  6. @BeforeTest:多个测试类的集合,在设定好的测试集合前运行的方
  7. @AfterTest:多个测试类的集合,在设定好的测试集合后运行的方法
  8. @ BeforeSuite: 测试套件,在类运行之前运行的方法:
  9. @ AfterSuite: 测试套件,在类运行之后运行的方法
  10. @Test(dependsOnMethods = {“test1”}):依赖测试,在执行次方法之前执行test1方法之前
  11. 1@Test(expectedExceptions = RuntimeException.class):异常测试,方法里面要抛出异常throw new RuntimeException();,否则会是一个失败的异常测试
  12. @Test(enabled = true):忽略测试
  13. @Test(timeOut = 2000):超时测试
  14. @Test(invocationCount = 10,threadPoolSize = 3),多线程测试,一般用xml配置

总结:testng的annotations运行顺序为:
@BeforeSuite>@BeforeTest>@BeforeClass>@BeforeMethod>@test>@AfterMethod>@AfterClass>@AfterTest>@AfterSuite

基本测试

1、基本注解

import org.testng.annotations.*;
public class BasicAnnotation {

    //最基本的注解,用来把方法标记为测试的一部分
    @Test
    public void testCase1(){
        System.out.printf("Thread Id : %s%n",Thread.currentThread().getId());
        System.out.println("Test这是测试用例1");
    }

    @Test
    public void testCase2(){
        System.out.printf("Thread Id : %s%n",Thread.currentThread().getId());
        System.out.println("Test这是测试用例2");
    }

    @BeforeMethod
    public void beforeMethod(){
        System.out.println("BeforeMethod这是在测试方法之前运行的");
    }

    @AfterMethod
    public void afterMethod(){
        System.out.println("AfterMethod这是在测试方法之后运行的");
    }

    @BeforeClass
    public void beforeClass(){
        System.out.println("beforeClass这是在类运行之前运行的方法");
    }

    @AfterClass
    public void afterClass(){
        System.out.println("afterClass这是在类运行之后运行的方法");
    }

    @BeforeSuite
    public void beforeSuite(){
        System.out.println("BeforeSuite测试套件");
    }

    @AfterSuite
    public void afterSuite(){
        System.out.println("AfterSuite测试套件");
    }
}

在这里插入图片描述

2、忽略测试

public class IgnoreTest {
    @Test
    public void ignore1(){
        System.out.println("ignore1 执行!");
    }
    @Test(enabled = false)
    public void ignore2(){
        System.out.println("ignore2 执行");
    }
    @Test(enabled = true)
    public void ignore3(){
        System.out.println("ignore3 执行");
    }
}

在这里插入图片描述

3、依赖测试

依赖测试,被依赖测测试方法必须成功,不能失败,失败的话会不成功

public class DependTest {

    @Test
    public void test1(){
        System.out.println("test1 run");
        throw new RuntimeException();
    }

    @Test(dependsOnMethods = {"test1"}) //执行前依赖于test1()
    public void test2(){
        System.out.println("test2 run");
    }

}

在这里插入图片描述

4、超时测试

public class TimeOutTest {

    @Test(timeOut = 3000)//单位为毫秒值
    public void testSuccess() throws InterruptedException {
        Thread.sleep(2000);
    }

    @Test(timeOut = 2000)
    public void testFailed() throws InterruptedException {
        Thread.sleep(3000);
    }
}

在这里插入图片描述

5、异常测试

public class ExpectedException {

    /**
     * 什么时候会用到异常测试??
     * 在我们期望结果为某一个异常的时候
     * 比如:我们传入了某些不合法的参数,程序抛出了异常
     * 也就是说我的语气结果就是这个异常。
     */

    //这是一个测试结果会失败的异常测试
    @Test(expectedExceptions = RuntimeException.class)
    public void runTimeExceptionFailed(){
        System.out.println("这是一个失败的异常测试");
    }

    //这是一个成功的异常测试
    @Test(expectedExceptions = RuntimeException.class)
    public void runTimeExceptionSuccess(){
        System.out.println("这是我的异常测试");
        throw new RuntimeException();
    }

}

在这里插入图片描述

在这里插入图片描述

6、通过xml文件参数测试

public class ParamterTest {

    @Test
    @Parameters({"name","age"})
    public void paramTest1(String name,int age){
        System.out.println("name = " + name + ";  age = " + age);
    }
}

<?xml version="1.0" encoding="UTF-8" ?>

<suite name="parameter">
    <test name="param">
        <classes>
            <parameter name="name" value="zhangsan"/>
            <parameter name="age" value="10"/>
            <class name="com.course.testng.paramter.ParamterTest"/>
        </classes>
    </test>

</suite>

在这里插入图片描述
在这里插入图片描述

7、通过data实现数据驱动


public class DataProviderTest {
    @DataProvider(name="data")
    public Object[][] providerData(){
        Object[][] o = new Object[][]{
                {"zhangsan",10},
                {"lisi",20},
                {"wangwu",30}
        };
        return o;
    }

    @Test(dataProvider = "data")
    public void testDataProvider(String name,int age){
        System.out.println("name =" + name +"; age=" + age);
    }

}

在这里插入图片描述


public class DataProviderTest {

    @DataProvider(name="methodData")
    public Object[][] methodDataTest(Method method){
        Object[][] result=null;
        if(method.getName().equals("test1")){
            result = new Object[][]{
                    {"zhangsan",20},
                    {"lisi",25}
            };
        }else if(method.getName().equals("test2")){
            result = new Object[][]{
                    {"wangwu",50},
                    {"zhaoliu",60}
            };
        }

        return result;
    }


    @Test(dataProvider = "methodData")
    public void test1(String name,int age){
        System.out.println("test111方法 name="+name+";age="+age);
    }
    @Test(dataProvider = "methodData")
    public void test2(String name,int age){
        System.out.println("test222方法 name="+name+";age="+age);
    }


}

在这里插入图片描述

2.1、suit套件测试(按类分)

上述内容大家基本都了解,而在实际使用中我们再配合套件时,需要注意:
假设现在有如图所示项目:

  • LogInTest类:登录测试类;
public class LoginTest {
    @Test
    public void loginTaoBao(){
        System.out.println("淘宝登陆成功");
    }
}
  • ParTest类:支付测试类;
public class PayTest {
    @Test
    public  void paySuccess(){
        System.out.println("支付宝支付成功");
    }
}
  • SuiteConfig类:套件配置类;
public class SuiteConfig {
    @BeforeSuite
    public void beforeSuite(){
        System.out.println("before suite运行啦");
    }
    @AfterSuite
    public  void aftersuite(){
        System.out.println("after suite 运行啦");
    }
    @BeforeTest
    public void beforeTest(){
        System.out.println("beforeTest");
    }
    @AfterTest
    public void afterTest(){
        System.out.println("afterTest");
    }
}
  • suite.xml:套件xml文件
<?xml version="1.0" encoding="UTF-8" ?>
<suite name="test">
    <test name="login"> <!--名字自定义-->
        <classes>
            <class name="com.course.testng.suite.SuiteConfig"/>
            <class name="com.course.testng.suite.LoginTest"/>
        </classes>
    </test>
    <test name="pay"> <!--名字自定义-->
        <classes>
            <class name="com.course.testng.suite.SuiteConfig"/>
            <class name="com.course.testng.suite.PayTest"/>
        </classes>
    </test>
</suite>

在这里插入图片描述
在这里插入图片描述
套件的xml文件内容,xml的内容如上面代码所示
点击对应的xml,右键,实际运行结果如下:
在这里插入图片描述

2.2、分组测试(按方法分组,也可以分开写)

2.2.1、在class文件中分组


public class GroupsOnMethod {
    @Test(groups = "server")
    public void test1(){
        System.out.println("这是服务端组的测试方法1111");
    }

    @Test(groups = "server")
    public void test2(){
        System.out.println("这是服务端组的测试方法2222");
    }

    @Test(groups = "client")
    public void test3(){
        System.out.println("这是客户端组的测试方法3333");
    }
    @Test(groups = "client")
    public void test4(){
        System.out.println("这是客户端组的测试方法4444");
    }

    @BeforeGroups("server")
    public void beforeGroupsOnServer(){
        System.out.println("这是服务端组运行之前运行的方法");
    }

    @AfterGroups("server")
    public void afterGroupsOnServer(){
        System.out.println("这是服务端组运行之后运行的方法!!!!!");
    }

    @BeforeGroups("client")
    public void beforeGroupsOnClient(){
        System.out.println("这是客户端组运行之前运行的方法");
    }

    @AfterGroups("client")
    public void afterGroupsOnClient(){
        System.out.println("这是客户端组运行之后运行的方法!!!!!");
    }

}

在这里插入图片描述

2.2.2、使用xml文件分组

2.2.1、@Test(groups = “stu”)组
在这里插入图片描述
在这里插入图片描述

2.2.2、@Test(groups = “teacher”)组
在这里插入图片描述

2.2.3、分组的配置文件

<?xml version="1.0" encoding="UTF-8" ?>
<suite name="suitename">  名字自定义
    <test name="runAll">  名字自定义
        <classes>
            <class name="com.course.testng.groups.GroupsOnClass1"/>
            <class name="com.course.testng.groups.GroupsOnClass2"/>
            <class name="com.course.testng.groups.GroupsOnClass3"/>
        </classes>
    </test>

    <test name="onlyRunStu">
        <groups>
            <run>
                <include name="stu"/> 这个组名和测试类中groups命名的一致
            </run>
        </groups>

        <classes>
            <class name="com.course.testng.groups.GroupsOnClass1"/>
            <class name="com.course.testng.groups.GroupsOnClass2"/>
            <class name="com.course.testng.groups.GroupsOnClass3"/>
        </classes>

    </test>
</suite>

在这里插入图片描述
2.2.4、现在对应的xml,右键,查看运行结果
在这里插入图片描述

2.3、多线程运行(注解的方式配置)

2.3.1、通过注解的方式实现三个线程池运行里随机运行10次方法

public class MultiThreadOnAnnotion {

    @Test(invocationCount = 10,threadPoolSize = 3)
    public void test(){
        System.out.println(1);
        System.out.printf("Thread Id : %s%n",Thread.currentThread().getId());
    }

}

运行结果
在这里插入图片描述

2.4、多线程运行(在xml里面配置)

public class MultiThreadOnXml {

    @Test
    public void test1(){
        System.out.printf("Thread Id : %s%n",Thread.currentThread().getId());
    }

    @Test
    public void test2(){
        System.out.printf("Thread Id : %s%n",Thread.currentThread().getId());
    }

    @Test
    public void test3(){
        System.out.printf("Thread Id : %s%n",Thread.currentThread().getId());
    }

}
<?xml version="1.0" encoding="UTF-8" ?>
<suite name="thread" parallel="methods" thread-count="3">
    <!--
    tests级别:不同的test tag下的用例可以在不同的线程下执行
            相同的test tag下的用例只能在同一个线程中去执行
    classs级别:相同的class tag 下的用例在同一个线程中执行
                不同的class tag 下的用例可以在不同的线程中执行
    methods级别:所有用例都可以在不同的线程下去执行

    thread-count:代表了最大并发线程数

    xml文件配置这种方式不能指定线程池,只有方法上才可以指定线程池
    -->
    <test name = "demo1">
        <classes name="d">
            <class name="com.course.testng.multiThread.MultiThreadOnXml"/>
            <class name="com.course.testng.BasicAnnotation"/>
            <class name="com.course.testng.multiThread.MultiThreadOnXml"/>
        </classes>
        <classes name="d1">
            <class name="com.course.testng.multiThread.MultiThreadOnXml"/>
            <class name="com.course.testng.BasicAnnotation"/>
            <class name="com.course.testng.multiThread.MultiThreadOnXml"/>
        </classes>

    </test>

    <test name = "demo2">
        <classes name="d3">
            <class name="com.course.testng.BasicAnnotation"/>
        </classes>

    </test>



</suite>

xml文件配置这种方式不能指定线程池,只有方法上才可以指定线程池
运行结果;

在这里插入图片描述

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Leoon123

你的鼓励是我最大的动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值