0

我的代码中有一个奇怪的行为:在模拟器上使用 asyncTask 解析本地 xml 文件比在主线程上解析它需要更长的时间。

这是我的 AsyncTask 代码:

public class MostraTutti extends SherlockActivity {
ListView lv;
final List<ListViewItem> items = new ArrayList<MostraTutti.ListViewItem>();
final ArrayList<String> nome = new ArrayList<String>();
final ArrayList<String> immagine = new ArrayList<String>();
...

final   int array_image2[] ={R.drawable.iodocloroidrossichinolina,R.drawable.acidoacetilsalicilico,
        ...};

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    setContentView(R.layout.mostra_tutti);
    lv = (ListView) findViewById(R.id.listView);
    getSupportActionBar().setDisplayShowHomeEnabled(false);

    RssFeedTask rssTask = new RssFeedTask();
       rssTask.execute();

}

private class RssFeedTask extends AsyncTask<String, Void, String> {
    private ProgressDialog Dialog;
    String response = "";

    @Override
    protected void onPreExecute() {
        Dialog = new ProgressDialog(MostraTutti.this);
        Dialog.setMessage("Leggo le sostanze...");
        Dialog.show();
    }

        @Override
        protected String doInBackground(String... urls) {
            InputStream xmlFile = getResources().openRawResource(R.raw.sostanze);
            try {

            DocumentBuilderFactory documentBuilderFactory = DocumentBuilderFactory.newInstance();
            DocumentBuilder documentBuilder = documentBuilderFactory.newDocumentBuilder();
            final Document document = documentBuilder.parse(xmlFile);
            document.getDocumentElement().normalize();
            NodeList nodeList = document.getElementsByTagName("sostanza");

            for (int i = 0; i < nodeList.getLength(); i++) {
            final int   indice = i;

            nome.add(document.getElementsByTagName("nome").item(indice).getTextContent());
            iupac.add(document.getElementsByTagName("iupac").item(indice).getTextContent());
            aspetto.add(document.getElementsByTagName("aspetto").item(indice).getTextContent());
            formula.add(document.getElementsByTagName("formula").item(indice).getTextContent());
            immagine.add(document.getElementsByTagName("immagine").item(indice).getTextContent());
            appartenenza.add(document.getElementsByTagName("appartenenza").item(indice).getTextContent());
            spiegazione.add(document.getElementsByTagName("spiegazione").item(indice).getTextContent());
            tempFus.add(document.getElementsByTagName("temperaturaFusione").item(indice).getTextContent());
            tempEboll.add(document.getElementsByTagName("temperaturaEbollizione").item(indice).getTextContent());
            solubilita.add(document.getElementsByTagName("solubilita").item(indice).getTextContent());
            note.add(document.getElementsByTagName("eccezioni").item(indice).getTextContent());

            String str = document.getElementsByTagName("formula").item(indice).getTextContent();

            str = str.replaceAll("0", "\u2080");
            str = str.replaceAll("1", "\u2081");
            str = str.replaceAll("2", "\u2082");
            str = str.replaceAll("3", "\u2083");
            str = str.replaceAll("4", "\u2084");
            str = str.replaceAll("5", "\u2085");
            str = str.replaceAll("6", "\u2086");
            str = str.replaceAll("7", "\u2087");
            str = str.replaceAll("8", "\u2088");
            str = str.replaceAll("9", "\u2089");

            final String stringa = str;
            formulaConvertita.add(stringa);

                    //CustomListViewAdapter adapter = new CustomListViewAdapter(MostraTutti.this,items);
                    //lv.setAdapter(adapter);
                    items.add(new ListViewItem()
                    {{
                        ThumbnailResource = array_image2[indice];
                        Title = document.getElementsByTagName("nome").item(indice).getTextContent();
                        SubTitle = stringa;
                    }});   
            };

        } catch (IOException e1) {
            // TODO Auto-generated catch block
            e1.printStackTrace();
        } catch (ParserConfigurationException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (SAXException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }   

        return response;
}
        @Override
        protected void onPostExecute(String result) {
            Dialog.dismiss();
            CustomListViewAdapter adapter = new CustomListViewAdapter(MostraTutti.this,items);
            lv.setAdapter(adapter);

            lv.setOnItemClickListener(
                    new OnItemClickListener()
                    {
                        public void onItemClick(AdapterView<?> arg0, View v, int position, long id)
                        {                           
                            Context context = getBaseContext();
                           Intent myIntent = new Intent(context, Dettagli.class);

                           myIntent.putExtra("nome_sostanza",nome.get(position));
                         //  myIntent.putExtra("formula",formula.get(position));
                           myIntent.putExtra("iupac",iupac.get(position));                  
                           myIntent.putExtra("aspetto",aspetto.get(position));                 
                           myIntent.putExtra("appartenenza",appartenenza.get(position));
                           myIntent.putExtra("solubilita",solubilita.get(position));
                           myIntent.putExtra("tempFus",tempFus.get(position));
                           myIntent.putExtra("tempEboll",tempEboll.get(position));
                           myIntent.putExtra("spiegazione",spiegazione.get(position));
                           myIntent.putExtra("immagine", array_image2[position]);
                           myIntent.putExtra("formulaConvertita", formulaConvertita.get(position));
                           myIntent.putExtra("note", note.get(position));
                           startActivityForResult(myIntent, 0);
                        }

                        }
                 );
        }
        }

@Override
public boolean onCreateOptionsMenu(Menu menu) {
    // Inflate the menu; this adds items to the action bar if it is present.
    getSupportMenuInflater().inflate(R.menu.activity_main, menu);
    return true;
}

class ListViewItem {
public int ThumbnailResource;
public String Title;
public String SubTitle;
}
}

这是我没有 AsyncTask 的代码:

public class MostraTutti extends SherlockActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    setContentView(R.layout.mostra_tutti);
    getSupportActionBar().setDisplayShowHomeEnabled(false);

    final ListView lv = (ListView) findViewById(R.id.listView);
    final List<ListViewItem> items = new ArrayList<MostraTutti.ListViewItem>();
    final ArrayList<String> nome = new ArrayList<String>();
    final ArrayList<String> immagine = new ArrayList<String>();
    ...

    final   int array_image2[] ={R.drawable.iodocloroidrossichinolina,R.drawable.acidoacetilsalicilico,
            ...};

InputStream xmlFile = getResources().openRawResource(R.raw.sostanze);

    try {

        DocumentBuilderFactory documentBuilderFactory = DocumentBuilderFactory.newInstance();
        DocumentBuilder documentBuilder = documentBuilderFactory.newDocumentBuilder();
        final Document document = documentBuilder.parse(xmlFile);
        document.getDocumentElement().normalize();
        //tagHandler.handleChannelTag(document);
        NodeList nodeList = document.getElementsByTagName("sostanza");

        for (int i = 0; i < nodeList.getLength(); i++) {
        final int   indice = i;

        nome.add(document.getElementsByTagName("nome").item(indice).getTextContent());
        iupac.add(document.getElementsByTagName("iupac").item(indice).getTextContent());
        aspetto.add(document.getElementsByTagName("aspetto").item(indice).getTextContent());
        formula.add(document.getElementsByTagName("formula").item(indice).getTextContent());
        immagine.add(document.getElementsByTagName("immagine").item(indice).getTextContent());
        appartenenza.add(document.getElementsByTagName("appartenenza").item(indice).getTextContent());
        spiegazione.add(document.getElementsByTagName("spiegazione").item(indice).getTextContent());
        tempFus.add(document.getElementsByTagName("temperaturaFusione").item(indice).getTextContent());
        tempEboll.add(document.getElementsByTagName("temperaturaEbollizione").item(indice).getTextContent());
        solubilita.add(document.getElementsByTagName("solubilita").item(indice).getTextContent());
        note.add(document.getElementsByTagName("eccezioni").item(indice).getTextContent());

        String str = document.getElementsByTagName("formula").item(indice).getTextContent();

        str = str.replaceAll("0", "\u2080");
        str = str.replaceAll("1", "\u2081");
        str = str.replaceAll("2", "\u2082");
        str = str.replaceAll("3", "\u2083");
        str = str.replaceAll("4", "\u2084");
        str = str.replaceAll("5", "\u2085");
        str = str.replaceAll("6", "\u2086");
        str = str.replaceAll("7", "\u2087");
        str = str.replaceAll("8", "\u2088");
        str = str.replaceAll("9", "\u2089");

        final String stringa = str;
        formulaConvertita.add(stringa);

        items.add(new ListViewItem()
            {{
                ThumbnailResource = array_image2[indice];
                Title = document.getElementsByTagName("nome").item(indice).getTextContent();
                SubTitle = stringa;
            }});
        }
    } catch (IOException e1) {
        // TODO Auto-generated catch block
        e1.printStackTrace();
    } catch (ParserConfigurationException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    } catch (SAXException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }       
    CustomListViewAdapter adapter = new CustomListViewAdapter(this,items);
    lv.setAdapter(adapter);

    lv.setOnItemClickListener(
            new OnItemClickListener()
            {
                public void onItemClick(AdapterView<?> arg0, View v, int position, long id)
                {                           
                    Context context = getBaseContext();
                   Intent myIntent = new Intent(context, Dettagli.class);

                   myIntent.putExtra("nome_sostanza",nome.get(position));
                 //  myIntent.putExtra("formula",formula.get(position));
                   myIntent.putExtra("iupac",iupac.get(position));                  
                   myIntent.putExtra("aspetto",aspetto.get(position));                 
                   myIntent.putExtra("appartenenza",appartenenza.get(position));
                   myIntent.putExtra("solubilita",solubilita.get(position));
                   myIntent.putExtra("tempFus",tempFus.get(position));
                   myIntent.putExtra("tempEboll",tempEboll.get(position));
                   myIntent.putExtra("spiegazione",spiegazione.get(position));
                   myIntent.putExtra("immagine", array_image2[position]);
                   myIntent.putExtra("formulaConvertita", formulaConvertita.get(position));
                   myIntent.putExtra("note", note.get(position));
                   startActivityForResult(myIntent, 0);
                }

                }
         );
}

@Override
public boolean onCreateOptionsMenu(Menu menu) {
    getSupportMenuInflater().inflate(R.menu.activity_main, menu);
    return true;
}

class ListViewItem {
public int ThumbnailResource;
public String Title;
public String SubTitle;
}
}

模拟器上的差异是主线程一的 16 秒和 AsyncTask 一的 1 分钟!

4

2 回答 2

2

我已将此问题标记为重复,但上次我这样做时,审阅者投了反对票。

无论哪种方式,原因是它AsyncTaskdoInBackground()后台优先级中运行,并且必须与所有其他后台任务(例如您的 RSS 阅读器等)共享最多 10% 的 CPU 时间——无论系统实际上有多空闲。

有关详细讨论和解决方案,请参见此处

于 2013-04-25T11:12:56.690 回答
0

UI 线程上的 16 秒仍然太多,这不是解决方案 - 您的用户每次都会受到 ANR 对话框的打击。所以有几点建议:

  1. 优化里面的代码doInBackground()
  2. 考虑使用另一个 XML 解析器,它可能会提高性能。
  3. 如果适合您的应用程序逻辑,请将此代码移至后台Service,使其在用户未与您的应用程序交互时处理数据。Activity将信息存储在数据库中并在打开时加载它。
于 2013-04-25T11:26:43.350 回答