sta*_*ej2 48 junit android unit-testing rx-java rx-java2
我在尝试为正在使用的演示者运行JUnit测试时遇到RuntimeException observeOn(AndroidSchedulers.mainThread()).
由于它们是纯JUnit测试而不是Android检测测试,因此它们无法访问Android依赖项,导致我在执行测试时遇到以下错误:
java.lang.ExceptionInInitializerError
    at io.reactivex.android.schedulers.AndroidSchedulers$1.call(AndroidSchedulers.java:35)
    at io.reactivex.android.schedulers.AndroidSchedulers$1.call(AndroidSchedulers.java:33)
    at io.reactivex.android.plugins.RxAndroidPlugins.callRequireNonNull(RxAndroidPlugins.java:70)
    at io.reactivex.android.plugins.RxAndroidPlugins.initMainThreadScheduler(RxAndroidPlugins.java:40)
    at io.reactivex.android.schedulers.AndroidSchedulers.<clinit>(AndroidSchedulers.java:32)
    …
Caused by: java.lang.RuntimeException: Method getMainLooper in android.os.Looper not mocked. See http://g.co/androidstudio/not-mocked for details.
    at android.os.Looper.getMainLooper(Looper.java)
    at io.reactivex.android.schedulers.AndroidSchedulers$MainHolder.<clinit>(AndroidSchedulers.java:29)
    ...
java.lang.NoClassDefFoundError: Could not initialize class io.reactivex.android.schedulers.AndroidSchedulers
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:498)
    …
sta*_*ej2 66
发生此错误的原因是返回的默认调度程序AndroidSchedulers.mainThread()是一个实例,LooperScheduler并依赖于JUnit测试中不可用的Android依赖项.
我们可以通过RxAndroidPlugins在运行测试之前使用不同的Scheduler 初始化来避免此问题.您可以在这样的@BeforeClass方法中执行此操作:
@BeforeClass
public static void setUpRxSchedulers() {
    Scheduler immediate = new Scheduler() {
        @Override
        public Disposable scheduleDirect(@NonNull Runnable run, long delay, @NonNull TimeUnit unit) {
            // this prevents StackOverflowErrors when scheduling with a delay
            return super.scheduleDirect(run, 0, unit);
        }
        @Override
        public Worker createWorker() {
            return new ExecutorScheduler.ExecutorWorker(Runnable::run);
        }
    };
    RxJavaPlugins.setInitIoSchedulerHandler(scheduler -> immediate);
    RxJavaPlugins.setInitComputationSchedulerHandler(scheduler -> immediate);
    RxJavaPlugins.setInitNewThreadSchedulerHandler(scheduler -> immediate);
    RxJavaPlugins.setInitSingleSchedulerHandler(scheduler -> immediate);
    RxAndroidPlugins.setInitMainThreadSchedulerHandler(scheduler -> immediate);
}
或者,您可以创建一个自定义TestRule,允许您在多个测试类中重用初始化逻辑.
public class RxImmediateSchedulerRule implements TestRule {
    private Scheduler immediate = new Scheduler() {
        @Override
        public Disposable scheduleDirect(@NonNull Runnable run, long delay, @NonNull TimeUnit unit) {
            // this prevents StackOverflowErrors when scheduling with a delay
            return super.scheduleDirect(run, 0, unit);
        }
        @Override
        public Worker createWorker() {
            return new ExecutorScheduler.ExecutorWorker(Runnable::run);
        }
    };
    @Override
    public Statement apply(final Statement base, Description description) {
        return new Statement() {
            @Override
            public void evaluate() throws Throwable {
                RxJavaPlugins.setInitIoSchedulerHandler(scheduler -> immediate);
                RxJavaPlugins.setInitComputationSchedulerHandler(scheduler -> immediate);
                RxJavaPlugins.setInitNewThreadSchedulerHandler(scheduler -> immediate);
                RxJavaPlugins.setInitSingleSchedulerHandler(scheduler -> immediate);
                RxAndroidPlugins.setInitMainThreadSchedulerHandler(scheduler -> immediate);
                try {
                    base.evaluate();
                } finally {
                    RxJavaPlugins.reset();
                    RxAndroidPlugins.reset();
                }
            }
        };
    }
}
然后,您可以将其应用于您的测试类
public class TestClass {
    @ClassRule public static final RxImmediateSchedulerRule schedulers = new RxImmediateSchedulerRule();
    @Test
    public void testStuff_stuffHappens() {
       ...
    }
}
这两种方法都将确保在执行任何测试之前和AndroidSchedulers访问之前覆盖默认调度程序.
使用立即调度程序覆盖RxJava调度程序以进行单元测试还将确保正在测试的代码中的RxJava用法同步运行,这将使编写单元测试变得更加容易.
资料来源:
https
://www.infoq.com/articles/Testing-RxJava2 https://medium.com/@peter.tackage/overriding-rxandroid-schedulers-in-rxjava-2-5561b3d14212
Jkn*_*air 38
我刚才补充道
RxAndroidPlugins.setInitMainThreadSchedulerHandler(scheduler -> Schedulers.trampoline());
在@Before注释方法中.
s-h*_*ter 15
我在测试LiveData时遇到了同样的错误.在测试LiveData时,如果被测试的类同时具有后台线程和LiveData ,则除了RxImmediateSchedulerRule之外,还需要此InstantTaskExecutorRule.
@RunWith(MockitoJUnitRunner::class)
class MainViewModelTest {
    companion object {
        @ClassRule @JvmField
        val schedulers = RxImmediateSchedulerRule()
    }
    @Rule
    @JvmField
    val rule = InstantTaskExecutorRule()
    @Mock
    lateinit var dataRepository: DataRepository
    lateinit var model: MainViewModel
    @Before
    fun setUp() {
      model = MainViewModel(dataRepository)
    }
    @Test
    fun fetchData() {
      //given    
      val returnedItem = createDummyItem()    
      val observer = mock<Observer<List<Post>>>()    
      model.getPosts().observeForever(observer)    
      //when    
      liveData.value = listOf(returnedItem)    
      //than    
      verify(observer).onChanged(listOf(Post(returnedItem.id, returnedItem.title, returnedItem.url)))
    }
}
参考:https: //pbochenski.pl/blog/07-12-2017-testing_livedata.html
Gen*_*ani 11
基于@ starkej2答案,经过一些更改,Kotlin开发人员的正确答案是:
RxImmediateSchedulerRule.kt类:,
import io.reactivex.Scheduler
import io.reactivex.android.plugins.RxAndroidPlugins
import io.reactivex.internal.schedulers.ExecutorScheduler
import io.reactivex.plugins.RxJavaPlugins
import org.junit.rules.TestRule
import org.junit.runner.Description
import org.junit.runners.model.Statement
import java.util.concurrent.Executor
class RxImmediateSchedulerRule : TestRule {
    private val immediate = object : Scheduler() {
        override fun createWorker(): Worker {
            return ExecutorScheduler.ExecutorWorker(Executor { it.run() })
        }
    }
    override fun apply(base: Statement, description: Description): Statement {
        return object : Statement() {
            @Throws(Throwable::class)
            override fun evaluate() {
                RxJavaPlugins.setInitIoSchedulerHandler { immediate }
                RxJavaPlugins.setInitComputationSchedulerHandler { immediate }
                RxJavaPlugins.setInitNewThreadSchedulerHandler { immediate }
                RxJavaPlugins.setInitSingleSchedulerHandler { immediate }
                RxAndroidPlugins.setInitMainThreadSchedulerHandler { immediate }
                try {
                    base.evaluate()
                } finally {
                    RxJavaPlugins.reset()
                    RxAndroidPlugins.reset()
                }
            }
        }
    }
}
在您的测试类上,创建调度程序 ClassRule:
class TestViewModelTest {
companion object {
   @ClassRule
   @JvmField
   val schedulers = RxImmediateSchedulerRule()
}
@Before
fun setUp() {
    //your setup code here
}
@Test
fun yourTestMethodHere{}
}
就像Peter Tackage在这篇中型文章中的建议一样,您可以自己注入Scheduler。
我们都知道,直接调用静态方法会导致难以测试的类,并且如果您使用像Dagger 2这样的依赖项注入框架,则注入Scheduler会特别容易。示例如下:
在您的项目中定义一个接口:
public interface SchedulerProvider {
    Scheduler ui();
    Scheduler computation();
    Scheduler io();
    Scheduler special();
    // Other schedulers as required…
}
定义一个实现:
final class AppSchedulerProvider implements SchedulerProvider {
    @Override 
    public Scheduler ui() {
        return AndroidSchedulers.mainThread();
    }
    @Override 
    public Scheduler computation() {
        return Schedulers.computation();
    }
    @Override 
    public Scheduler io() {
        return Schedulers.io();
    }
    @Override 
    public Scheduler special() {
        return MyOwnSchedulers.special();
    }
}
现在,不要像这样直接使用对调度程序的引用:
 bookstoreModel.getFavoriteBook()
               .map(Book::getTitle)
               .delay(5, TimeUnit.SECONDS)
               .observeOn(AndroidSchedulers.mainThread())
               .subscribe(view::setBookTitle));
您使用对接口的引用:
bookstoreModel.getFavoriteBook()
          .map(Book::getTitle)
          .delay(5, TimeUnit.SECONDS, 
                 this.schedulerProvider.computation())
          .observeOn(this.schedulerProvider.ui())
          .subscribe(view::setBookTitle));
现在为您的测试,您可以这样定义一个TestSchedulersProvider:
public final class TestSchedulersProvider implements SchedulerProvider {
      @Override
      public Scheduler ui() {
          return new TestScheduler();
      }
      @Override
      public Scheduler io() {
          return Schedulers.trampoline(); //or test scheduler if you want
      }
      //etc
}
现在TestScheduler,您可以在单元测试中拥有使用时的所有优点。对于需要测试延迟的情况,这很有用:
@Test
public void testIntegerOneIsEmittedAt20Seconds() {
    //arrange
    TestObserver<Integer> o = delayedRepository.delayedInt()
            .test();
    //act
    testScheduler.advanceTimeTo(20, TimeUnit.SECONDS);
    //assert
    o.assertValue(1);
}
否则,如果您不想使用注入的Scheduler,则可以使用lambdas完成其他方法中提到的静态钩子:
@Before
public void setUp() {
    RxAndroidPlugins.setInitMainThreadSchedulerHandler(h -> Schedulers.trampoline());
    RxJavaPlugins.setIoSchedulerHandler(h -> Schedulers.trampoline());
//etc
}
| 归档时间: | 
 | 
| 查看次数: | 18952 次 | 
| 最近记录: |