Android AsyncTask:跳过 X 帧 - 主线程内工作过多

2024-04-26

我正在为一些网站制作一些RSS阅读器,所以我想在4.0的较低版本的Androd上实现actionbar和viewpager,所以我使用Jake Wharton的ActionBarSherlock和ViewPagerIndicator,所以我正在使用片段。

我想从 URL 读取一些 RSS 提要,并且我有一个名为 LoadXMLData 的 AsyncTask 类,下面是该类的代码。

加载XML数据 class:

public class LoadXMLData extends AsyncTask<String, RSSFeed, RSSFeed>{


private ProgressDialog mProgressDialog;


private Context context;
RSSFeed feed;
private String RSSFEEDURL = "http://balkanandroid.com/feed/";

public LoadXMLData(Context context) {
    this.context = context;
    mProgressDialog = new ProgressDialog(context);
    mProgressDialog.setMessage("Molimo Vas, sačekajte. Podaci se učitavaju.");
}

@Override
protected void onPreExecute() {
    super.onPreExecute();
    mProgressDialog.show();
    Log.d("OVDE SAM:", "onPreExecute()");
}

@Override
protected RSSFeed doInBackground(String... urls) {
    // Obtain feed
    DOMParser myParser = new DOMParser();
    feed = myParser.parseXml(urls[0]);
    Log.d("OVDE SAM:", "PARSIRAM XML");
    return feed;
}



@Override
protected void onPostExecute(RSSFeed result) {
    mProgressDialog.dismiss();
    super.onPostExecute(result);
}

}

我也有课主要活动,它扩展了 SherlockFragmentActivity。

public class MainActivity extends SherlockFragmentActivity {
BAFragmentAdapter mAdapter;
RSSFeed feed;
Application myApp;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    Log.d("OVDE SAM:", "MAIN ACTIVITY");

    myApp = getApplication();



    mAdapter = new BAFragmentAdapter(getSupportFragmentManager());

    ViewPager pager = (ViewPager) findViewById(R.id.pager);
    pager.setAdapter(mAdapter);

    TabPageIndicator indicator = (TabPageIndicator) findViewById(R.id.indicator);
    indicator.setViewPager(pager);

}

public RSSFeed getFeed() {
    return feed;
}

public void setFeed(RSSFeed feed) {
    this.feed = feed;
}

最重要的是,我有几个片段类(最新片段、电话片段、平板电脑片段、应用程序片段等),并且在每个片段上我都有几乎相同的代码,如下所示。

这是完整的代码最新片段:

public class LatestFragment extends Fragment {

GridView lv;
RSSFeed feed;
CustomListAdapter adapter;
private String RSSFEEDURL = "http://balkanandroid.com/feed/";

public View onCreateView(LayoutInflater inflater, ViewGroup container,
        Bundle savedInstanceState) {
    View view = inflater.inflate(R.layout.fragment_najnovije, container,
            false);


    AsyncTask<String, RSSFeed, RSSFeed> xml = new LoadXMLData(getActivity())
            .execute(RSSFEEDURL);

    // AsyncTask<String, RSSFeed, RSSFeed> load = new
    // LoadXMLData(getActivity()).execute(RSSFEEDURL);

    try {
        feed = xml.get();
    } catch (InterruptedException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    } catch (ExecutionException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }

    lv = (GridView) view.findViewById(R.id.GridView1);

    // Set an Adapter to the ListView
    adapter = new CustomListAdapter();
    lv.setAdapter(adapter);

    // Set on item click listener to the ListView
    lv.setOnItemClickListener(new OnItemClickListener() {

        public void onItemClick(AdapterView<?> arg0, View arg1, int arg2,
                long arg3) {
            // actions to be performed when a list item clicked
            int pos = arg2;

            Bundle bundle = new Bundle();
            bundle.putSerializable("feed", feed);
            Intent intent = new Intent(getActivity(), DetailsActivity.class);
            intent.putExtras(bundle);
            intent.putExtra("pos", pos);
            startActivity(intent);

        }
    });

    return view;
}

@Override
public void onDestroy() {
    super.onDestroy();
    adapter.imageLoader.clearCache();
    adapter.notifyDataSetChanged();
}

class CustomListAdapter extends BaseAdapter {

    private LayoutInflater layoutInflater;
    public ImageLoader imageLoader;

    public CustomListAdapter() {

        layoutInflater = (LayoutInflater) getActivity().getSystemService(
                Context.LAYOUT_INFLATER_SERVICE);
        imageLoader = new ImageLoader(getActivity().getApplicationContext());
    }

    public int getCount() {
        // TODO Auto-generated method stub
        // Set the total list item count
        return feed.getItemCount();
    }

    public Object getItem(int arg0) {
        // TODO Auto-generated method stub
        return null;
    }

    public long getItemId(int arg0) {
        // TODO Auto-generated method stub
        return 0;
    }

    public View getView(int position, View convertView, ViewGroup parent) {
        // Inflate the item layout and set the views
        View listItem = convertView;
        int pos = position;
        if (listItem == null) {
            listItem = layoutInflater.inflate(R.layout.list_item, null);
        }

        // Initialize the views in the layout
        ImageView iv = (ImageView) listItem.findViewById(R.id.thumb);
        TextView tvTitle = (TextView) listItem.findViewById(R.id.title);
        TextView tvDate = (TextView) listItem.findViewById(R.id.tvDate);

        // Set the views in the layout
        imageLoader.DisplayImage(feed.getItem(pos).getImage(), iv);
        tvTitle.setText(feed.getItem(pos).getTitle());
        tvDate.setText(feed.getItem(pos).getDate());

        return listItem;
    }

}

}

正如你所看到的,我使用AsyncTask 类 LoadXMLData读取数据doInBackground()方法,然后我使用此代码在所有片段类下调用该 AsyncTask 类,因为我需要 RSSFeed 的结果,因为我需要该数据将其显示给用户。

  AsyncTask<String, RSSFeed, RSSFeed> xml = new LoadXMLData(getActivity())
            .execute(RSSFEEDURL);   

    try {
        feed = xml.get();
    } catch (InterruptedException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    } catch (ExecutionException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }

它可以工作,但是当我单击从视图寻呼机打开其他视图时,速度很慢,几乎冻结了我的应用程序,并且我通过 Logcat 收到此消息。

07-10 00:22:40.598:我/Choreographer(623):跳过了 316 帧!应用程序可能在其主线程上做了太多工作。


那是因为你正在打电话get()在你的AsyncTask,它会阻塞。不要这样做。

相反,使用您的 feedonPostExecute() of the AsyncTask.

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Android AsyncTask:跳过 X 帧 - 主线程内工作过多 的相关文章

随机推荐