首页 > 解决方案 > 处理程序未在 Instrumentation Test 中执行 Runnable

问题描述

我编写了一个 Android 仪器测试,它调用我的服务并通过广播接收答案。

与服务对话的要测试的代码使用处理程序。

在测试我的测试过程中 ^^ 我注意到处理程序的行为不符合预期。所以我写了一个测试来检查这种行为:

import android.os.Handler;
import android.support.test.annotation.UiThreadTest;

import org.junit.Assert;
import org.junit.Test;

import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;

@RunWith(AndroidJUnit4.class)
public class HandlerTest {

    private CountDownLatch countDownLatch;

    @Test
    @UiThreadTest
    public void handlerTest() {

        final Handler handler = new Handler();
        countDownLatch = new CountDownLatch(1);

        final Runnable r = new Runnable() {
            @Override
            public void run() {
                // this code gets not executed
                countDownLatch.countDown();
            }
        };

        handler.postDelayed(r, 1000);

        try {
            final boolean finishedWithoutTimeout
                    = countDownLatch.await(5, TimeUnit.SECONDS);
            Assert.assertTrue(finishedWithoutTimeout);

        } catch (final InterruptedException e) {
            e.printStackTrace();
        }
    }
}

处理程序不执行可运行代码。这也是我的生产代码的问题。

我用注释Looper.prepare()修复了一个问题。@UiThreadTest

对我的处理程序问题有什么建议吗?

标签: javaandroidtestingandroid-serviceandroid-handler

解决方案


原因是你把你的锁在thread这里:

final boolean finishedWithoutTimeout = countDownLatch.await(5, TimeUnit.SECONDS);

并且当thread被锁定时,您无法rannable使用handler. 刚刚被Thread锁定。您可以通过将您的链接handler到另一个来解决它thread。这是一个简单的解决方案handlerThread

@Test
@UiThreadTest
public void handlerTest() {
    countDownLatch = new CountDownLatch(1);
    final HandlerThread handlerThread = new HandlerThread("solution!");
    handlerThread.start();
    final Runnable r = new Runnable() {
        @Override
        public void run() {
            countDownLatch.countDown();
        }
    };
    Handler handler = new Handler(handlerThread.getLooper());
    handler.postDelayed(r, 1000);
    try {
        final boolean finishedWithoutTimeout = countDownLatch.await(5, TimeUnit.SECONDS);
        Assert.assertTrue(finishedWithoutTimeout);
    } catch (final InterruptedException e) {
        e.printStackTrace();
    }
}

推荐阅读