HarmonyOS非UI单元测试在DevEco Studio上的应用

开发 前端 OpenHarmony
单元测试是测试某个类的某个方法能否正常工作的一种手段。单元测试的粒度:一般一个public方法需要一个test case。

[[415057]]

想了解更多内容,请访问:

51CTO和华为官方合作共建的鸿蒙技术社区

https://harmonyos.51cto.com

一、什么是单元测试

单元测试是测试某个类的某个方法能否正常工作的一种手段。

单元测试的粒度:一般一个public方法需要一个test case

二、单元测试目的

  • 验收(改动和重构)
  • 快速验证逻辑
  • 优化代码设计

三、单元测试工具

junit4 + mockito + powermock

junit4:JUnit是Java最基础的测试框架,主要的作用就是断言

Mock的作用:解决测试类对其他类的依赖问题。Mock的类所有方法都是空,所有变量都是初始值。

PowerMock:PowerMock是Mockito的扩展增强版,支持mock private、static、final方法和类,还增加了很多反射方法可以方便修改静态和非静态成员等。功能比Mockito增加很多。

  1. // build.gradle中引入powermock 
  2. testImplementation 'org.powermock:powermock-api-mockito2:2.0.2' 
  3. testImplementation 'org.powermock:powermock-module-junit4:2.0.2' 

四、单元测试流程

1、新建测试类(快捷导航键: ctrl+shift+T),新建测试用例名

2、setUp 初始化一些公共的东西

3、编写测试代码,执行操作

4、验证结果

一般我们依据被测方法是否有返回值选用不同的验证方法。

有返回值的,直接调用该方法得到返回结果,使用JUnit的Asset验证结果;

没有返回值的,则看方法最终调用了依赖对象的哪个方法,然后再校验依赖对象的该方法有没有被调用,以及获取到的参入参数是否正确

举例说明:

  1. public void login(String username, String password) { 
  2.      if (username == null || username.length() == 0) { 
  3.          return
  4.        } 
  5.      if (password == null || password.length() < 6) { 
  6.          return
  7.        } 
  8.      mUserManager.performLogin(username, password); 
  9.  } 

我们要验证该login方法是否正确,则依据传入的参数,判断mUserManager的performLogin方法是否得要了调用。

五、基础用法

常见注解:

  • @Before: 如果一个方法被@Before修饰过了,那么在每个测试方法调用之前,这个方法都会得到调用。
  • @After: 每个测试方法运行结束之后,会得到运行的方法
  • @Test:如果一个方法被@Before修饰过了,那么这个方法为可执行的测试用例,注解设置expected参数 可验证一个方法是否抛出了异常
  • @Ignore:忽略的测试方法
  • @RunWith 指定该测试类使用某个运行器
  • @Rule:重新制定测试类中方法的行为,可以理解为在测试用例执行前和执行后插桩
  • @Mock: 创建一个类的虚假的对象,在测试环境中,用来替换掉真实的对象,以达到两大目的:

a.验证这个对象的某些方法的调用情况,调用了多少次,参数是什么等等

b.指定这个对象的某些方法的行为,返回特定的值,或者是执行特定的动作

注意:mock出来的对象并不会自动替换掉正式代码里面的对象,你必须要有某种方式把mock对象应用到正式代码里面

junit框架中Assert类的常用方法

  • assertEquals: 断言传入的预期值与实际值是相等的
  • assertNotEquals: 断言传入的预期值与实际值是不相等的
  • assertArrayEquals: 断言传入的预期数组与实际数组是相等的
  • assertNull: 断言传入的对象是为空
  • assertTrue: 断言条件为真
  • assertFalse: 断言条件为假
  • assertSame: 断言两个对象引用同一个对象,相当于“==”

Mockito的使用

Mockito的使用主要分三步:Mock/spy对象 + 打桩 + 验证

示例:

  1. when(mockObj.methodName(params)).thenReturn(result) 
  • mock: 所有方法都是空方法,非void方法都将返回默认值,比如int方法返回0,对象方法将返回null,而void方法将什么都不做。 适用于类对外部依赖较多,只关新少数函数的具体实现;
  • spy:跟正常类对象一样,是正常对象的替身。适用场景跟mock相反,类对外依赖较少,关心大部分函数的具体实现。

四种Mock方式:

  • 普通方法:
  1. @Test 
  2. public void testIsNotNull(){ 
  3.     Person mPerson = mock(Person.class); //<--使用mock方法 
  4.  
  5.    assertNotNull(mPerson); 
  • 注解方法:
  1. public class MockitoAnnotationsTest { 
  2.  
  3.     @Mock //<--使用@Mock注解 
  4.     Person mPerson; 
  5.  
  6.     @Before 
  7.     public void setup(){ 
  8.         MockitoAnnotations.initMocks(this); //<--初始化 
  9.     } 
  10.  
  11.     @Test 
  12.     public void testIsNotNull(){ 
  13.         assertNotNull(mPerson); 
  14.     } 
  15.  
  • 运行器方法:
  1. @RunWith(MockitoJUnitRunner.class) //<--使用MockitoJUnitRunner 
  2. public class MockitoJUnitRunnerTest { 
  3.  
  4.     @Mock //<--使用@Mock注解 
  5.     Person mPerson; 
  6.  
  7.     @Test 
  8.     public void testIsNotNull(){ 
  9.         assertNotNull(mPerson); 
  10.     } 
  • MockitoRule方法:
  1. public class MockitoRuleTest { 
  2.  
  3.     @Mock //<--使用@Mock注解 
  4.     Person mPerson; 
  5.  
  6.     @Rule //<--使用@Rule 
  7.     public MockitoRule mockitoRule = MockitoJUnit.rule(); 
  8.  
  9.     @Test 
  10.     public void testIsNotNull(){ 
  11.         assertNotNull(mPerson); 
  12.     } 
  13.  

常用参数匹配

  • anyObject() 匹配任何对象
  • any(Class type) 与anyObject()一样
  • any() 与anyObject()一样 (慎用,有些场景会导致测试用例执行失败)
  • anyBoolean() 匹配任何boolean和非空Boolean
  • anyByte() 匹配任何byte和非空Byte
  • anyInt() 匹配任何int和非空Integer
  • anyString() 匹配任何非空String

常用打桩方法

  • thenReturn(T value) 设置要返回的值
  • thenThrow(Throwable… throwables) 设置要抛出的异常
  • thenAnswer(Answer answer) 对结果进行拦截
  • doReturn(Object toBeReturned) 提前设置要返回的值
  • doThrow(Throwable… toBeThrown) 提前设置要抛出的异常
  • doAnswer(Answer answer) 提前对结果进行拦截
  • doCallRealMethod() 调用某一个方法的真实实现
  • doNothing() 设置void方法什么也不做

PowerMock使用

首先使用PowerMock必须加注解@PrepareForTest和@RunWith(PowerMockRunner.class)。注解@PrepareForTest里写的是静态方法所在的类,如果@RunWith被占用。这时我们可以使用@Rule来解决

  1. @Rule 
  2. public PowerMockRule rule = new PowerMockRule(); 
  • mock静态方法
  1. @RunWith(PowerMockRunner.class) 
  2. public class PowerMockitoStaticMethodTest { 
  3.  
  4.     @Test 
  5.     @PrepareForTest({Banana.class}) 
  6.     public void testStaticMethod() {  
  7.         PowerMockito.mockStatic(Banana.class); //<-- mock静态类 
  8.         Mockito.when(Banana.getColor()).thenReturn("绿色"); 
  9.         Assert.assertEquals("绿色", Banana.getColor()); 
  10.  
  11.         //更改类的私有属性 
  12.         Whitebox.setInternalState(Banana.class, "COLOR""红色的"); 
  13.     } 
  • mock私有方法
  1. @RunWith(PowerMockRunner.class) 
  2. public class PowerMockitoPrivateMethodTest { 
  3.  
  4.     @Test 
  5.     @PrepareForTest({Banana.class}) 
  6.     public void testPrivateMethod() throws Exception { 
  7.         Banana mBanana = PowerMockito.mock(Banana.class); 
  8.         PowerMockito.when(mBanana.getBananaInfo()).thenCallRealMethod(); 
  9.         PowerMockito.when(mBanana, "flavor").thenReturn("苦苦的"); 
  10.         Assert.assertEquals("苦苦的黄色的", mBanana.getBananaInfo()); 
  11.         //验证flavor是否调用了一次 
  12.         PowerMockito.verifyPrivate(mBanana).invoke("flavor");  
  13.     } 
  • mock final方法,使用方式同 mock 私有方法
  • mock 构造方法
  1. @Test 
  2. @PrepareForTest({Banana.class}) 
  3. public void testNewClass() throws Exception { 
  4.     Banana mBanana = PowerMockito.mock(Banana.class); 
  5.     PowerMockito.when(mBanana.getBananaInfo()).thenReturn("大香蕉"); 
  6.     //如果new新对象,则返回这个上面设置的这个对象 
  7.     PowerMockito.whenNew(Banana.class).withNoArguments().thenReturn(mBanana); 
  8.     //new新的对象 
  9.     Banana newBanana = new Banana(); 
  10.     Assert.assertEquals("大香蕉", newBanana.getBananaInfo()); 

@Rule用法

自定义@Rule很简单,就是实现TestRule 接口,实现apply方法。

  1. public class MyRule implements TestRule { 
  2.  
  3.     @Override 
  4.     public Statement apply(final Statement base, final Description description) { 
  5.  
  6.         return new Statement() { 
  7.             @Override 
  8.             public void evaluate() throws Throwable { 
  9.                 // evaluate前执行方法相当于@Before 
  10.                 String methodName = description.getMethodName(); // 获取测试方法的名字 
  11.                 System.out.println(methodName + "测试开始!"); 
  12.  
  13.                 base.evaluate();  // 运行的测试方法 
  14.  
  15.                 // evaluate后执行方法相当于@After 
  16.                 System.out.println(methodName + "测试结束!"); 
  17.             } 
  18.         }; 
  19.     } 
  20.  

六、RxJava与单元测试

RxJava的火热程度不用多说,由于其基于事件流的链式调用、逻辑简洁 & 使用简单的特点,深受各大开发者的欢迎。我们经常用它来进行线程的切换操作

例如:

  1. public void threadSwitch() { 
  2.       Observable.just("one""two""three""four""five"
  3.               .subscribeOn(Schedulers.newThread()) 
  4.               .observeOn(OpenHarmonySchedulers.mainThread()) 
  5.               .subscribe(new Observer<String>() { 
  6.                   @Override 
  7.                   public void onSubscribe(@NonNull Disposable d) { 
  8.  
  9.                   } 
  10.  
  11.                   @Override 
  12.                   public void onNext(@NonNull String s) { 
  13.                       System.out.println(s); 
  14.                       if (callBack != null) { 
  15.                           callBack.success(s); 
  16.                       } 
  17.                   } 
  18.  
  19.                   @Override 
  20.                   public void onError(@NonNull Throwable e) { 
  21.                       if (callBack != null) { 
  22.                           callBack.failed(); 
  23.                       } 
  24.                   } 
  25.  
  26.                   @Override 
  27.                   public void onComplete() { 
  28.  
  29.                   } 
  30.               }); 
  31.   } 

Observable.just执行在子线程中, callBack回调执行在主线程中

基于mockito,我们直接写出对应的单元测试代码:

  1. @Test 
  2. public void threadSwitch() { 
  3.     presenter.threadSwitch(); 
  4.     // 验证callBack的success方法被调用了5次 
  5.     verify(callBack,times(5)).success(anyString()); 

执行此用例,我们会发现它会报如下错误:

  1. java.lang.ExceptionInInitializerError 
  2.   at io.reactivex.rxjava3.openharmony.schedulers.OpenHarmonySchedulers.lambda$static$0(Unknown Source) 
  3.   at io.reactivex.rxjava3.openharmony.plugins.RxOpenHarmonyPlugins.callRequireNonNull(Unknown Source) 
  4.   at io.reactivex.rxjava3.openharmony.plugins.RxOpenHarmonyPlugins.initMainThreadScheduler(Unknown Source) 
  5.   at io.reactivex.rxjava3.openharmony.schedulers.OpenHarmonySchedulers.<clinit>(Unknown Source) 
  6.   at kale.ui.shatter.test.RxSchedulerPresenter.threadSwitch(RxSchedulerPresenter.java:65) 
  7.   at kale.ui.shatter.test.RxSchedulerTestTest.threadSwitch(RxSchedulerTestTest.java:52) 
  8.   at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) 
  9.   at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) 
  10.   at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) 
  11.   at java.lang.reflect.Method.invoke(Method.java:498) 
  12.   at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:59) 
  13.   at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12) 
  14.   at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:56) 
  15.   at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17) 
  16.   at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:26) 
  17.   at org.junit.runners.ParentRunner$3.evaluate(ParentRunner.java:306) 
  18.   at org.junit.runners.BlockJUnit4ClassRunner$1.evaluate(BlockJUnit4ClassRunner.java:100) 
  19.   at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:366) 
  20.   at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:103) 
  21.   at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:63) 
  22.   at org.junit.runners.ParentRunner$4.run(ParentRunner.java:331) 
  23.   at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:79) 
  24.   at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:329) 
  25.   at org.junit.runners.ParentRunner.access$100(ParentRunner.java:66) 
  26.   at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:293) 
  27.   at org.junit.runners.ParentRunner$3.evaluate(ParentRunner.java:306) 
  28.   at org.junit.runners.ParentRunner.run(ParentRunner.java:413) 
  29.   at org.mockito.internal.runners.DefaultInternalRunner$1.run(DefaultInternalRunner.java:79) 
  30.   at org.mockito.internal.runners.DefaultInternalRunner.run(DefaultInternalRunner.java:85) 
  31.   at org.mockito.internal.runners.StrictRunner.run(StrictRunner.java:39) 
  32.   at org.mockito.junit.MockitoJUnitRunner.run(MockitoJUnitRunner.java:163) 
  33.   at org.junit.runner.JUnitCore.run(JUnitCore.java:137) 
  34.   at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:69) 
  35.   at com.intellij.rt.junit.IdeaTestRunner$Repeater.startRunnerWithArgs(IdeaTestRunner.java:33) 
  36.   at com.intellij.rt.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:220) 
  37.   at com.intellij.rt.junit.JUnitStarter.main(JUnitStarter.java:53) 
  38.   at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) 
  39.   at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) 
  40.   at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) 
  41.   at java.lang.reflect.Method.invoke(Method.java:498) 
  42.   at com.intellij.rt.execution.application.AppMainV2.main(AppMainV2.java:128) 
  43. Caused by: java.lang.RuntimeException: Stub! 
  44.   at ohos.eventhandler.EventRunner.getMainEventRunner(EventRunner.java:110) 
  45.   at io.reactivex.rxjava3.openharmony.schedulers.OpenHarmonySchedulers$MainHolder.<clinit>(Unknown Source) 
  46.   ... 41 more 

那么怎么解决呢?那就是设置用到的Schedulers.进行hook,修改用例如下:

  1. @Test 
  2.  public void threadSwitch() { 
  3.      RxJavaPlugins.setIoSchedulerHandler(scheduler -> Schedulers.trampoline()); 
  4.      RxJavaPlugins.setComputationSchedulerHandler(scheduler -> Schedulers.trampoline()); 
  5.      RxJavaPlugins.setNewThreadSchedulerHandler(scheduler -> Schedulers.trampoline()); 
  6.      RxOpenHarmonyPlugins.setInitMainThreadSchedulerHandler(scheduler -> Schedulers.trampoline()); 
  7.  
  8.      presenter.threadSwitch(); 
  9.  
  10.      // 验证callBack的success方法被调用了5次 
  11.      verify(callBack,times(5)).success(anyString()); 
  12.  } 

原理就是当进行线程调度时,都让它切换到Schedulers.trampoline(),这样我们就能正确的输出了。但通常情况下,我们使用到线程切换的场景会很多,这样写毕竟还是不够优雅,稍后我会给出更好的解决方式。

除了上面的线程切换场景,我们还经常会使用到时间轮询之类的场景,例如:

  1. public void interval() { 
  2.        Observable.interval(1, TimeUnit.SECONDS) 
  3.                .take(5) 
  4.                .flatMap((Function<Long, ObservableSource<String>>) 
  5.                        aLong -> Observable.just(aLong + "")) 
  6.                .subscribeOn(Schedulers.newThread()) 
  7.                .observeOn(OpenHarmonySchedulers.mainThread()) 
  8.                .subscribe(new Observer<String>() { 
  9.                    @Override 
  10.                    public void onSubscribe(@NonNull Disposable d) { 
  11.  
  12.                    } 
  13.  
  14.                    @Override 
  15.                    public void onNext(@NonNull String s) { 
  16.                        System.out.println(s); 
  17.                        if (callBack != null) { 
  18.                            callBack.success(s); 
  19.                        } 
  20.                    } 
  21.  
  22.                    @Override 
  23.                    public void onError(@NonNull Throwable e) { 
  24.                        if (callBack != null) { 
  25.                            callBack.failed(); 
  26.                        } 
  27.                    } 
  28.  
  29.                    @Override 
  30.                    public void onComplete() { 
  31.  
  32.                    } 
  33.                }); 
  34.    } 

我们每隔1秒发射一次数据,一共发送5次,我们写出以下单元测试:

  1. @Test 
  2.    public void interval() { 
  3.        RxJavaPlugins.setIoSchedulerHandler(scheduler -> Schedulers.trampoline()); 
  4.        RxJavaPlugins.setComputationSchedulerHandler(scheduler -> Schedulers.trampoline()); 
  5.        RxJavaPlugins.setNewThreadSchedulerHandler(scheduler -> Schedulers.trampoline()); 
  6.        RxOpenHarmonyPlugins.setInitMainThreadSchedulerHandler(scheduler -> Schedulers.trampoline()); 
  7.         
  8.        presenter.interval(); 
  9.  
  10.        // 验证callBack的success方法被调用了5次 
  11.        verify(callBack,times(5)).success(anyString()); 
  12.    } 

使用上面线程异步变同步的方法确实可以进行测试,但是需要等到5秒后才能执行完成,这显然不符合单元测试执行快的特点。这里,RxJava给我们提供了TestScheduler,调用TestScheduler的advanceTimeTo或advanceTimeBy方法来进行时间操作。

  1. @Test 
  2.    public void interval() { 
  3.        TestScheduler testScheduler = new TestScheduler(); 
  4.        RxJavaPlugins.setIoSchedulerHandler(scheduler -> testScheduler); 
  5.        RxJavaPlugins.setComputationSchedulerHandler(scheduler -> testScheduler); 
  6.        RxJavaPlugins.setNewThreadSchedulerHandler(scheduler -> testScheduler); 
  7.        RxOpenHarmonyPlugins.setInitMainThreadSchedulerHandler(scheduler -> testScheduler); 
  8.        presenter.interval(); 
  9.        //将时间设到3秒后 
  10.        testScheduler.advanceTimeTo(3,TimeUnit.SECONDS); 
  11.        verify(callBack,times(3)).success(anyString()); 
  12.        //将时间设到10秒后 
  13.        testScheduler.advanceTimeTo(10,TimeUnit.SECONDS); 
  14.        verify(callBack,times(5)).success(anyString()); 
  15.    } 

这样我们就不用每次执行到该用例的时候,还得等待设定的时间。每次这样写毕竟也不够优雅,下面我给出基于rxjava3和Rxohos:1.0.0,使用TestRule来进行RxJava线程切换及时间操作的工具类,供大家参考:

  1. /** 
  2.  * Created by xiongwg on 2021-07-08. 
  3.  * <p> 
  4.  * 这个类是让Obserable从异步变同步。 
  5.  * 
  6.  * 注意: 当有操作时间的测试时,必须调用{@link #setScheduler(Scheduler)}方法 
  7.  */ 
  8.  
  9. public class RxJavaTestSchedulerRule implements TestRule { 
  10.      /** 
  11.      * 运行在当前线程,可异步变同步 
  12.      */ 
  13.     public static final Scheduler DEFAULT_SCHEDULER = Schedulers.trampoline(); 
  14.  
  15.     /** 
  16.      * 操作时间类的  Scheduler 
  17.      */ 
  18.     public static final Scheduler TIME_SCHEDULER = new TestScheduler(); 
  19.  
  20.  
  21.     private Scheduler mScheduler = DEFAULT_SCHEDULER; 
  22.  
  23.  
  24.     /** 
  25.      * 切换 Scheduler 
  26.      * 
  27.      * @param scheduler 单元测试用例执行所在的 Scheduler 
  28.      */ 
  29.     public void setScheduler(Scheduler scheduler) { 
  30.         if (scheduler != mScheduler) { 
  31.             mScheduler = scheduler; 
  32.             resetTestSchecduler(); 
  33.         } 
  34.     } 
  35.  
  36.     @Override 
  37.     public Statement apply(final Statement base, Description description) { 
  38.         return new Statement() { 
  39.             @Override 
  40.             public void evaluate() throws Throwable { 
  41.                 resetTestSchecduler(); 
  42.                 base.evaluate(); 
  43.             } 
  44.         }; 
  45.     } 
  46.  
  47.  
  48.     public void advanceTimeBy(long delayTime, TimeUnit unit) { 
  49.         if (mScheduler instanceof TestScheduler) { 
  50.             ((TestScheduler) mScheduler).advanceTimeBy(delayTime, unit); 
  51.         } 
  52.     } 
  53.  
  54.     public void advanceTimeTo(long delayTime, TimeUnit unit) { 
  55.         if (mScheduler instanceof TestScheduler) { 
  56.             ((TestScheduler) mScheduler).advanceTimeTo(delayTime, unit); 
  57.         } 
  58.     } 
  59.  
  60.     public void triggerActions() { 
  61.         if (mScheduler instanceof TestScheduler) { 
  62.             ((TestScheduler) mScheduler).triggerActions(); 
  63.         } 
  64.     } 
  65.  
  66.     private void resetTestSchecduler() { 
  67.         RxJavaPlugins.reset(); 
  68.         RxJavaPlugins.setIoSchedulerHandler(scheduler -> mScheduler); 
  69.         RxJavaPlugins.setComputationSchedulerHandler(scheduler -> mScheduler); 
  70.         RxJavaPlugins.setNewThreadSchedulerHandler(scheduler -> mScheduler); 
  71.  
  72.         RxOpenHarmonyPlugins.reset(); 
  73.         RxOpenHarmonyPlugins.setInitMainThreadSchedulerHandler(scheduler -> mScheduler); 
  74.     } 

使用起来很简单

  1. // 1、声明RxJavaTestSchedulerRule Rule 
  2.  @Rule 
  3.  public RxJavaTestSchedulerRule rxJavaTestSchedulerRule = new RxJavaTestSchedulerRule(); 
  4.  @Test 
  5.  public void interval() { 
  6.      //2、在需要进行时间操作的方法前,设置Scheduler为TIME_SCHEDULER 
  7.      rxJavaTestSchedulerRule.setScheduler(TIME_SCHEDULER); 
  8.      presenter.interval(); 
  9.      //3、操作时间,将时间设置为3秒后 
  10.      rxJavaTestSchedulerRule.advanceTimeTo(3, TimeUnit.SECONDS); 
  11.      verify(callBack,times(3)).success(anyString()); 
  12.      //将时间设置为10秒后 
  13.      rxJavaTestSchedulerRule.advanceTimeTo(10, TimeUnit.SECONDS); 
  14.      verify(callBack,times(5)).success(anyString()); 
  15.  } 

七、其它

Java单元测试中引入了ohos相关类的解决方案

1、尝试Mock出该对象

2、在java单元测试包下新建同包名同类名的Java文件,重写调用到的方法

项目本地查看测试覆盖率

右击需要测试覆盖率的包名 ==> 点击“run test in ‘xxx’ with Coverage” 

【中软国际】HarmonyOS 非UI单元测试在DevEco Studio上的应用-鸿蒙HarmonyOS技术社区

 

项目本地查看测试案例通过率

【中软国际】HarmonyOS 非UI单元测试在DevEco Studio上的应用-鸿蒙HarmonyOS技术社区
【中软国际】HarmonyOS 非UI单元测试在DevEco Studio上的应用-鸿蒙HarmonyOS技术社区
【中软国际】HarmonyOS 非UI单元测试在DevEco Studio上的应用-鸿蒙HarmonyOS技术社区

想了解更多内容,请访问:

51CTO和华为官方合作共建的鸿蒙技术社区

https://harmonyos.51cto.com

 

责任编辑:jianghua 来源: 鸿蒙社区
相关推荐

2017-01-14 23:42:49

单元测试框架软件测试

2017-03-28 12:25:36

2022-04-08 09:01:56

脚本Go应用单元

2009-06-23 18:19:32

单元测试Hibernate配置

2022-12-08 08:01:02

Python测试单元

2012-06-18 10:03:46

Visual Stud

2010-08-27 09:11:27

Python单元测试

2023-07-26 08:58:45

Golang单元测试

2023-09-21 22:12:06

单元测试数据工程

2021-03-24 09:30:02

Jupyter not单元测试代码

2011-01-25 10:42:29

Visual Stud

2011-05-16 16:52:09

单元测试彻底测试

2017-01-14 23:26:17

单元测试JUnit测试

2017-01-16 12:12:29

单元测试JUnit

2010-10-13 09:29:53

JUnit单元测试Android

2011-06-14 15:56:42

单元测试

2020-08-18 08:10:02

单元测试Java

2022-05-12 09:37:03

测试JUnit开发

2010-03-02 09:10:41

Visual Stud

2021-09-02 15:29:42

鸿蒙HarmonyOS应用
点赞
收藏

51CTO技术栈公众号