您好,我正在考虑一起处理Activity
,Fragment
,AsyncTask
和DialogFragments
的正确和最佳方法是什么。
我目前的状态是,我启动Activity
并将其ContentView替换为我的Fragment
,其中我有一个EditText
和一个Button
。
点击我的Button
执行AsyncTasks
请求随机内容并花费一些时间。与此同时,我显示DialogFragment
请求耐心。
期望的行为是,例如,我在我的DialogFragment
正在运行的时间内保持显示AsyncTask
。之后,我想展示一个简单的吐司,显示我从HttpRequest
获得的信息。
关于我认为它如何运作的简要概述:
BaseFragment
保留WeakReference
与Activity
相关联的AsyncTask
保留WeakReference
到Fragment
,以便对其进行解析AsyncTasks
onPreExecute()显示DialogFragment
AsyncTasks
onPostExecute()驳回DialogFragment
BaseFragment
持有DialogFragment
不幸的是,这不是它的工作方式,在方向更改时,我的DialogFragment会一直显示,并且没有显示toast。 我究竟做错了什么 ?
public class BaseFragment extends Fragment{
private static final String TAG = BaseFragment.class.getSimpleName();
protected WeakReference<AppCompatActivity> mActivity;
private TemplateDialogFragment dialogFragment;
public WeakReference<AppCompatActivity> getAppCompatActivity(){ return mActivity; }
@Override
public void onAttach(Context context) {
if(!(context instanceof AppCompatActivity)) {
throw new IllegalStateException(TAG + " is not attached to an AppCompatActivity.");
}
mActivity = new WeakReference<>((AppCompatActivity) context);
super.onAttach(context);
}
@Override
public void onDetach() {
mActivity = null;
super.onDetach();
}
@Override
public void onStart() {
super.onStart();
showContent();
}
public void showContent(){
}
public void showDialog(String title, String content){
dialogFragment = new TemplateDialogFragment();
Bundle bundle = new Bundle();
bundle.putString(TemplateDialogFragment.DIALOG_TITLE, title);
bundle.putString(TemplateDialogFragment.DIALOG_MESSAGE, content);
dialogFragment.setArguments(bundle);
dialogFragment.show(getFragmentManager(), TemplateDialogFragment.FRAGMENT_TAG);
}
public void notifyTaskFinished(String result) {
dismissDialog();
if(mActivity != null && !mActivity.get().isFinishing()) {
Toast.makeText(mActivity.get().getApplicationContext(), result, Toast.LENGTH_LONG).show();
}
}
private void dismissDialog(){
if(dialogFragment != null && dialogFragment.isAdded()) {
dialogFragment.dismissAllowingStateLoss();
}
}
}
...
public class TemplateFragment extends BaseFragment {
private static final String TAG = TemplateFragment.class.getSimpleName();
@Nullable
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
return inflater.inflate(R.layout.test_fragment, container, false);
}
@Override
public void showContent() {
super.showContent();
Button startTask = (Button) getAppCompatActivity().get().findViewById(R.id.button0);
final BaseFragment instance = this;
startTask.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
CustomAsyncTask task = new CustomAsyncTask(instance);
EditText input = (EditText) getAppCompatActivity().get().findViewById(R.id.text0);
task.execute(input.getText().toString());
}
});
}
private static class CustomAsyncTask extends AsyncTask<String, Void, String> {
WeakReference<BaseFragment> weakBaseFragmentReference;
private CustomAsyncTask(BaseFragment fragment) {
weakBaseFragmentReference = new WeakReference<>(fragment);
}
@Override
protected void onPreExecute() {
super.onPreExecute();
weakBaseFragmentReference.get().showDialog("Executing", "Working on the request...");
}
@Override
protected String doInBackground(String... params) {
HttpURLConnection con = HttpUrlConnectionFactory.createUrlConnection("https://www.httpbin.org/bytes/" + (params[0] == null ? "1" : params[0]));
return HttpRequester.doGet(con).getResponseAsString();
}
@Override
protected void onPostExecute(String response) {
super.onPostExecute(response);
if(weakBaseFragmentReference.get() == null) {
return;
}
weakBaseFragmentReference.get().notifyTaskFinished(response);
}
}
}
*编辑:
经过一段时间研究这个主题后,我确信Service
是我大部分使用领域的最佳解决方案。我也经常使用AsyncTaskLoaders
,因为可以顺利控制生命周期 ....
答案 0 :(得分:1)
使用进度条而不是DialogFragment。
AsyncTask只应用于需要几秒钟的任务。
AsyncTask不尊重Activity生命周期,可能导致内存泄漏。
检查一些gotchas。
您可以尝试使用AsyncTaskLoader来更改配置。