- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我有一个从远程服务器检索 json 的配置管理器。然后它被解析为一个 ConfigWrapper 对象,该对象在提供的回调监听器中作为参数返回。
所以在测试类中我调用:
@Test
public void init_Configuration_With_Network_Load_JSON_From_Server_Return_To_Listener() {
mockConnectivityCheck(true);
...
mManager.initConfiguration(mContext, eq(anyString()), mListener);
verify(mListener, times(1)).onConfigurationLoaded(any(ConfigWrapper.class));
}
mContext 被模拟,mListener 也被模拟。这将调用被测类的方法:
public void initConfiguration(Context context, String url, ConfigurationManagerListener listener){
// Get a reference to shared preferences
mSharedPref = context.getSharedPreferences(PREF_FILE, Context.MODE_PRIVATE);
// If we have network, we should load configurations from remote, otherwise, try to load it from assets folder OR shared pref
if (NetworkTools.hasNetworkConnection(context)){
getConfigurationRemote(context, url, listener);
} else {
getConfigurationOffline(context, listener);
}
}
所以如果我们有网络,我们可以从服务器获取配置。这个方法做到了:
private void getConfigurationRemote(final Context context, String url, final ConfigurationManagerListener
listener) {
// Send a request to get the configuration
new AsyncTask<String, Void, HashMap<String, Object>> () {
@Override
protected HashMap<String, Object> doInBackground(String... params) {
InputStream in = null;
HashMap result = null;
try {
URL url = new URL(params[0]);
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
conn.setReadTimeout(10000 /* milliseconds */);
conn.setConnectTimeout(10000 /* milliseconds */);
conn.setRequestMethod("GET");
conn.setDoInput(true);
conn.connect();
in = conn.getInputStream();
result = new ObjectMapper().readValue(in, HashMap.class);
in.close();
} catch (IOException e) {
e.printStackTrace();
} finally {
try {
if (in != null){
in.close();
}
} catch (IOException e) {
e.printStackTrace();
}
return result;
}
}
@Override
protected void onPostExecute(HashMap<String, Object> config) {
// We if have a valid result, save the data to shared pref AND save it to a ConfigurationWrapper object
// other wise, try to load from assets folder or shared pref if available
if (config != null && config.size() > 0){
// We want to save the hierarchy of the JSON so we save its string representation to shared pref
JSONObject object = new JSONObject(config);
mSharedPref.edit().putString(CONFIGURATIONS_KEY, object.toString()).apply();
mConfigurationWrapper = new ConfigWrapper(config);
listener.onConfigurationLoaded(mConfigurationWrapper);
} else {
// No valid configuration from remote server, so load it from local source
getConfigurationOffline(context, listener);
}
}
}.execute(url);
}
现在,我正在尝试使用 Mockito(或 PowerMockito,如果需要)编写单元测试来测试此代码。我不完全确定如何解决这种情况,我调用的方法中有一个新的 AsyncTask().execute() 调用。到目前为止,调用 initConfiguration 方法并模拟网络检查返回 true,在调用 execute() 后停止。 doInBackground() 似乎没有被调用。
您将如何测试此类代码?谢谢!
最佳答案
代码的设置方式使单元测试变得困难。我建议重新安排它。
getConfigurationRemote 方法目前做 3 件独立的事情:
AsyncTask
我的解决方案:
AsyncTask
移到它自己的类中。new AsyncTask(....
) 移至工厂类,或者更好的是,使用依赖注入(inject)框架,如 Dagger。这就是我想象中的最终样子:
private void getConfigurationRemote(final Context context, String url, final ConfigurationManagerListener listener) {
// create the task
ConfigurationRemoteAsyncTask task = taskFactory.createConfigurationRemoteTask(listener);
// Send a request to get the configuration
task.execute(url);
}
现在您可以更轻松地模拟和测试:
通过简单地模拟任务工厂返回的任务,验证在调用 .initConfiguration 时使用给定的 url 调用了 task.execute。
验证在调用 task.onPostExecute 时调用了 listener.onConfigurationLoaded,而不模拟整个网络基础设施。这可能看起来像这样:
@Test
public void init_Configuration_With_Network_Load_JSON_From_Server_Return_To_Listener() {
ConfigurationRemoteAsyncTask task = new ConfigurationRemoteAsyncTask(mockedListener);
HashMap<String, Object> config = getNotEmptyConfig();
task.onPostExecute(config);
verify(mockedListener).onConfigurationLoaded(any(ConfigWrapper.class));
}
关于Android AsyncTask 和 Mockito(或 PowreMockito),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33632389/
我今天在我的 Android 项目中遇到了一个涉及 aSyncTasks 的问题,经过一些研究找到了答案,并且与我交谈过的人都没有意识到,所以我想我会与 SO 社区分享,以防万一有人发现它有任何用处。
我目前正在 AsyncTask 的 onPostExecute 方法中执行类似的操作,其中 NewTask 不是当前正在执行的任务: private class OlderTask extends A
所以,我正在研究一个条形码解码器,一旦我们有了条形码,它就会通过互联网进入多个 API 来解码刚刚扫描的内容。问题是我必须将一些 XML 解析链接在一起,我不知道我做的是否正确。 因此,扫描条形码后,
我的团队开发了一个新的 Android 应用程序,它广泛使用了 Room。 我不确定我们是否正确使用 AsyncTask。 我们不得不在 AsyncTasks 中包装所有对 insert/update
我查看了其他问题,但未能澄清我对从另一个任务调用任务的疑问,我有以下代码: protected List doInBackground(String... params) try {
我正在开发一个访问 Web 服务的应用程序,并使用从中获取的 JSON 创建一个对象并在我的代码中使用它。尽管我的应用程序正在运行,但我不知道它是否写得很好且完美无缺。 我将解释我的内容,然后放置一些
虽然我还没有尝试过,但从理论上讲,我问这个问题只是为了消除我的疑虑。 我有这样一个场景:1. 向服务器发送请求并接收 JSON 响应。为此,我正在使用 AsyncTask,因为接收响应可能会有延迟。2
我有以下 AsyncTask 的实现,允许多个 AsyncTask 同时运行: public abstract class MyAsyncTask extends AsyncTask { pu
花了很多时间试图解决这个问题,我已经阅读了很多问题、论坛、答案......但它仍然不会更新 UI。 我的最终目标是从用户那里获取一个搜索词,并将一个 httprequest 发送到用 JSON 回复的
我有一个异步任务 private class LoadData extends AsyncTask { private String WEBURL; LoadData(String u
我正在用 android 做一些编码工作。我几乎遇到了一个问题,为了解决这个问题,我需要一个匿名 AsyncTask 类来执行。但我还需要在执行之前传递并反对这个类。我尝试了下面的代码,但它不起作用,
我有两个 AsyncTask:第一个寻找素数,如果成功,我必须调用第二个 AsyncTask 显示单词“Yop!” (将这个词添加到数组列表中,并显示在AsyncTask三)。 如果我从 onProg
我想使用 AsyncTask 将图像加载到 ListView。 private class LoadImageTask extends AsyncTask,Void,Bitmap>{ @Sup
在我的 AsyncTask 的某个时刻,在完成一些验证之后,我需要派生另一个线程来做一些其他工作。所以我现在想要两个后台线程,每个都做自己的事情(每个执行大约 2-3 秒)。我们的想法是最大限度地提高
(这与空指针无关):我在 AsyncTask 中有一个进度条,并且添加了一个取消按钮来取消 asynctask。 我可以从异步任务外部取消异步任务,但我需要在异步任务下实现的progressdialo
我有一个 Activity ,在启动时调用“json”来获取歌曲的数据类别,之后我调用方法“AsyncTask”来获取来自另一个“JSON”的歌曲列表问题是,当我启动 Activity 时,它被锁定,
我想做以下事情。我想显示一个包含信息和图像的列表。这些图像需要一段时间才能加载,所以我想我会采取不同的方式。我会使用两个 AsyncTasks。第一个创建所有布局并用除图像之外的数据填充它。第二个只是
如果我做了这样的事情: public class MyFragment extends Fragment { private GetDataTask mGDT; //onCreat
在 Android Activity 中,我在 onCreate 方法中执行 AsyncTask。我应该在 AsyncTask 的 onPostExecute 中还是在 OnCreate 方法中声明
我对 AsyncTask 有疑问我尝试为 10 个 Json 文件向互联网打开 10 个请求,因此我读取它并将其保存到用户设备_由于数据差异,此文件必须分开,包括。 那么,将每个请求放在单个 Asyn
我是一名优秀的程序员,十分优秀!