gpt4 book ai didi

android - 使用 Robolectric 和 Mockito 测试 CursorLoader

转载 作者:塔克拉玛干 更新时间:2023-11-02 20:08:52 26 4
gpt4 key购买 nike

鉴于我正在开发一个简单的 ListFragment(在本例中,它从 MediaStore 读取艺术家列表,但稍后也会从不同的来源读取数据),如下所示:

@EFragment
public class ArtistsFragment extends ListFragment implements LoaderManager.LoaderCallbacks<Cursor> {
private static final String TAG = ArtistsFragment.class.getName();
private SimpleCursorAdapter mAdapter;

Uri uri = MediaStore.Audio.Artists.EXTERNAL_CONTENT_URI;

CursorLoader mCursorLoader;

@AfterViews
void setupView() {
mAdapter = new SimpleCursorAdapter(getActivity(),
android.R.layout.simple_list_item_1, null,
new String[]{MediaStore.Audio.Artists.ARTIST}, // lists path of files
new int[]{android.R.id.text1}, 0);

setListAdapter(mAdapter);

getLoaderManager().initLoader(0, null, this);
}

@Override
public Loader<Cursor> onCreateLoader(int id, Bundle args) {
if (mCursorLoader == null) {
mCursorLoader = new CursorLoader(getActivity(), uri, new String[]{MediaStore.Audio.Artists._ID, MediaStore.Audio.Artists.ARTIST},
null, null, MediaStore.Audio.Artists.ARTIST + " ASC");
} else {
System.out.println("mCursorLoader.count: " + mCursorLoader.loadInBackground().getCount());
}
return mCursorLoader;
}

@Override
public void onLoadFinished(Loader<Cursor> loader, Cursor data) {
setListShown(true);
mAdapter.swapCursor(data);
}

@Override
public void onLoaderReset(Loader<Cursor> loader) {
mAdapter.swapCursor(null);
}
}

我想使用 Robolectric + Mockito + awaitility 以证明 Fragment 在各种条件下(例如空列表或无效数据等)表现正常。我的测试类如下所示:

@RunWith(RobolectricTestRunner.class)
public class ArtistsFragmentTest {
@Test
public void shouldNotBeNull() {
final ArtistsFragment myFragment = ArtistsFragment_.builder().build();
assertNotNull(myFragment);

// Create a mock cursor.
final Cursor mc = getSampleArtistCursor();
when(mc.getCount()).thenReturn(1);
when(mc.getInt(0)).thenReturn(1);
when(mc.getString(1)).thenReturn("Sample Title");

myFragment.mCursorLoader = mock(CursorLoader.class);
when(myFragment.mCursorLoader.loadInBackground()).thenReturn(mc);

startFragment(myFragment);

assertNotNull(myFragment.getListView());
await().atMost(5, TimeUnit.SECONDS).until(new Callable<Integer>() {
@Override
public Integer call() throws Exception {
return myFragment.getListAdapter().getCount();
}
}, equalTo(1));

System.out.println(myFragment.getListAdapter().getCount());
}

private Cursor getSampleArtistCursor() {
return new CursorWrapper(mock(MockCursor.class));
}
}

然后在 IntelliJ 或 Maven 中运行此测试时,测试将失败,适配器将始终返回零计数。

然而,onCreateLoader 中的 System.out.println 语句返回 1。我是否需要特别注意 后台线程 中的 Mockito? (loadInBackground 方法在工作线程上运行)。

最佳答案

我刚刚在我的代码中进行了 Loader 测试。在我的例子中,我发现测试加载器本身比尝试通过 Fragment 代码路由它更直接。

我最终使用了这篇文章中代码的一个稍微修改过的版本: https://groups.google.com/forum/#!msg/robolectric/xY5MF399jA8/V5PnUfh1D-wJ

首先,我必须实现一些影子类,因为 Robolectric2 不包含 AsyncTaskLoader 或 Loader 类的影子代码。如果您从未添加影子类,请知道将它们放在正确的包中很重要。这两个影子都应该存在于 android.support.v4.content 中。

影子加载器

@Implements(Loader.class)
public class ShadowLoader<D> {

// //////////////////////
// Constants

// //////////////////////
// Fields
protected boolean reset;

// //////////////////////
// Constructors

// //////////////////////
// Getter & Setter

// //////////////////////
// Methods from SuperClass/Interfaces

@Implementation
public void reset() {
reset = true;
}

@Implementation
public boolean isReset() {
return reset;
}
// //////////////////////
// Methods

// //////////////////////
// Inner and Anonymous Classes
}

ShadowAsyncTaskLoader

@Implements(AsyncTaskLoader.class)
public class ShadowAsyncTaskLoader<D> extends ShadowLoader {

@RealObject
private AsyncTaskLoader<D> asyncTaskLoader;

@Implementation
void executePendingTask() {
new AsyncTask<Void, Void, D>() {
@Override
protected D doInBackground(Void... params) {
return (D) asyncTaskLoader.loadInBackground();
}

@Override
protected void onPostExecute(D data) {
updateLastLoadCompleteTimeField();
asyncTaskLoader.deliverResult(data);
}

@Override
protected void onCancelled(D data) {
updateLastLoadCompleteTimeField();
asyncTaskLoader.onCanceled(data);

executePendingTask();
}
}.execute((Void)null);
}


public void setReset(boolean reset) {
this.reset = reset;
}

private void updateLastLoadCompleteTimeField() {
try {
Field mLastLoadCompleteTimeField = AsyncTaskLoader.class.getDeclaredField("mLastLoadCompleteTime");
if(!mLastLoadCompleteTimeField.isAccessible()) {
mLastLoadCompleteTimeField.setAccessible(true);
}
mLastLoadCompleteTimeField.set(asyncTaskLoader, SystemClock.uptimeMillis());

} catch(NoSuchFieldException e) {
e.printStackTrace();
} catch(IllegalAccessException e) {
e.printStackTrace();
}
}
}

然后,根据您的配置,您可以添加注释以使用自定义类

@Config( shadows = { ShadowAsyncTaskLoader.class, ShadowLoader.class})

此时调用 loader.onStartLoading() 导致加载程序按预期运行,而无需将任何等待命令破解到我的测试用例中。

希望这对您有所帮助。我还没有尝试将 LoaderManager 与这种测试方法结合使用,因此我无法通过该调用验证它是否有效。

注意:我添加 ShadowLoader 的原因是因为我发现 isReset() 在我没想到的时候返回了 true。

关于android - 使用 Robolectric 和 Mockito 测试 CursorLoader,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17486689/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com