0

学校向けの非常に単純なゲームを書いていますが、asynctask を使用する必要があります。私はADT(Eclipse)を使用しています。関連するコードは次のとおりです。

 public void oklog(View view) throws ParserConfigurationException, TransformerConfigurationException, TransformerException, URISyntaxException, ClientProtocolException, IOException, IllegalStateException, SAXException {

    new log_async().execute();
}

String DocumentToString(Document doc) throws TransformerConfigurationException, TransformerException{
    TransformerFactory tf = TransformerFactory.newInstance();
    Transformer transformer = tf.newTransformer();
    transformer.setOutputProperty(OutputKeys.OMIT_XML_DECLARATION, "yes");
    StringWriter writer = new StringWriter();
    transformer.transform(new DOMSource(doc), new StreamResult(writer));
    String output = writer.getBuffer().toString();//.replaceAll("\n|\r", "");

    return output;
}
class log_async extends AsyncTask<String, Void, String> {

    protected void onPreExecute(){

        TextView tv1 = (TextView) findViewById(R.id.textView3);
        tv1.setText("Pracuję...");

    }

    @Override
    protected String doInBackground(String...voids ) {

        TextView tv1 = (TextView) findViewById(R.id.textView3);
        tv1.setText("haha");
        String a = "";
        try {
            tv1.setText("haha2");

        HttpClient client = new DefaultHttpClient();
        DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
        factory.setNamespaceAware(true);
        DocumentBuilder docBuilder = null;
        docBuilder = factory.newDocumentBuilder();
        tv1.setText("haha3");
        Document doc = docBuilder.newDocument();

        Element env = doc.createElementNS("http://schemas.xmlsoap.org/soap/envelope/", "soap:Envelope");
        Element body = doc.createElementNS("http://schemas.xmlsoap.org/soap/envelope/", "soap:Body");
        Element zaloguj = doc.createElementNS("http://tempuri.org/", "tc:Zaloguj");
        zaloguj.appendChild(doc.createElementNS("http://tempuri.org/", "tc:login"));
        zaloguj.appendChild(doc.createElementNS("http://tempuri.org/", "tc:pass"));

        EditText et1 = (EditText)findViewById(R.id.editText1);
        zaloguj.getElementsByTagNameNS("http://tempuri.org/", "login").item(0).setTextContent(et1.getText().toString());
        EditText et2 = (EditText)findViewById(R.id.editText2);
        zaloguj.getElementsByTagNameNS("http://tempuri.org/", "pass").item(0).setTextContent(et2.getText().toString());

        tv1.setText("haha4");
        body.appendChild(zaloguj);
        env.appendChild(body);
        doc.appendChild(env);

        String s = null;
        s = DocumentToString(doc);

        Log.i("SOAP", s);

        StringEntity entity = null;

        entity = new StringEntity(s);
        HttpPost post = null;
        post = new HttpPost(new URI("http://www.kdkade.somee.com/oldschoolrpg_main.asmx"));
        tv1.setText("haha5");
        post.addHeader("SOAPAction", "http://tempuri.org/Zaloguj");
        post.addHeader("Content-Type", "text/xml; charset=utf-8");

        post.setEntity(entity);
        HttpResponse response = null;
        response = client.execute(post);
        Document responseDoc = null;

        responseDoc = docBuilder.parse(response.getEntity().getContent());

        tv1.setText("haha6");
        s = DocumentToString(responseDoc);

        Log.i("RESPONSE", s);

        //TextView tv1 = (TextView) findViewById(R.id.textView3);
        String[] temp1 = s.split("<ZalogujResult>");
        String[] temp2 = temp1[1].split("</");
        tv1.setText(temp2[0]);
        a = temp2[0];
        //tv1.setText(responseDoc.getElementsByTagNameNS("http://tempuri.org/","ZalogujResult").toString());
        //tv1.setText(s);
        //return null;
        //return null;
        }
        catch (Exception e){
            tv1.setText(e.getMessage());
        }
        tv1.setText("wtf");
        return a;
    }

    //protected void onProgressUpdate(Integer... progress) {
        //setProgressPercent(progress[0]);
    //}

    @Override
    protected void onPostExecute(String result) {
        TextView tv1 = (TextView) findViewById(R.id.textView3);
        tv1.setText(result);
    }

doinbackground 内のコードは、以前は oklog メソッドにあり、正常に機能していました。ご覧のとおり、TextView の textchanges をあちこちに配置して、エミュレーターでどこまで進んでいるかを確認します。「Pracuję...」(OnPreExecute のもの) だけになることもあれば、「haha5」になることもあります。ときどきアプリがクラッシュします (かなりランダムに見えます)。残念ながら、ここで何が問題なのかわかりません。どこが間違っているのか、それともエミュレータの問題なのか誰か教えてもらえますか?

4

2 に答える 2

1

バックグラウンド スレッドで UI を更新しようとしています。メインの ui スレッドで ui を更新する必要があります。doInbackground()実行が終了した直後にバックグラウンド スレッドで呼び出されますonPreExecute()。このステップは、時間がかかる可能性のあるバックグラウンド計算を実行するために使用されます。runOnUiThread()の UI を更新するために使用しdoInBackground()ます。

onProgressUpdate(Progress...)UI スレッドで呼び出されます。プログレスバーをアニメーション化したり、テキストフィールドにログを表示したりするために使用できます。これも使えます。

しかし、UI を更新することをお勧めしますonPostExecute(param)。の結果doInBackground()は のパラメータonPostExecute(param)です。onPostExecute(param)UI スレッドで呼び出されます。

また、テキストビューを一度初期化onCreate()して同じものを使用することもできます。

The 4 steps @ http://developer.android.com/reference/android/os/AsyncTask.htmlでトピックを確認できます。

runOnUiThread(new Runnable(){

@Override
public void run(){

// update ui here

}
});
于 2013-05-26T18:34:11.057 に答える
1

上記の回答で述べたように、非 UI スレッド (つまり、log_async実行中のバックグラウンド スレッド) から UI を更新しています。

doInBackground()メソッド内で進行状況を通信するためのオプションはpublishProgress()、 を使用することです。これにより、onProgressUpdate()が引き続き呼び出されます。

onProgressUpdate()UI を適切に更新するには、オーバーライドする必要もあります。

于 2013-05-26T19:06:52.403 に答える